mirror of
https://github.com/stleary/JSON-java.git
synced 2026-01-24 00:03:17 -05:00
Compare commits
135 Commits
20240205
...
pre-releas
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4cd70cf9d9 | ||
|
|
ac40a6faa3 | ||
|
|
2dcef89a6f | ||
|
|
d3c7eaf17e | ||
|
|
09536cd6c8 | ||
|
|
1f0729cadb | ||
|
|
80b2672f77 | ||
|
|
1f308db7c4 | ||
|
|
2ee5bf13f4 | ||
|
|
e7e52dad82 | ||
|
|
39ee5e092f | ||
|
|
215ec9bb9c | ||
|
|
61dc2644d8 | ||
|
|
ab1b9a3459 | ||
|
|
14e9cdc485 | ||
|
|
0d71dcf713 | ||
|
|
14f71274f7 | ||
|
|
a8ab79e3f3 | ||
|
|
783577be19 | ||
|
|
054786e300 | ||
|
|
48dfeb84b0 | ||
|
|
1ae43bdb90 | ||
|
|
cf00ef3e8a | ||
|
|
0180bd90f0 | ||
|
|
fa2f3402d6 | ||
|
|
f4944fbf1e | ||
|
|
1881cbe91a | ||
|
|
209837357b | ||
|
|
d1fd901bdb | ||
|
|
6529a7e536 | ||
|
|
4319b71934 | ||
|
|
1e3f37be98 | ||
|
|
7a8c21621c | ||
|
|
9216a19366 | ||
|
|
b6e347d7f8 | ||
|
|
879579d3bb | ||
|
|
898dd5a39d | ||
|
|
ce13ebd5fe | ||
|
|
7cc19483fb | ||
|
|
0bace72ced | ||
|
|
03def9c7fc | ||
|
|
3dcd5b2fab | ||
|
|
8983ca6da1 | ||
|
|
ce074e9f9a | ||
|
|
fe597d296e | ||
|
|
5bd4257c54 | ||
|
|
d02ac0f2a3 | ||
|
|
cfd47615d0 | ||
|
|
3200275881 | ||
|
|
d92d62afc2 | ||
|
|
46534b56ad | ||
|
|
87406e4db1 | ||
|
|
c0918c2428 | ||
|
|
d2cb38dba7 | ||
|
|
4929fc99c1 | ||
|
|
372f5caac4 | ||
|
|
0fcf352848 | ||
|
|
78cdb3d0d6 | ||
|
|
7324cc0b26 | ||
|
|
75a34a246f | ||
|
|
78151beea2 | ||
|
|
ccb4a9b95f | ||
|
|
4e39d8ccf2 | ||
|
|
d69d5e284b | ||
|
|
f1c9d0787b | ||
|
|
49de92224d | ||
|
|
d335447ab4 | ||
|
|
6c160b7d1a | ||
|
|
30dc22790c | ||
|
|
8f66865e0a | ||
|
|
45dede448c | ||
|
|
6aed1cfeb6 | ||
|
|
3672b5e158 | ||
|
|
f3b3491f4d | ||
|
|
e2fe14d951 | ||
|
|
0ff368ca07 | ||
|
|
c51efe8b08 | ||
|
|
e67abb3842 | ||
|
|
c140e91bb8 | ||
|
|
63e8314deb | ||
|
|
dcbbccc76c | ||
|
|
b75da07545 | ||
|
|
6c35b08ad6 | ||
|
|
60090a7167 | ||
|
|
0c5cf18255 | ||
|
|
5974fc1a38 | ||
|
|
a3f15e5883 | ||
|
|
045324ab42 | ||
|
|
eda08415ca | ||
|
|
48c092acfb | ||
|
|
dab29ec1d5 | ||
|
|
712859d771 | ||
|
|
05b0897f41 | ||
|
|
c010033591 | ||
|
|
5407423e43 | ||
|
|
63625b3c62 | ||
|
|
f9b5587c87 | ||
|
|
3d69990ab5 | ||
|
|
ba05e1a98c | ||
|
|
390d442054 | ||
|
|
3eb8a62af6 | ||
|
|
989cdb61bc | ||
|
|
8de0628bd1 | ||
|
|
569be9d19e | ||
|
|
3f97826462 | ||
|
|
d520210ea2 | ||
|
|
f38452a00c | ||
|
|
4f456d9432 | ||
|
|
898288810f | ||
|
|
d878c38d40 | ||
|
|
e2194bc190 | ||
|
|
d672b44a25 | ||
|
|
06778bd2d9 | ||
|
|
771c82c4eb | ||
|
|
6660e40915 | ||
|
|
0ff635c456 | ||
|
|
7c7a98da71 | ||
|
|
d36066cf82 | ||
|
|
af8cb376c2 | ||
|
|
c1107fa987 | ||
|
|
cd631d970e | ||
|
|
f0289413d6 | ||
|
|
b4b39bb441 | ||
|
|
6dc1ed0a02 | ||
|
|
86253211c2 | ||
|
|
77c899d325 | ||
|
|
b314611230 | ||
|
|
cb2c8d3962 | ||
|
|
f164b8c597 | ||
|
|
6358b7f681 | ||
|
|
21a9fae7b0 | ||
|
|
8550175556 | ||
|
|
10514e48cb | ||
|
|
72214f1b43 | ||
|
|
99c84fdf3a |
82
.github/workflows/deployment.yml
vendored
82
.github/workflows/deployment.yml
vendored
@@ -1,82 +0,0 @@
|
|||||||
# For more information see:
|
|
||||||
# * https://docs.github.com/en/actions/learn-github-actions
|
|
||||||
# * https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions
|
|
||||||
# * https://github.com/actions/setup-java/blob/v3.13.0/docs/advanced-usage.md#Publishing-using-Apache-Maven
|
|
||||||
#
|
|
||||||
|
|
||||||
name: Deployment workflow
|
|
||||||
|
|
||||||
on:
|
|
||||||
release:
|
|
||||||
types: [published]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
# old-school build and jar method. No tests run or compiled.
|
|
||||||
publish-1_6:
|
|
||||||
name: Publish Java 1.6 to GitHub Release
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
- name: Setup java
|
|
||||||
uses: actions/setup-java@v1
|
|
||||||
with:
|
|
||||||
java-version: 1.6
|
|
||||||
- name: Compile Java 1.6
|
|
||||||
run: |
|
|
||||||
mkdir -p target/classes
|
|
||||||
javac -version
|
|
||||||
javac -source 1.6 -target 1.6 -d target/classes/ src/main/java/org/json/*.java
|
|
||||||
- name: Create JAR 1.6
|
|
||||||
run: |
|
|
||||||
jar cvf "target/org.json-1.6-${{ github.ref_name }}.jar" -C target/classes .
|
|
||||||
- name: Add 1.6 Jar To Release
|
|
||||||
uses: softprops/action-gh-release@v1
|
|
||||||
with:
|
|
||||||
append_body: true
|
|
||||||
files: |
|
|
||||||
target/*.jar
|
|
||||||
publish:
|
|
||||||
name: Publish Java 8 to Maven Central and GitHub Release
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
packages: write
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
- name: Set up Java for publishing to Maven Central Repository
|
|
||||||
uses: actions/setup-java@v3
|
|
||||||
with:
|
|
||||||
# Use lowest supported LTS Java version
|
|
||||||
java-version: '8'
|
|
||||||
distribution: 'temurin'
|
|
||||||
server-id: ossrh # Value of the distributionManagement/repository/id field of the pom.xml
|
|
||||||
server-username: MAVEN_USERNAME # env variable for username in deploy
|
|
||||||
server-password: MAVEN_PASSWORD # env variable for token in deploy
|
|
||||||
gpg-private-key: ${{ secrets.MAVEN_GPG_PRIVATE_KEY }} # Value of the GPG private key to import
|
|
||||||
gpg-passphrase: MAVEN_GPG_PASSPHRASE # env variable for GPG private key passphrase
|
|
||||||
|
|
||||||
- name: Publish to the Maven Central Repository
|
|
||||||
run: mvn --batch-mode deploy
|
|
||||||
env:
|
|
||||||
MAVEN_USERNAME: ${{ secrets.OSSRH_USERNAME }}
|
|
||||||
MAVEN_PASSWORD: ${{ secrets.OSSRH_TOKEN }}
|
|
||||||
MAVEN_GPG_PASSPHRASE: ${{ secrets.MAVEN_GPG_PASSPHRASE }}
|
|
||||||
|
|
||||||
- name: Add Jar To Release
|
|
||||||
uses: softprops/action-gh-release@v1
|
|
||||||
with:
|
|
||||||
append_body: true
|
|
||||||
files: |
|
|
||||||
target/*.jar
|
|
||||||
# - name: Set up Java for publishing to GitHub Packages
|
|
||||||
# uses: actions/setup-java@v3
|
|
||||||
# with:
|
|
||||||
# # Use lowest supported LTS Java version
|
|
||||||
# java-version: '8'
|
|
||||||
# distribution: 'temurin'
|
|
||||||
# - name: Publish to GitHub Packages
|
|
||||||
# run: mvn --batch-mode deploy
|
|
||||||
# env:
|
|
||||||
# GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
151
.github/workflows/pipeline.yml
vendored
151
.github/workflows/pipeline.yml
vendored
@@ -34,14 +34,159 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
name: Create java 1.6 JAR
|
name: Create java 1.6 JAR
|
||||||
path: target/*.jar
|
path: target/*.jar
|
||||||
build:
|
|
||||||
|
build-8:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
max-parallel: 2
|
max-parallel: 1
|
||||||
matrix:
|
matrix:
|
||||||
# build against supported Java LTS versions:
|
# build against supported Java LTS versions:
|
||||||
java: [ 8, 11, 17, 21 ]
|
java: [ 8 ]
|
||||||
|
name: Java ${{ matrix.java }}
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- name: Set up JDK ${{ matrix.java }}
|
||||||
|
uses: actions/setup-java@v3
|
||||||
|
with:
|
||||||
|
distribution: 'temurin'
|
||||||
|
java-version: ${{ matrix.java }}
|
||||||
|
cache: 'maven'
|
||||||
|
- name: Compile Java ${{ matrix.java }}
|
||||||
|
run: mvn clean compile -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }} -D maven.test.skip=true -D maven.site.skip=true -D maven.javadoc.skip=true
|
||||||
|
- name: Run Tests ${{ matrix.java }}
|
||||||
|
run: |
|
||||||
|
mvn test -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }}
|
||||||
|
- name: Build Test Report ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
run: |
|
||||||
|
mvn surefire-report:report-only -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }}
|
||||||
|
mvn site -D generateReports=false -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }}
|
||||||
|
- name: Upload Test Results ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: Test Results ${{ matrix.java }}
|
||||||
|
path: target/surefire-reports/
|
||||||
|
- name: Upload Test Report ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: Test Report ${{ matrix.java }}
|
||||||
|
path: target/site/
|
||||||
|
- name: Package Jar ${{ matrix.java }}
|
||||||
|
run: mvn clean package -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }} -D maven.test.skip=true -D maven.site.skip=true
|
||||||
|
- name: Upload Package Results ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: Package Jar ${{ matrix.java }}
|
||||||
|
path: target/*.jar
|
||||||
|
|
||||||
|
build-11:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
max-parallel: 1
|
||||||
|
matrix:
|
||||||
|
# build against supported Java LTS versions:
|
||||||
|
java: [ 11 ]
|
||||||
|
name: Java ${{ matrix.java }}
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- name: Set up JDK ${{ matrix.java }}
|
||||||
|
uses: actions/setup-java@v3
|
||||||
|
with:
|
||||||
|
distribution: 'temurin'
|
||||||
|
java-version: ${{ matrix.java }}
|
||||||
|
cache: 'maven'
|
||||||
|
- name: Compile Java ${{ matrix.java }}
|
||||||
|
run: mvn clean compile -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }} -D maven.test.skip=true -D maven.site.skip=true -D maven.javadoc.skip=true
|
||||||
|
- name: Run Tests ${{ matrix.java }}
|
||||||
|
run: |
|
||||||
|
mvn test -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }}
|
||||||
|
- name: Build Test Report ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
run: |
|
||||||
|
mvn surefire-report:report-only -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }}
|
||||||
|
mvn site -D generateReports=false -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }}
|
||||||
|
- name: Upload Test Results ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: Test Results ${{ matrix.java }}
|
||||||
|
path: target/surefire-reports/
|
||||||
|
- name: Upload Test Report ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: Test Report ${{ matrix.java }}
|
||||||
|
path: target/site/
|
||||||
|
- name: Package Jar ${{ matrix.java }}
|
||||||
|
run: mvn clean package -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }} -D maven.test.skip=true -D maven.site.skip=true
|
||||||
|
- name: Upload Package Results ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: Package Jar ${{ matrix.java }}
|
||||||
|
path: target/*.jar
|
||||||
|
|
||||||
|
build-17:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
max-parallel: 1
|
||||||
|
matrix:
|
||||||
|
# build against supported Java LTS versions:
|
||||||
|
java: [ 17 ]
|
||||||
|
name: Java ${{ matrix.java }}
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- name: Set up JDK ${{ matrix.java }}
|
||||||
|
uses: actions/setup-java@v3
|
||||||
|
with:
|
||||||
|
distribution: 'temurin'
|
||||||
|
java-version: ${{ matrix.java }}
|
||||||
|
cache: 'maven'
|
||||||
|
- name: Compile Java ${{ matrix.java }}
|
||||||
|
run: mvn clean compile -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }} -D maven.test.skip=true -D maven.site.skip=true -D maven.javadoc.skip=true
|
||||||
|
- name: Run Tests ${{ matrix.java }}
|
||||||
|
run: |
|
||||||
|
mvn test -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }}
|
||||||
|
- name: Build Test Report ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
run: |
|
||||||
|
mvn surefire-report:report-only -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }}
|
||||||
|
mvn site -D generateReports=false -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }}
|
||||||
|
- name: Upload Test Results ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: Test Results ${{ matrix.java }}
|
||||||
|
path: target/surefire-reports/
|
||||||
|
- name: Upload Test Report ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: Test Report ${{ matrix.java }}
|
||||||
|
path: target/site/
|
||||||
|
- name: Package Jar ${{ matrix.java }}
|
||||||
|
run: mvn clean package -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }} -D maven.test.skip=true -D maven.site.skip=true
|
||||||
|
- name: Upload Package Results ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: Package Jar ${{ matrix.java }}
|
||||||
|
path: target/*.jar
|
||||||
|
|
||||||
|
build-21:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
max-parallel: 1
|
||||||
|
matrix:
|
||||||
|
# build against supported Java LTS versions:
|
||||||
|
java: [ 21 ]
|
||||||
name: Java ${{ matrix.java }}
|
name: Java ${{ matrix.java }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v3
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ JSON in Java [package org.json]
|
|||||||
[](https://github.com/stleary/JSON-java/actions/workflows/pipeline.yml)
|
[](https://github.com/stleary/JSON-java/actions/workflows/pipeline.yml)
|
||||||
[](https://github.com/stleary/JSON-java/actions/workflows/codeql-analysis.yml)
|
[](https://github.com/stleary/JSON-java/actions/workflows/codeql-analysis.yml)
|
||||||
|
|
||||||
**[Click here if you just want the latest release jar file.](https://search.maven.org/remotecontent?filepath=org/json/json/20240205/json-20240205.jar)**
|
**[Click here if you just want the latest release jar file.](https://search.maven.org/remotecontent?filepath=org/json/json/20241224/json-20241224.jar)**
|
||||||
|
|
||||||
|
|
||||||
# Overview
|
# Overview
|
||||||
|
|||||||
@@ -21,7 +21,7 @@ repositories {
|
|||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
testImplementation 'junit:junit:4.13.2'
|
testImplementation 'junit:junit:4.13.2'
|
||||||
testImplementation 'com.jayway.jsonpath:json-path:2.4.0'
|
testImplementation 'com.jayway.jsonpath:json-path:2.9.0'
|
||||||
testImplementation 'org.mockito:mockito-core:4.2.0'
|
testImplementation 'org.mockito:mockito-core:4.2.0'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,6 +5,10 @@ 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)
|
[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)
|
||||||
|
|
||||||
~~~
|
~~~
|
||||||
|
20241224....Strict mode opt-in feature, and recent commits.
|
||||||
|
|
||||||
|
20240303 Revert optLong/getLong changes, and recent commits.
|
||||||
|
|
||||||
20240205 Recent commits.
|
20240205 Recent commits.
|
||||||
|
|
||||||
20231013 First release with minimum Java version 1.8. Recent commits, including fixes for CVE-2023-5072.
|
20231013 First release with minimum Java version 1.8. Recent commits, including fixes for CVE-2023-5072.
|
||||||
|
|||||||
4
pom.xml
4
pom.xml
@@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
<groupId>org.json</groupId>
|
<groupId>org.json</groupId>
|
||||||
<artifactId>json</artifactId>
|
<artifactId>json</artifactId>
|
||||||
<version>20240205</version>
|
<version>20241224</version>
|
||||||
<packaging>bundle</packaging>
|
<packaging>bundle</packaging>
|
||||||
|
|
||||||
<name>JSON in Java</name>
|
<name>JSON in Java</name>
|
||||||
@@ -70,7 +70,7 @@
|
|||||||
<dependency>
|
<dependency>
|
||||||
<groupId>com.jayway.jsonpath</groupId>
|
<groupId>com.jayway.jsonpath</groupId>
|
||||||
<artifactId>json-path</artifactId>
|
<artifactId>json-path</artifactId>
|
||||||
<version>2.4.0</version>
|
<version>2.9.0</version>
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
|
|||||||
@@ -5,15 +5,15 @@ Public Domain.
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This provides static methods to convert comma delimited text into a
|
* This provides static methods to convert comma (or otherwise) delimited text into a
|
||||||
* JSONArray, and to convert a JSONArray into comma delimited text. Comma
|
* JSONArray, and to convert a JSONArray into comma (or otherwise) delimited text. Comma
|
||||||
* delimited text is a very popular format for data interchange. It is
|
* delimited text is a very popular format for data interchange. It is
|
||||||
* understood by most database, spreadsheet, and organizer programs.
|
* understood by most database, spreadsheet, and organizer programs.
|
||||||
* <p>
|
* <p>
|
||||||
* Each row of text represents a row in a table or a data record. Each row
|
* Each row of text represents a row in a table or a data record. Each row
|
||||||
* ends with a NEWLINE character. Each row contains one or more values.
|
* ends with a NEWLINE character. Each row contains one or more values.
|
||||||
* Values are separated by commas. A value can contain any character except
|
* Values are separated by commas. A value can contain any character except
|
||||||
* for comma, unless is is wrapped in single quotes or double quotes.
|
* for comma, unless it is wrapped in single quotes or double quotes.
|
||||||
* <p>
|
* <p>
|
||||||
* The first row usually contains the names of the columns.
|
* The first row usually contains the names of the columns.
|
||||||
* <p>
|
* <p>
|
||||||
@@ -25,25 +25,30 @@ Public Domain.
|
|||||||
*/
|
*/
|
||||||
public class CDL {
|
public class CDL {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new CDL object.
|
||||||
|
*/
|
||||||
|
public CDL() {
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the next value. The value can be wrapped in quotes. The value can
|
* Get the next value. The value can be wrapped in quotes. The value can
|
||||||
* be empty.
|
* be empty.
|
||||||
* @param x A JSONTokener of the source text.
|
* @param x A JSONTokener of the source text.
|
||||||
|
* @param delimiter used in the file
|
||||||
* @return The value string, or null if empty.
|
* @return The value string, or null if empty.
|
||||||
* @throws JSONException if the quoted string is badly formed.
|
* @throws JSONException if the quoted string is badly formed.
|
||||||
*/
|
*/
|
||||||
private static String getValue(JSONTokener x) throws JSONException {
|
private static String getValue(JSONTokener x, char delimiter) throws JSONException {
|
||||||
char c;
|
char c;
|
||||||
char q;
|
char q;
|
||||||
StringBuilder sb;
|
StringBuilder sb;
|
||||||
do {
|
do {
|
||||||
c = x.next();
|
c = x.next();
|
||||||
} while (c == ' ' || c == '\t');
|
} while (c == ' ' || c == '\t');
|
||||||
switch (c) {
|
if (c == 0) {
|
||||||
case 0:
|
|
||||||
return null;
|
return null;
|
||||||
case '"':
|
} else if (c == '"' || c == '\'') {
|
||||||
case '\'':
|
|
||||||
q = c;
|
q = c;
|
||||||
sb = new StringBuilder();
|
sb = new StringBuilder();
|
||||||
for (;;) {
|
for (;;) {
|
||||||
@@ -51,9 +56,9 @@ public class CDL {
|
|||||||
if (c == q) {
|
if (c == q) {
|
||||||
//Handle escaped double-quote
|
//Handle escaped double-quote
|
||||||
char nextC = x.next();
|
char nextC = x.next();
|
||||||
if(nextC != '\"') {
|
if (nextC != '\"') {
|
||||||
// if our quote was the end of the file, don't step
|
// if our quote was the end of the file, don't step
|
||||||
if(nextC > 0) {
|
if (nextC > 0) {
|
||||||
x.back();
|
x.back();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -65,13 +70,12 @@ public class CDL {
|
|||||||
sb.append(c);
|
sb.append(c);
|
||||||
}
|
}
|
||||||
return sb.toString();
|
return sb.toString();
|
||||||
case ',':
|
} else if (c == delimiter) {
|
||||||
x.back();
|
x.back();
|
||||||
return "";
|
return "";
|
||||||
default:
|
|
||||||
x.back();
|
|
||||||
return x.nextTo(',');
|
|
||||||
}
|
}
|
||||||
|
x.back();
|
||||||
|
return x.nextTo(delimiter);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -81,17 +85,28 @@ public class CDL {
|
|||||||
* @throws JSONException if a called function fails
|
* @throws JSONException if a called function fails
|
||||||
*/
|
*/
|
||||||
public static JSONArray rowToJSONArray(JSONTokener x) throws JSONException {
|
public static JSONArray rowToJSONArray(JSONTokener x) throws JSONException {
|
||||||
|
return rowToJSONArray(x, ',');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Produce a JSONArray of strings from a row of comma delimited values.
|
||||||
|
* @param x A JSONTokener of the source text.
|
||||||
|
* @param delimiter custom delimiter char
|
||||||
|
* @return A JSONArray of strings.
|
||||||
|
* @throws JSONException if a called function fails
|
||||||
|
*/
|
||||||
|
public static JSONArray rowToJSONArray(JSONTokener x, char delimiter) throws JSONException {
|
||||||
JSONArray ja = new JSONArray();
|
JSONArray ja = new JSONArray();
|
||||||
for (;;) {
|
for (;;) {
|
||||||
String value = getValue(x);
|
String value = getValue(x,delimiter);
|
||||||
char c = x.next();
|
char c = x.next();
|
||||||
if (value == null ||
|
if (value == null ||
|
||||||
(ja.length() == 0 && value.length() == 0 && c != ',')) {
|
(ja.length() == 0 && value.length() == 0 && c != delimiter)) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
ja.put(value);
|
ja.put(value);
|
||||||
for (;;) {
|
for (;;) {
|
||||||
if (c == ',') {
|
if (c == delimiter) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (c != ' ') {
|
if (c != ' ') {
|
||||||
@@ -116,9 +131,23 @@ public class CDL {
|
|||||||
* @return A JSONObject combining the names and values.
|
* @return A JSONObject combining the names and values.
|
||||||
* @throws JSONException if a called function fails
|
* @throws JSONException if a called function fails
|
||||||
*/
|
*/
|
||||||
public static JSONObject rowToJSONObject(JSONArray names, JSONTokener x)
|
public static JSONObject rowToJSONObject(JSONArray names, JSONTokener x) throws JSONException {
|
||||||
throws JSONException {
|
return rowToJSONObject(names, x, ',');
|
||||||
JSONArray ja = rowToJSONArray(x);
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Produce a JSONObject from a row of comma delimited text, using a
|
||||||
|
* parallel JSONArray of strings to provides the names of the elements.
|
||||||
|
* @param names A JSONArray of names. This is commonly obtained from the
|
||||||
|
* first row of a comma delimited text file using the rowToJSONArray
|
||||||
|
* method.
|
||||||
|
* @param x A JSONTokener of the source text.
|
||||||
|
* @param delimiter custom delimiter char
|
||||||
|
* @return A JSONObject combining the names and values.
|
||||||
|
* @throws JSONException if a called function fails
|
||||||
|
*/
|
||||||
|
public static JSONObject rowToJSONObject(JSONArray names, JSONTokener x, char delimiter) throws JSONException {
|
||||||
|
JSONArray ja = rowToJSONArray(x, delimiter);
|
||||||
return ja != null ? ja.toJSONObject(names) : null;
|
return ja != null ? ja.toJSONObject(names) : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -130,15 +159,27 @@ public class CDL {
|
|||||||
* @return A string ending in NEWLINE.
|
* @return A string ending in NEWLINE.
|
||||||
*/
|
*/
|
||||||
public static String rowToString(JSONArray ja) {
|
public static String rowToString(JSONArray ja) {
|
||||||
|
return rowToString(ja, ',');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Produce a comma delimited text row from a JSONArray. Values containing
|
||||||
|
* the comma character will be quoted. Troublesome characters may be
|
||||||
|
* removed.
|
||||||
|
* @param ja A JSONArray of strings.
|
||||||
|
* @param delimiter custom delimiter char
|
||||||
|
* @return A string ending in NEWLINE.
|
||||||
|
*/
|
||||||
|
public static String rowToString(JSONArray ja, char delimiter) {
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
for (int i = 0; i < ja.length(); i += 1) {
|
for (int i = 0; i < ja.length(); i += 1) {
|
||||||
if (i > 0) {
|
if (i > 0) {
|
||||||
sb.append(',');
|
sb.append(delimiter);
|
||||||
}
|
}
|
||||||
Object object = ja.opt(i);
|
Object object = ja.opt(i);
|
||||||
if (object != null) {
|
if (object != null) {
|
||||||
String string = object.toString();
|
String string = object.toString();
|
||||||
if (string.length() > 0 && (string.indexOf(',') >= 0 ||
|
if (string.length() > 0 && (string.indexOf(delimiter) >= 0 ||
|
||||||
string.indexOf('\n') >= 0 || string.indexOf('\r') >= 0 ||
|
string.indexOf('\n') >= 0 || string.indexOf('\r') >= 0 ||
|
||||||
string.indexOf(0) >= 0 || string.charAt(0) == '"')) {
|
string.indexOf(0) >= 0 || string.charAt(0) == '"')) {
|
||||||
sb.append('"');
|
sb.append('"');
|
||||||
@@ -167,7 +208,19 @@ public class CDL {
|
|||||||
* @throws JSONException if a called function fails
|
* @throws JSONException if a called function fails
|
||||||
*/
|
*/
|
||||||
public static JSONArray toJSONArray(String string) throws JSONException {
|
public static JSONArray toJSONArray(String string) throws JSONException {
|
||||||
return toJSONArray(new JSONTokener(string));
|
return toJSONArray(string, ',');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Produce a JSONArray of JSONObjects from a comma delimited text string,
|
||||||
|
* using the first row as a source of names.
|
||||||
|
* @param string The comma delimited text.
|
||||||
|
* @param delimiter custom delimiter char
|
||||||
|
* @return A JSONArray of JSONObjects.
|
||||||
|
* @throws JSONException if a called function fails
|
||||||
|
*/
|
||||||
|
public static JSONArray toJSONArray(String string, char delimiter) throws JSONException {
|
||||||
|
return toJSONArray(new JSONTokener(string), delimiter);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -178,7 +231,19 @@ public class CDL {
|
|||||||
* @throws JSONException if a called function fails
|
* @throws JSONException if a called function fails
|
||||||
*/
|
*/
|
||||||
public static JSONArray toJSONArray(JSONTokener x) throws JSONException {
|
public static JSONArray toJSONArray(JSONTokener x) throws JSONException {
|
||||||
return toJSONArray(rowToJSONArray(x), x);
|
return toJSONArray(x, ',');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Produce a JSONArray of JSONObjects from a comma delimited text string,
|
||||||
|
* using the first row as a source of names.
|
||||||
|
* @param x The JSONTokener containing the comma delimited text.
|
||||||
|
* @param delimiter custom delimiter char
|
||||||
|
* @return A JSONArray of JSONObjects.
|
||||||
|
* @throws JSONException if a called function fails
|
||||||
|
*/
|
||||||
|
public static JSONArray toJSONArray(JSONTokener x, char delimiter) throws JSONException {
|
||||||
|
return toJSONArray(rowToJSONArray(x, delimiter), x, delimiter);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -189,9 +254,21 @@ public class CDL {
|
|||||||
* @return A JSONArray of JSONObjects.
|
* @return A JSONArray of JSONObjects.
|
||||||
* @throws JSONException if a called function fails
|
* @throws JSONException if a called function fails
|
||||||
*/
|
*/
|
||||||
public static JSONArray toJSONArray(JSONArray names, String string)
|
public static JSONArray toJSONArray(JSONArray names, String string) throws JSONException {
|
||||||
throws JSONException {
|
return toJSONArray(names, string, ',');
|
||||||
return toJSONArray(names, new JSONTokener(string));
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Produce a JSONArray of JSONObjects from a comma delimited text string
|
||||||
|
* using a supplied JSONArray as the source of element names.
|
||||||
|
* @param names A JSONArray of strings.
|
||||||
|
* @param string The comma delimited text.
|
||||||
|
* @param delimiter custom delimiter char
|
||||||
|
* @return A JSONArray of JSONObjects.
|
||||||
|
* @throws JSONException if a called function fails
|
||||||
|
*/
|
||||||
|
public static JSONArray toJSONArray(JSONArray names, String string, char delimiter) throws JSONException {
|
||||||
|
return toJSONArray(names, new JSONTokener(string), delimiter);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -202,14 +279,26 @@ public class CDL {
|
|||||||
* @return A JSONArray of JSONObjects.
|
* @return A JSONArray of JSONObjects.
|
||||||
* @throws JSONException if a called function fails
|
* @throws JSONException if a called function fails
|
||||||
*/
|
*/
|
||||||
public static JSONArray toJSONArray(JSONArray names, JSONTokener x)
|
public static JSONArray toJSONArray(JSONArray names, JSONTokener x) throws JSONException {
|
||||||
throws JSONException {
|
return toJSONArray(names, x, ',');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Produce a JSONArray of JSONObjects from a comma delimited text string
|
||||||
|
* using a supplied JSONArray as the source of element names.
|
||||||
|
* @param names A JSONArray of strings.
|
||||||
|
* @param x A JSONTokener of the source text.
|
||||||
|
* @param delimiter custom delimiter char
|
||||||
|
* @return A JSONArray of JSONObjects.
|
||||||
|
* @throws JSONException if a called function fails
|
||||||
|
*/
|
||||||
|
public static JSONArray toJSONArray(JSONArray names, JSONTokener x, char delimiter) throws JSONException {
|
||||||
if (names == null || names.length() == 0) {
|
if (names == null || names.length() == 0) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
JSONArray ja = new JSONArray();
|
JSONArray ja = new JSONArray();
|
||||||
for (;;) {
|
for (;;) {
|
||||||
JSONObject jo = rowToJSONObject(names, x);
|
JSONObject jo = rowToJSONObject(names, x, delimiter);
|
||||||
if (jo == null) {
|
if (jo == null) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -231,11 +320,24 @@ public class CDL {
|
|||||||
* @throws JSONException if a called function fails
|
* @throws JSONException if a called function fails
|
||||||
*/
|
*/
|
||||||
public static String toString(JSONArray ja) throws JSONException {
|
public static String toString(JSONArray ja) throws JSONException {
|
||||||
|
return toString(ja, ',');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Produce a comma delimited text from a JSONArray of JSONObjects. The
|
||||||
|
* first row will be a list of names obtained by inspecting the first
|
||||||
|
* JSONObject.
|
||||||
|
* @param ja A JSONArray of JSONObjects.
|
||||||
|
* @param delimiter custom delimiter char
|
||||||
|
* @return A comma delimited text.
|
||||||
|
* @throws JSONException if a called function fails
|
||||||
|
*/
|
||||||
|
public static String toString(JSONArray ja, char delimiter) throws JSONException {
|
||||||
JSONObject jo = ja.optJSONObject(0);
|
JSONObject jo = ja.optJSONObject(0);
|
||||||
if (jo != null) {
|
if (jo != null) {
|
||||||
JSONArray names = jo.names();
|
JSONArray names = jo.names();
|
||||||
if (names != null) {
|
if (names != null) {
|
||||||
return rowToString(names) + toString(names, ja);
|
return rowToString(names, delimiter) + toString(names, ja, delimiter);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
@@ -250,8 +352,21 @@ public class CDL {
|
|||||||
* @return A comma delimited text.
|
* @return A comma delimited text.
|
||||||
* @throws JSONException if a called function fails
|
* @throws JSONException if a called function fails
|
||||||
*/
|
*/
|
||||||
public static String toString(JSONArray names, JSONArray ja)
|
public static String toString(JSONArray names, JSONArray ja) throws JSONException {
|
||||||
throws JSONException {
|
return toString(names, ja, ',');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Produce a comma delimited text from a JSONArray of JSONObjects using
|
||||||
|
* a provided list of names. The list of names is not included in the
|
||||||
|
* output.
|
||||||
|
* @param names A JSONArray of strings.
|
||||||
|
* @param ja A JSONArray of JSONObjects.
|
||||||
|
* @param delimiter custom delimiter char
|
||||||
|
* @return A comma delimited text.
|
||||||
|
* @throws JSONException if a called function fails
|
||||||
|
*/
|
||||||
|
public static String toString(JSONArray names, JSONArray ja, char delimiter) throws JSONException {
|
||||||
if (names == null || names.length() == 0) {
|
if (names == null || names.length() == 0) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@@ -259,7 +374,7 @@ public class CDL {
|
|||||||
for (int i = 0; i < ja.length(); i += 1) {
|
for (int i = 0; i < ja.length(); i += 1) {
|
||||||
JSONObject jo = ja.optJSONObject(i);
|
JSONObject jo = ja.optJSONObject(i);
|
||||||
if (jo != null) {
|
if (jo != null) {
|
||||||
sb.append(rowToString(jo.toJSONArray(names)));
|
sb.append(rowToString(jo.toJSONArray(names), delimiter));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return sb.toString();
|
return sb.toString();
|
||||||
|
|||||||
@@ -15,6 +15,12 @@ Public Domain.
|
|||||||
*/
|
*/
|
||||||
public class Cookie {
|
public class Cookie {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new Cookie object.
|
||||||
|
*/
|
||||||
|
public Cookie() {
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Produce a copy of a string in which the characters '+', '%', '=', ';'
|
* Produce a copy of a string in which the characters '+', '%', '=', ';'
|
||||||
* and control characters are replaced with "%hh". This is a gentle form
|
* and control characters are replaced with "%hh". This is a gentle form
|
||||||
|
|||||||
@@ -11,6 +11,12 @@ Public Domain.
|
|||||||
*/
|
*/
|
||||||
public class CookieList {
|
public class CookieList {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new CookieList object.
|
||||||
|
*/
|
||||||
|
public CookieList() {
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert a cookie list into a JSONObject. A cookie list is a sequence
|
* Convert a cookie list into a JSONObject. A cookie list is a sequence
|
||||||
* of name/value pairs. The names are separated from the values by '='.
|
* of name/value pairs. The names are separated from the values by '='.
|
||||||
|
|||||||
@@ -13,6 +13,12 @@ import java.util.Locale;
|
|||||||
*/
|
*/
|
||||||
public class HTTP {
|
public class HTTP {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new HTTP object.
|
||||||
|
*/
|
||||||
|
public HTTP() {
|
||||||
|
}
|
||||||
|
|
||||||
/** Carriage return/line feed. */
|
/** Carriage return/line feed. */
|
||||||
public static final String CRLF = "\r\n";
|
public static final String CRLF = "\r\n";
|
||||||
|
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ Public Domain.
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.StringWriter;
|
|
||||||
import java.io.Writer;
|
import java.io.Writer;
|
||||||
import java.lang.reflect.Array;
|
import java.lang.reflect.Array;
|
||||||
import java.math.BigDecimal;
|
import java.math.BigDecimal;
|
||||||
@@ -68,6 +67,12 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
*/
|
*/
|
||||||
private final ArrayList<Object> myArrayList;
|
private final ArrayList<Object> myArrayList;
|
||||||
|
|
||||||
|
// strict mode checks after constructor require access to this object
|
||||||
|
private JSONTokener jsonTokener;
|
||||||
|
|
||||||
|
// strict mode checks after constructor require access to this object
|
||||||
|
private JSONParserConfiguration jsonParserConfiguration;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct an empty JSONArray.
|
* Construct an empty JSONArray.
|
||||||
*/
|
*/
|
||||||
@@ -84,7 +89,27 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
* If there is a syntax error.
|
* If there is a syntax error.
|
||||||
*/
|
*/
|
||||||
public JSONArray(JSONTokener x) throws JSONException {
|
public JSONArray(JSONTokener x) throws JSONException {
|
||||||
|
this(x, new JSONParserConfiguration());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a JSONArray from a JSONTokener and a JSONParserConfiguration.
|
||||||
|
*
|
||||||
|
* @param x A JSONTokener instance from which the JSONArray is constructed.
|
||||||
|
* @param jsonParserConfiguration A JSONParserConfiguration instance that controls the behavior of the parser.
|
||||||
|
* @throws JSONException If a syntax error occurs during the construction of the JSONArray.
|
||||||
|
*/
|
||||||
|
public JSONArray(JSONTokener x, JSONParserConfiguration jsonParserConfiguration) throws JSONException {
|
||||||
this();
|
this();
|
||||||
|
|
||||||
|
if (this.jsonParserConfiguration == null) {
|
||||||
|
this.jsonParserConfiguration = jsonParserConfiguration;
|
||||||
|
}
|
||||||
|
if (this.jsonTokener == null) {
|
||||||
|
this.jsonTokener = x;
|
||||||
|
this.jsonTokener.setJsonParserConfiguration(this.jsonParserConfiguration);
|
||||||
|
}
|
||||||
|
|
||||||
if (x.nextClean() != '[') {
|
if (x.nextClean() != '[') {
|
||||||
throw x.syntaxError("A JSONArray text must start with '['");
|
throw x.syntaxError("A JSONArray text must start with '['");
|
||||||
}
|
}
|
||||||
@@ -115,6 +140,17 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
throw x.syntaxError("Expected a ',' or ']'");
|
throw x.syntaxError("Expected a ',' or ']'");
|
||||||
}
|
}
|
||||||
if (nextChar == ']') {
|
if (nextChar == ']') {
|
||||||
|
// trailing commas are not allowed in strict mode
|
||||||
|
if (jsonParserConfiguration.isStrictMode()) {
|
||||||
|
throw x.syntaxError("Strict mode error: Expected another array element");
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (nextChar == ',') {
|
||||||
|
// consecutive commas are not allowed in strict mode
|
||||||
|
if (jsonParserConfiguration.isStrictMode()) {
|
||||||
|
throw x.syntaxError("Strict mode error: Expected a valid array element");
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
x.back();
|
x.back();
|
||||||
@@ -139,7 +175,32 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
* If there is a syntax error.
|
* If there is a syntax error.
|
||||||
*/
|
*/
|
||||||
public JSONArray(String source) throws JSONException {
|
public JSONArray(String source) throws JSONException {
|
||||||
this(new JSONTokener(source));
|
this(source, new JSONParserConfiguration());
|
||||||
|
// Strict mode does not allow trailing chars
|
||||||
|
if (this.jsonParserConfiguration.isStrictMode() &&
|
||||||
|
this.jsonTokener.nextClean() != 0) {
|
||||||
|
throw jsonTokener.syntaxError("Strict mode error: Unparsed characters found at end of input text");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a JSONArray from a source JSON text.
|
||||||
|
*
|
||||||
|
* @param source
|
||||||
|
* A string that begins with <code>[</code> <small>(left
|
||||||
|
* bracket)</small> and ends with <code>]</code>
|
||||||
|
* <small>(right bracket)</small>.
|
||||||
|
* @param jsonParserConfiguration the parser config object
|
||||||
|
* @throws JSONException
|
||||||
|
* If there is a syntax error.
|
||||||
|
*/
|
||||||
|
public JSONArray(String source, JSONParserConfiguration jsonParserConfiguration) throws JSONException {
|
||||||
|
this(new JSONTokener(source), jsonParserConfiguration);
|
||||||
|
// Strict mode does not allow trailing chars
|
||||||
|
if (this.jsonParserConfiguration.isStrictMode() &&
|
||||||
|
this.jsonTokener.nextClean() != 0) {
|
||||||
|
throw jsonTokener.syntaxError("Strict mode error: Unparsed characters found at end of input text");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -360,7 +421,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
if (object instanceof Number) {
|
if (object instanceof Number) {
|
||||||
return (Number)object;
|
return (Number)object;
|
||||||
}
|
}
|
||||||
return NumberConversionUtil.stringToNumber(object.toString());
|
return JSONObject.stringToNumber(object.toString());
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw wrongValueFormatException(index, "number", object, e);
|
throw wrongValueFormatException(index, "number", object, e);
|
||||||
}
|
}
|
||||||
@@ -1107,7 +1168,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
|
|
||||||
if (val instanceof String) {
|
if (val instanceof String) {
|
||||||
try {
|
try {
|
||||||
return NumberConversionUtil.stringToNumber((String) val);
|
return JSONObject.stringToNumber((String) val);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
return defaultValue;
|
return defaultValue;
|
||||||
}
|
}
|
||||||
@@ -1695,7 +1756,10 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
*/
|
*/
|
||||||
@SuppressWarnings("resource")
|
@SuppressWarnings("resource")
|
||||||
public String toString(int indentFactor) throws JSONException {
|
public String toString(int indentFactor) throws JSONException {
|
||||||
StringWriter sw = new StringWriter();
|
// each value requires a comma, so multiply the count by 2
|
||||||
|
// We don't want to oversize the initial capacity
|
||||||
|
int initialSize = myArrayList.size() * 2;
|
||||||
|
Writer sw = new StringBuilderWriter(Math.max(initialSize, 16));
|
||||||
return this.write(sw, indentFactor, 0).toString();
|
return this.write(sw, indentFactor, 0).toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1937,7 +2001,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
// JSONArray
|
// JSONArray
|
||||||
this.myArrayList.addAll(((JSONArray)array).myArrayList);
|
this.myArrayList.addAll(((JSONArray)array).myArrayList);
|
||||||
} else if (array instanceof Collection) {
|
} else if (array instanceof Collection) {
|
||||||
this.addAll((Collection<?>)array, wrap, recursionDepth);
|
this.addAll((Collection<?>)array, wrap, recursionDepth, jsonParserConfiguration);
|
||||||
} else if (array instanceof Iterable) {
|
} else if (array instanceof Iterable) {
|
||||||
this.addAll((Iterable<?>)array, wrap);
|
this.addAll((Iterable<?>)array, wrap);
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -13,6 +13,13 @@ Public Domain.
|
|||||||
* @version 2016-01-30
|
* @version 2016-01-30
|
||||||
*/
|
*/
|
||||||
public class JSONML {
|
public class JSONML {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new JSONML object.
|
||||||
|
*/
|
||||||
|
public JSONML() {
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Parse XML values and store them in a JSONArray.
|
* Parse XML values and store them in a JSONArray.
|
||||||
* @param x The XMLTokener containing the source string.
|
* @param x The XMLTokener containing the source string.
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ Public Domain.
|
|||||||
|
|
||||||
import java.io.Closeable;
|
import java.io.Closeable;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.StringWriter;
|
|
||||||
import java.io.Writer;
|
import java.io.Writer;
|
||||||
import java.lang.annotation.Annotation;
|
import java.lang.annotation.Annotation;
|
||||||
import java.lang.reflect.Field;
|
import java.lang.reflect.Field;
|
||||||
@@ -15,22 +14,10 @@ import java.lang.reflect.Method;
|
|||||||
import java.lang.reflect.Modifier;
|
import java.lang.reflect.Modifier;
|
||||||
import java.math.BigDecimal;
|
import java.math.BigDecimal;
|
||||||
import java.math.BigInteger;
|
import java.math.BigInteger;
|
||||||
import java.util.Collection;
|
import java.util.*;
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.Enumeration;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.IdentityHashMap;
|
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.Locale;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Map.Entry;
|
import java.util.Map.Entry;
|
||||||
import java.util.ResourceBundle;
|
|
||||||
import java.util.Set;
|
|
||||||
import java.util.regex.Pattern;
|
import java.util.regex.Pattern;
|
||||||
|
|
||||||
import static org.json.NumberConversionUtil.potentialNumber;
|
|
||||||
import static org.json.NumberConversionUtil.stringToNumber;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A JSONObject is an unordered collection of name/value pairs. Its external
|
* A JSONObject is an unordered collection of name/value pairs. Its external
|
||||||
* form is a string wrapped in curly braces with colons between the names and
|
* form is a string wrapped in curly braces with colons between the names and
|
||||||
@@ -148,6 +135,11 @@ public class JSONObject {
|
|||||||
*/
|
*/
|
||||||
private final Map<String, Object> map;
|
private final Map<String, Object> map;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the type of the underlying Map in this class.
|
||||||
|
*
|
||||||
|
* @return The class object representing the type of the underlying Map.
|
||||||
|
*/
|
||||||
public Class<? extends Map> getMapType() {
|
public Class<? extends Map> getMapType() {
|
||||||
return map.getClass();
|
return map.getClass();
|
||||||
}
|
}
|
||||||
@@ -160,6 +152,12 @@ public class JSONObject {
|
|||||||
*/
|
*/
|
||||||
public static final Object NULL = new Null();
|
public static final Object NULL = new Null();
|
||||||
|
|
||||||
|
// strict mode checks after constructor require access to this object
|
||||||
|
private JSONTokener jsonTokener;
|
||||||
|
|
||||||
|
// strict mode checks after constructor require access to this object
|
||||||
|
private JSONParserConfiguration jsonParserConfiguration;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct an empty JSONObject.
|
* Construct an empty JSONObject.
|
||||||
*/
|
*/
|
||||||
@@ -203,7 +201,31 @@ public class JSONObject {
|
|||||||
* duplicated key.
|
* duplicated key.
|
||||||
*/
|
*/
|
||||||
public JSONObject(JSONTokener x) throws JSONException {
|
public JSONObject(JSONTokener x) throws JSONException {
|
||||||
|
this(x, new JSONParserConfiguration());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a JSONObject from a JSONTokener with custom json parse configurations.
|
||||||
|
*
|
||||||
|
* @param x
|
||||||
|
* A JSONTokener object containing the source string.
|
||||||
|
* @param jsonParserConfiguration
|
||||||
|
* Variable to pass parser custom configuration for json parsing.
|
||||||
|
* @throws JSONException
|
||||||
|
* If there is a syntax error in the source string or a
|
||||||
|
* duplicated key.
|
||||||
|
*/
|
||||||
|
public JSONObject(JSONTokener x, JSONParserConfiguration jsonParserConfiguration) throws JSONException {
|
||||||
this();
|
this();
|
||||||
|
|
||||||
|
if (this.jsonParserConfiguration == null) {
|
||||||
|
this.jsonParserConfiguration = jsonParserConfiguration;
|
||||||
|
}
|
||||||
|
if (this.jsonTokener == null) {
|
||||||
|
this.jsonTokener = x;
|
||||||
|
this.jsonTokener.setJsonParserConfiguration(this.jsonParserConfiguration);
|
||||||
|
}
|
||||||
|
|
||||||
char c;
|
char c;
|
||||||
String key;
|
String key;
|
||||||
|
|
||||||
@@ -232,13 +254,14 @@ public class JSONObject {
|
|||||||
|
|
||||||
if (key != null) {
|
if (key != null) {
|
||||||
// Check if key exists
|
// Check if key exists
|
||||||
if (this.opt(key) != null) {
|
boolean keyExists = this.opt(key) != null;
|
||||||
// key already exists
|
if (keyExists && !jsonParserConfiguration.isOverwriteDuplicateKey()) {
|
||||||
throw x.syntaxError("Duplicate key \"" + key + "\"");
|
throw x.syntaxError("Duplicate key \"" + key + "\"");
|
||||||
}
|
}
|
||||||
// Only add value if non-null
|
|
||||||
Object value = x.nextValue();
|
Object value = x.nextValue();
|
||||||
if (value!=null) {
|
// Only add value if non-null
|
||||||
|
if (value != null) {
|
||||||
this.put(key, value);
|
this.put(key, value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -247,8 +270,16 @@ public class JSONObject {
|
|||||||
|
|
||||||
switch (x.nextClean()) {
|
switch (x.nextClean()) {
|
||||||
case ';':
|
case ';':
|
||||||
|
// In strict mode semicolon is not a valid separator
|
||||||
|
if (jsonParserConfiguration.isStrictMode()) {
|
||||||
|
throw x.syntaxError("Strict mode error: Invalid character ';' found");
|
||||||
|
}
|
||||||
case ',':
|
case ',':
|
||||||
if (x.nextClean() == '}') {
|
if (x.nextClean() == '}') {
|
||||||
|
// trailing commas are not allowed in strict mode
|
||||||
|
if (jsonParserConfiguration.isStrictMode()) {
|
||||||
|
throw x.syntaxError("Strict mode error: Expected another object element");
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (x.end()) {
|
if (x.end()) {
|
||||||
@@ -294,7 +325,6 @@ public class JSONObject {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct a JSONObject from a map with recursion depth.
|
* Construct a JSONObject from a map with recursion depth.
|
||||||
*
|
|
||||||
*/
|
*/
|
||||||
private JSONObject(Map<?, ?> m, int recursionDepth, JSONParserConfiguration jsonParserConfiguration) {
|
private JSONObject(Map<?, ?> m, int recursionDepth, JSONParserConfiguration jsonParserConfiguration) {
|
||||||
if (recursionDepth > jsonParserConfiguration.getMaxNestingDepth()) {
|
if (recursionDepth > jsonParserConfiguration.getMaxNestingDepth()) {
|
||||||
@@ -369,7 +399,6 @@ public class JSONObject {
|
|||||||
* @JSONPropertyIgnore
|
* @JSONPropertyIgnore
|
||||||
* public String getName() { return this.name; }
|
* public String getName() { return this.name; }
|
||||||
* </pre>
|
* </pre>
|
||||||
* <p>
|
|
||||||
*
|
*
|
||||||
* @param bean
|
* @param bean
|
||||||
* An object that has getter methods that should be used to make
|
* An object that has getter methods that should be used to make
|
||||||
@@ -426,7 +455,35 @@ public class JSONObject {
|
|||||||
* duplicated key.
|
* duplicated key.
|
||||||
*/
|
*/
|
||||||
public JSONObject(String source) throws JSONException {
|
public JSONObject(String source) throws JSONException {
|
||||||
this(new JSONTokener(source));
|
this(source, new JSONParserConfiguration());
|
||||||
|
// Strict mode does not allow trailing chars
|
||||||
|
if (this.jsonParserConfiguration.isStrictMode() &&
|
||||||
|
this.jsonTokener.nextClean() != 0) {
|
||||||
|
throw new JSONException("Strict mode error: Unparsed characters found at end of input text");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a JSONObject from a source JSON text string with custom json parse configurations.
|
||||||
|
* This is the most commonly used JSONObject constructor.
|
||||||
|
*
|
||||||
|
* @param source
|
||||||
|
* A string beginning with <code>{</code> <small>(left
|
||||||
|
* brace)</small> and ending with <code>}</code>
|
||||||
|
* <small>(right brace)</small>.
|
||||||
|
* @param jsonParserConfiguration
|
||||||
|
* Variable to pass parser custom configuration for json parsing.
|
||||||
|
* @exception JSONException
|
||||||
|
* If there is a syntax error in the source string or a
|
||||||
|
* duplicated key.
|
||||||
|
*/
|
||||||
|
public JSONObject(String source, JSONParserConfiguration jsonParserConfiguration) throws JSONException {
|
||||||
|
this(new JSONTokener(source), jsonParserConfiguration);
|
||||||
|
// Strict mode does not allow trailing chars
|
||||||
|
if (this.jsonParserConfiguration.isStrictMode() &&
|
||||||
|
this.jsonTokener.nextClean() != 0) {
|
||||||
|
throw new JSONException("Strict mode error: Unparsed characters found at end of input text");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -2223,7 +2280,10 @@ public class JSONObject {
|
|||||||
*/
|
*/
|
||||||
@SuppressWarnings("resource")
|
@SuppressWarnings("resource")
|
||||||
public static String quote(String string) {
|
public static String quote(String string) {
|
||||||
StringWriter sw = new StringWriter();
|
if (string == null || string.isEmpty()) {
|
||||||
|
return "\"\"";
|
||||||
|
}
|
||||||
|
Writer sw = new StringBuilderWriter(string.length() + 2);
|
||||||
try {
|
try {
|
||||||
return quote(string, sw).toString();
|
return quote(string, sw).toString();
|
||||||
} catch (IOException ignored) {
|
} catch (IOException ignored) {
|
||||||
@@ -2232,6 +2292,14 @@ public class JSONObject {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Quotes a string and appends the result to a given Writer.
|
||||||
|
*
|
||||||
|
* @param string The input string to be quoted.
|
||||||
|
* @param w The Writer to which the quoted string will be appended.
|
||||||
|
* @return The same Writer instance after appending the quoted string.
|
||||||
|
* @throws IOException If an I/O error occurs while writing to the Writer.
|
||||||
|
*/
|
||||||
public static Writer quote(String string, Writer w) throws IOException {
|
public static Writer quote(String string, Writer w) throws IOException {
|
||||||
if (string == null || string.isEmpty()) {
|
if (string == null || string.isEmpty()) {
|
||||||
w.write("\"\"");
|
w.write("\"\"");
|
||||||
@@ -2448,7 +2516,8 @@ public class JSONObject {
|
|||||||
* produced, then the value will just be a string.
|
* produced, then the value will just be a string.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
if (potentialNumber(string)) {
|
char initial = string.charAt(0);
|
||||||
|
if ((initial >= '0' && initial <= '9') || initial == '-') {
|
||||||
try {
|
try {
|
||||||
return stringToNumber(string);
|
return stringToNumber(string);
|
||||||
} catch (Exception ignore) {
|
} catch (Exception ignore) {
|
||||||
@@ -2457,8 +2526,75 @@ public class JSONObject {
|
|||||||
return string;
|
return string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts a string to a number using the narrowest possible type. Possible
|
||||||
|
* returns for this function are BigDecimal, Double, BigInteger, Long, and Integer.
|
||||||
|
* When a Double is returned, it should always be a valid Double and not NaN or +-infinity.
|
||||||
|
*
|
||||||
|
* @param val value to convert
|
||||||
|
* @return Number representation of the value.
|
||||||
|
* @throws NumberFormatException thrown if the value is not a valid number. A public
|
||||||
|
* caller should catch this and wrap it in a {@link JSONException} if applicable.
|
||||||
|
*/
|
||||||
|
protected static Number stringToNumber(final String val) throws NumberFormatException {
|
||||||
|
char initial = val.charAt(0);
|
||||||
|
if ((initial >= '0' && initial <= '9') || initial == '-') {
|
||||||
|
// decimal representation
|
||||||
|
if (isDecimalNotation(val)) {
|
||||||
|
// Use a BigDecimal all the time so we keep the original
|
||||||
|
// representation. BigDecimal doesn't support -0.0, ensure we
|
||||||
|
// keep that by forcing a decimal.
|
||||||
|
try {
|
||||||
|
BigDecimal bd = new BigDecimal(val);
|
||||||
|
if(initial == '-' && BigDecimal.ZERO.compareTo(bd)==0) {
|
||||||
|
return Double.valueOf(-0.0);
|
||||||
|
}
|
||||||
|
return bd;
|
||||||
|
} catch (NumberFormatException retryAsDouble) {
|
||||||
|
// this is to support "Hex Floats" like this: 0x1.0P-1074
|
||||||
|
try {
|
||||||
|
Double d = Double.valueOf(val);
|
||||||
|
if(d.isNaN() || d.isInfinite()) {
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
return d;
|
||||||
|
} catch (NumberFormatException ignore) {
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// block items like 00 01 etc. Java number parsers treat these as Octal.
|
||||||
|
if(initial == '0' && val.length() > 1) {
|
||||||
|
char at1 = val.charAt(1);
|
||||||
|
if(at1 >= '0' && at1 <= '9') {
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
} else if (initial == '-' && val.length() > 2) {
|
||||||
|
char at1 = val.charAt(1);
|
||||||
|
char at2 = val.charAt(2);
|
||||||
|
if(at1 == '0' && at2 >= '0' && at2 <= '9') {
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// integer representation.
|
||||||
|
// This will narrow any values to the smallest reasonable Object representation
|
||||||
|
// (Integer, Long, or BigInteger)
|
||||||
|
|
||||||
|
// BigInteger down conversion: We use a similar bitLength compare as
|
||||||
|
// BigInteger#intValueExact uses. Increases GC, but objects hold
|
||||||
|
// only what they need. i.e. Less runtime overhead if the value is
|
||||||
|
// long lived.
|
||||||
|
BigInteger bi = new BigInteger(val);
|
||||||
|
if(bi.bitLength() <= 31){
|
||||||
|
return Integer.valueOf(bi.intValue());
|
||||||
|
}
|
||||||
|
if(bi.bitLength() <= 63){
|
||||||
|
return Long.valueOf(bi.longValue());
|
||||||
|
}
|
||||||
|
return bi;
|
||||||
|
}
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Throw an exception if the object is a NaN or infinite number.
|
* Throw an exception if the object is a NaN or infinite number.
|
||||||
@@ -2546,7 +2682,10 @@ public class JSONObject {
|
|||||||
*/
|
*/
|
||||||
@SuppressWarnings("resource")
|
@SuppressWarnings("resource")
|
||||||
public String toString(int indentFactor) throws JSONException {
|
public String toString(int indentFactor) throws JSONException {
|
||||||
StringWriter w = new StringWriter();
|
// 6 characters are the minimum to serialise a key value pair e.g.: "k":1,
|
||||||
|
// and we don't want to oversize the initial capacity
|
||||||
|
int initialSize = map.size() * 6;
|
||||||
|
Writer w = new StringBuilderWriter(Math.max(initialSize, 16));
|
||||||
return this.write(w, indentFactor, 0).toString();
|
return this.write(w, indentFactor, 0).toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -2689,6 +2828,7 @@ public class JSONObject {
|
|||||||
if (value == null || value.equals(null)) {
|
if (value == null || value.equals(null)) {
|
||||||
writer.write("null");
|
writer.write("null");
|
||||||
} else if (value instanceof JSONString) {
|
} else if (value instanceof JSONString) {
|
||||||
|
// JSONString must be checked first, so it can overwrite behaviour of other types below
|
||||||
Object o;
|
Object o;
|
||||||
try {
|
try {
|
||||||
o = ((JSONString) value).toJSONString();
|
o = ((JSONString) value).toJSONString();
|
||||||
@@ -2696,6 +2836,10 @@ public class JSONObject {
|
|||||||
throw new JSONException(e);
|
throw new JSONException(e);
|
||||||
}
|
}
|
||||||
writer.write(o != null ? o.toString() : quote(value.toString()));
|
writer.write(o != null ? o.toString() : quote(value.toString()));
|
||||||
|
} else if (value instanceof String) {
|
||||||
|
// assuming most values are Strings, so testing it early
|
||||||
|
quote(value.toString(), writer);
|
||||||
|
return writer;
|
||||||
} else if (value instanceof Number) {
|
} else if (value instanceof Number) {
|
||||||
// not all Numbers may match actual JSON Numbers. i.e. fractions or Imaginary
|
// not all Numbers may match actual JSON Numbers. i.e. fractions or Imaginary
|
||||||
final String numberAsString = numberToString((Number) value);
|
final String numberAsString = numberToString((Number) value);
|
||||||
@@ -2884,5 +3028,23 @@ public class JSONObject {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* For a prospective number, remove the leading zeros
|
||||||
|
* @param value prospective number
|
||||||
|
* @return number without leading zeros
|
||||||
|
*/
|
||||||
|
private static String removeLeadingZerosOfNumber(String value){
|
||||||
|
if (value.equals("-")){return value;}
|
||||||
|
boolean negativeFirstChar = (value.charAt(0) == '-');
|
||||||
|
int counter = negativeFirstChar ? 1:0;
|
||||||
|
while (counter < value.length()){
|
||||||
|
if (value.charAt(counter) != '0'){
|
||||||
|
if (negativeFirstChar) {return "-".concat(value.substring(counter));}
|
||||||
|
return value.substring(counter);
|
||||||
|
}
|
||||||
|
++counter;
|
||||||
|
}
|
||||||
|
if (negativeFirstChar) {return "-0";}
|
||||||
|
return "0";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,23 +4,109 @@ package org.json;
|
|||||||
* Configuration object for the JSON parser. The configuration is immutable.
|
* Configuration object for the JSON parser. The configuration is immutable.
|
||||||
*/
|
*/
|
||||||
public class JSONParserConfiguration extends ParserConfiguration {
|
public class JSONParserConfiguration extends ParserConfiguration {
|
||||||
|
/**
|
||||||
|
* Used to indicate whether to overwrite duplicate key or not.
|
||||||
|
*/
|
||||||
|
private boolean overwriteDuplicateKey;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Configuration with the default values.
|
* Configuration with the default values.
|
||||||
*/
|
*/
|
||||||
public JSONParserConfiguration() {
|
public JSONParserConfiguration() {
|
||||||
super();
|
super();
|
||||||
|
this.overwriteDuplicateKey = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This flag, when set to true, instructs the parser to enforce strict mode when parsing JSON text.
|
||||||
|
* Garbage chars at the end of the doc, unquoted string, and single-quoted strings are all disallowed.
|
||||||
|
*/
|
||||||
|
private boolean strictMode;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected JSONParserConfiguration clone() {
|
protected JSONParserConfiguration clone() {
|
||||||
return new JSONParserConfiguration();
|
JSONParserConfiguration clone = new JSONParserConfiguration();
|
||||||
|
clone.overwriteDuplicateKey = overwriteDuplicateKey;
|
||||||
|
clone.maxNestingDepth = maxNestingDepth;
|
||||||
|
return clone;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Defines the maximum nesting depth that the parser will descend before throwing an exception
|
||||||
|
* when parsing a map into JSONObject or parsing a {@link java.util.Collection} instance into
|
||||||
|
* JSONArray. The default max nesting depth is 512, which means the parser will throw a JsonException
|
||||||
|
* if the maximum depth is reached.
|
||||||
|
*
|
||||||
|
* @param maxNestingDepth the maximum nesting depth allowed to the JSON parser
|
||||||
|
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||||
|
*/
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
@Override
|
@Override
|
||||||
public JSONParserConfiguration withMaxNestingDepth(final int maxNestingDepth) {
|
public JSONParserConfiguration withMaxNestingDepth(final int maxNestingDepth) {
|
||||||
return super.withMaxNestingDepth(maxNestingDepth);
|
JSONParserConfiguration clone = this.clone();
|
||||||
|
clone.maxNestingDepth = maxNestingDepth;
|
||||||
|
|
||||||
|
return clone;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Controls the parser's behavior when meeting duplicate keys.
|
||||||
|
* If set to false, the parser will throw a JSONException when meeting a duplicate key.
|
||||||
|
* Or the duplicate key's value will be overwritten.
|
||||||
|
*
|
||||||
|
* @param overwriteDuplicateKey defines should the parser overwrite duplicate keys.
|
||||||
|
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||||
|
*/
|
||||||
|
public JSONParserConfiguration withOverwriteDuplicateKey(final boolean overwriteDuplicateKey) {
|
||||||
|
JSONParserConfiguration clone = this.clone();
|
||||||
|
clone.overwriteDuplicateKey = overwriteDuplicateKey;
|
||||||
|
|
||||||
|
return clone;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the strict mode configuration for the JSON parser with default true value
|
||||||
|
* <p>
|
||||||
|
* When strict mode is enabled, the parser will throw a JSONException if it encounters an invalid character
|
||||||
|
* immediately following the final ']' character in the input. This is useful for ensuring strict adherence to the
|
||||||
|
* JSON syntax, as any characters after the final closing bracket of a JSON array are considered invalid.
|
||||||
|
* @return a new JSONParserConfiguration instance with the updated strict mode setting
|
||||||
|
*/
|
||||||
|
public JSONParserConfiguration withStrictMode() {
|
||||||
|
return withStrictMode(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the strict mode configuration for the JSON parser.
|
||||||
|
* <p>
|
||||||
|
* When strict mode is enabled, the parser will throw a JSONException if it encounters an invalid character
|
||||||
|
* immediately following the final ']' character in the input. This is useful for ensuring strict adherence to the
|
||||||
|
* JSON syntax, as any characters after the final closing bracket of a JSON array are considered invalid.
|
||||||
|
*
|
||||||
|
* @param mode a boolean value indicating whether strict mode should be enabled or not
|
||||||
|
* @return a new JSONParserConfiguration instance with the updated strict mode setting
|
||||||
|
*/
|
||||||
|
public JSONParserConfiguration withStrictMode(final boolean mode) {
|
||||||
|
JSONParserConfiguration clone = this.clone();
|
||||||
|
clone.strictMode = mode;
|
||||||
|
|
||||||
|
return clone;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The parser's behavior when meeting duplicate keys, controls whether the parser should
|
||||||
|
* overwrite duplicate keys or not.
|
||||||
|
*
|
||||||
|
* @return The <code>overwriteDuplicateKey</code> configuration value.
|
||||||
|
*/
|
||||||
|
public boolean isOverwriteDuplicateKey() {
|
||||||
|
return this.overwriteDuplicateKey;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the current strict mode setting.
|
||||||
|
*/
|
||||||
|
public boolean isStrictMode() {
|
||||||
|
return this.strictMode;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -42,6 +42,12 @@ public class JSONPointer {
|
|||||||
*/
|
*/
|
||||||
public static class Builder {
|
public static class Builder {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new Builder object.
|
||||||
|
*/
|
||||||
|
public Builder() {
|
||||||
|
}
|
||||||
|
|
||||||
// Segments for the eventual JSONPointer string
|
// Segments for the eventual JSONPointer string
|
||||||
private final List<String> refTokens = new ArrayList<String>();
|
private final List<String> refTokens = new ArrayList<String>();
|
||||||
|
|
||||||
@@ -163,6 +169,12 @@ public class JSONPointer {
|
|||||||
//}
|
//}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new JSONPointer instance with the provided list of reference tokens.
|
||||||
|
*
|
||||||
|
* @param refTokens A list of strings representing the reference tokens for the JSON Pointer.
|
||||||
|
* Each token identifies a step in the path to the targeted value.
|
||||||
|
*/
|
||||||
public JSONPointer(List<String> refTokens) {
|
public JSONPointer(List<String> refTokens) {
|
||||||
this.refTokens = new ArrayList<String>(refTokens);
|
this.refTokens = new ArrayList<String>(refTokens);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,10 +14,21 @@ Public Domain.
|
|||||||
public class JSONPointerException extends JSONException {
|
public class JSONPointerException extends JSONException {
|
||||||
private static final long serialVersionUID = 8872944667561856751L;
|
private static final long serialVersionUID = 8872944667561856751L;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new JSONPointerException with the specified error message.
|
||||||
|
*
|
||||||
|
* @param message The detail message describing the reason for the exception.
|
||||||
|
*/
|
||||||
public JSONPointerException(String message) {
|
public JSONPointerException(String message) {
|
||||||
super(message);
|
super(message);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new JSONPointerException with the specified error message and cause.
|
||||||
|
*
|
||||||
|
* @param message The detail message describing the reason for the exception.
|
||||||
|
* @param cause The cause of the exception.
|
||||||
|
*/
|
||||||
public JSONPointerException(String message, Throwable cause) {
|
public JSONPointerException(String message, Throwable cause) {
|
||||||
super(message, cause);
|
super(message, cause);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ import java.lang.annotation.Target;
|
|||||||
@Target({METHOD})
|
@Target({METHOD})
|
||||||
public @interface JSONPropertyName {
|
public @interface JSONPropertyName {
|
||||||
/**
|
/**
|
||||||
|
* The value of the JSON property.
|
||||||
* @return The name of the property as to be used in the JSON Object.
|
* @return The name of the property as to be used in the JSON Object.
|
||||||
*/
|
*/
|
||||||
String value();
|
String value();
|
||||||
|
|||||||
@@ -32,6 +32,8 @@ public class JSONTokener {
|
|||||||
/** the number of characters read in the previous line. */
|
/** the number of characters read in the previous line. */
|
||||||
private long characterPreviousLine;
|
private long characterPreviousLine;
|
||||||
|
|
||||||
|
// access to this object is required for strict mode checking
|
||||||
|
private JSONParserConfiguration jsonParserConfiguration;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct a JSONTokener from a Reader. The caller must close the Reader.
|
* Construct a JSONTokener from a Reader. The caller must close the Reader.
|
||||||
@@ -70,6 +72,21 @@ public class JSONTokener {
|
|||||||
this(new StringReader(s));
|
this(new StringReader(s));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Getter
|
||||||
|
* @return jsonParserConfiguration
|
||||||
|
*/
|
||||||
|
public JSONParserConfiguration getJsonParserConfiguration() {
|
||||||
|
return jsonParserConfiguration;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Setter
|
||||||
|
* @param jsonParserConfiguration new value for jsonParserConfiguration
|
||||||
|
*/
|
||||||
|
public void setJsonParserConfiguration(JSONParserConfiguration jsonParserConfiguration) {
|
||||||
|
this.jsonParserConfiguration = jsonParserConfiguration;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Back up one character. This provides a sort of lookahead capability,
|
* Back up one character. This provides a sort of lookahead capability,
|
||||||
@@ -299,7 +316,8 @@ public class JSONTokener {
|
|||||||
case 0:
|
case 0:
|
||||||
case '\n':
|
case '\n':
|
||||||
case '\r':
|
case '\r':
|
||||||
throw this.syntaxError("Unterminated string");
|
throw this.syntaxError("Unterminated string. " +
|
||||||
|
"Character with int code " + (int) c + " is not allowed within a quoted string.");
|
||||||
case '\\':
|
case '\\':
|
||||||
c = this.next();
|
c = this.next();
|
||||||
switch (c) {
|
switch (c) {
|
||||||
@@ -319,10 +337,12 @@ public class JSONTokener {
|
|||||||
sb.append('\r');
|
sb.append('\r');
|
||||||
break;
|
break;
|
||||||
case 'u':
|
case 'u':
|
||||||
|
String next = this.next(4);
|
||||||
try {
|
try {
|
||||||
sb.append((char)Integer.parseInt(this.next(4), 16));
|
sb.append((char)Integer.parseInt(next, 16));
|
||||||
} catch (NumberFormatException e) {
|
} catch (NumberFormatException e) {
|
||||||
throw this.syntaxError("Illegal escape.", e);
|
throw this.syntaxError("Illegal escape. " +
|
||||||
|
"\\u must be followed by a 4 digit hexadecimal number. \\" + next + " is not valid.", e);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case '"':
|
case '"':
|
||||||
@@ -332,7 +352,7 @@ public class JSONTokener {
|
|||||||
sb.append(c);
|
sb.append(c);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
throw this.syntaxError("Illegal escape.");
|
throw this.syntaxError("Illegal escape. Escape sequence \\" + c + " is not valid.");
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
@@ -406,14 +426,14 @@ public class JSONTokener {
|
|||||||
case '{':
|
case '{':
|
||||||
this.back();
|
this.back();
|
||||||
try {
|
try {
|
||||||
return new JSONObject(this);
|
return new JSONObject(this, jsonParserConfiguration);
|
||||||
} catch (StackOverflowError e) {
|
} catch (StackOverflowError e) {
|
||||||
throw new JSONException("JSON Array or Object depth too large to process.", e);
|
throw new JSONException("JSON Array or Object depth too large to process.", e);
|
||||||
}
|
}
|
||||||
case '[':
|
case '[':
|
||||||
this.back();
|
this.back();
|
||||||
try {
|
try {
|
||||||
return new JSONArray(this);
|
return new JSONArray(this, jsonParserConfiguration);
|
||||||
} catch (StackOverflowError e) {
|
} catch (StackOverflowError e) {
|
||||||
throw new JSONException("JSON Array or Object depth too large to process.", e);
|
throw new JSONException("JSON Array or Object depth too large to process.", e);
|
||||||
}
|
}
|
||||||
@@ -424,6 +444,12 @@ public class JSONTokener {
|
|||||||
Object nextSimpleValue(char c) {
|
Object nextSimpleValue(char c) {
|
||||||
String string;
|
String string;
|
||||||
|
|
||||||
|
// Strict mode only allows strings with explicit double quotes
|
||||||
|
if (jsonParserConfiguration != null &&
|
||||||
|
jsonParserConfiguration.isStrictMode() &&
|
||||||
|
c == '\'') {
|
||||||
|
throw this.syntaxError("Strict mode error: Single quoted strings are not allowed");
|
||||||
|
}
|
||||||
switch (c) {
|
switch (c) {
|
||||||
case '"':
|
case '"':
|
||||||
case '\'':
|
case '\'':
|
||||||
@@ -452,7 +478,14 @@ public class JSONTokener {
|
|||||||
if ("".equals(string)) {
|
if ("".equals(string)) {
|
||||||
throw this.syntaxError("Missing value");
|
throw this.syntaxError("Missing value");
|
||||||
}
|
}
|
||||||
return JSONObject.stringToValue(string);
|
Object obj = JSONObject.stringToValue(string);
|
||||||
|
// Strict mode only allows strings with explicit double quotes
|
||||||
|
if (jsonParserConfiguration != null &&
|
||||||
|
jsonParserConfiguration.isStrictMode() &&
|
||||||
|
obj instanceof String) {
|
||||||
|
throw this.syntaxError(String.format("Strict mode error: Value '%s' is not surrounded by quotes", obj));
|
||||||
|
}
|
||||||
|
return obj;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -525,6 +558,11 @@ public class JSONTokener {
|
|||||||
this.line + "]";
|
this.line + "]";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Closes the underlying reader, releasing any resources associated with it.
|
||||||
|
*
|
||||||
|
* @throws IOException If an I/O error occurs while closing the reader.
|
||||||
|
*/
|
||||||
public void close() throws IOException {
|
public void close() throws IOException {
|
||||||
if(reader!=null){
|
if(reader!=null){
|
||||||
reader.close();
|
reader.close();
|
||||||
|
|||||||
@@ -1,152 +0,0 @@
|
|||||||
package org.json;
|
|
||||||
|
|
||||||
import java.math.BigDecimal;
|
|
||||||
import java.math.BigInteger;
|
|
||||||
|
|
||||||
class NumberConversionUtil {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Converts a string to a number using the narrowest possible type. Possible
|
|
||||||
* returns for this function are BigDecimal, Double, BigInteger, Long, and Integer.
|
|
||||||
* When a Double is returned, it should always be a valid Double and not NaN or +-infinity.
|
|
||||||
*
|
|
||||||
* @param input value to convert
|
|
||||||
* @return Number representation of the value.
|
|
||||||
* @throws NumberFormatException thrown if the value is not a valid number. A public
|
|
||||||
* caller should catch this and wrap it in a {@link JSONException} if applicable.
|
|
||||||
*/
|
|
||||||
static Number stringToNumber(final String input) throws NumberFormatException {
|
|
||||||
String val = input;
|
|
||||||
if (val.startsWith(".")){
|
|
||||||
val = "0"+val;
|
|
||||||
}
|
|
||||||
if (val.startsWith("-.")){
|
|
||||||
val = "-0."+val.substring(2);
|
|
||||||
}
|
|
||||||
char initial = val.charAt(0);
|
|
||||||
if ( isNumericChar(initial) || initial == '-' ) {
|
|
||||||
// decimal representation
|
|
||||||
if (isDecimalNotation(val)) {
|
|
||||||
// Use a BigDecimal all the time so we keep the original
|
|
||||||
// representation. BigDecimal doesn't support -0.0, ensure we
|
|
||||||
// keep that by forcing a decimal.
|
|
||||||
try {
|
|
||||||
BigDecimal bd = new BigDecimal(val);
|
|
||||||
if(initial == '-' && BigDecimal.ZERO.compareTo(bd)==0) {
|
|
||||||
return Double.valueOf(-0.0);
|
|
||||||
}
|
|
||||||
return bd;
|
|
||||||
} catch (NumberFormatException retryAsDouble) {
|
|
||||||
// this is to support "Hex Floats" like this: 0x1.0P-1074
|
|
||||||
try {
|
|
||||||
Double d = Double.valueOf(val);
|
|
||||||
if(d.isNaN() || d.isInfinite()) {
|
|
||||||
throw new NumberFormatException("val ["+input+"] is not a valid number.");
|
|
||||||
}
|
|
||||||
return d;
|
|
||||||
} catch (NumberFormatException ignore) {
|
|
||||||
throw new NumberFormatException("val ["+input+"] is not a valid number.");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
val = removeLeadingZerosOfNumber(input);
|
|
||||||
initial = val.charAt(0);
|
|
||||||
if(initial == '0' && val.length() > 1) {
|
|
||||||
char at1 = val.charAt(1);
|
|
||||||
if(isNumericChar(at1)) {
|
|
||||||
throw new NumberFormatException("val ["+input+"] is not a valid number.");
|
|
||||||
}
|
|
||||||
} else if (initial == '-' && val.length() > 2) {
|
|
||||||
char at1 = val.charAt(1);
|
|
||||||
char at2 = val.charAt(2);
|
|
||||||
if(at1 == '0' && isNumericChar(at2)) {
|
|
||||||
throw new NumberFormatException("val ["+input+"] is not a valid number.");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// integer representation.
|
|
||||||
// This will narrow any values to the smallest reasonable Object representation
|
|
||||||
// (Integer, Long, or BigInteger)
|
|
||||||
|
|
||||||
// BigInteger down conversion: We use a similar bitLength compare as
|
|
||||||
// BigInteger#intValueExact uses. Increases GC, but objects hold
|
|
||||||
// only what they need. i.e. Less runtime overhead if the value is
|
|
||||||
// long lived.
|
|
||||||
BigInteger bi = new BigInteger(val);
|
|
||||||
if(bi.bitLength() <= 31){
|
|
||||||
return Integer.valueOf(bi.intValue());
|
|
||||||
}
|
|
||||||
if(bi.bitLength() <= 63){
|
|
||||||
return Long.valueOf(bi.longValue());
|
|
||||||
}
|
|
||||||
return bi;
|
|
||||||
}
|
|
||||||
throw new NumberFormatException("val ["+input+"] is not a valid number.");
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if the character is a numeric digit ('0' to '9').
|
|
||||||
*
|
|
||||||
* @param c The character to be checked.
|
|
||||||
* @return true if the character is a numeric digit, false otherwise.
|
|
||||||
*/
|
|
||||||
private static boolean isNumericChar(char c) {
|
|
||||||
return (c <= '9' && c >= '0');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if the value could be considered a number in decimal number system.
|
|
||||||
* @param value
|
|
||||||
* @return
|
|
||||||
*/
|
|
||||||
static boolean potentialNumber(String value){
|
|
||||||
if (value == null || value.isEmpty()){
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return potentialPositiveNumberStartingAtIndex(value, (value.charAt(0)=='-'?1:0));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Tests if the value should be tried as a decimal. It makes no test if there are actual digits.
|
|
||||||
*
|
|
||||||
* @param val value to test
|
|
||||||
* @return true if the string is "-0" or if it contains '.', 'e', or 'E', false otherwise.
|
|
||||||
*/
|
|
||||||
private static boolean isDecimalNotation(final String val) {
|
|
||||||
return val.indexOf('.') > -1 || val.indexOf('e') > -1
|
|
||||||
|| val.indexOf('E') > -1 || "-0".equals(val);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static boolean potentialPositiveNumberStartingAtIndex(String value,int index){
|
|
||||||
if (index >= value.length()){
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return digitAtIndex(value, (value.charAt(index)=='.'?index+1:index));
|
|
||||||
}
|
|
||||||
|
|
||||||
private static boolean digitAtIndex(String value, int index){
|
|
||||||
if (index >= value.length()){
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return value.charAt(index) >= '0' && value.charAt(index) <= '9';
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* For a prospective number, remove the leading zeros
|
|
||||||
* @param value prospective number
|
|
||||||
* @return number without leading zeros
|
|
||||||
*/
|
|
||||||
private static String removeLeadingZerosOfNumber(String value){
|
|
||||||
if (value.equals("-")){return value;}
|
|
||||||
boolean negativeFirstChar = (value.charAt(0) == '-');
|
|
||||||
int counter = negativeFirstChar ? 1:0;
|
|
||||||
while (counter < value.length()){
|
|
||||||
if (value.charAt(counter) != '0'){
|
|
||||||
if (negativeFirstChar) {return "-".concat(value.substring(counter));}
|
|
||||||
return value.substring(counter);
|
|
||||||
}
|
|
||||||
++counter;
|
|
||||||
}
|
|
||||||
if (negativeFirstChar) {return "-0";}
|
|
||||||
return "0";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -20,20 +20,29 @@ public class ParserConfiguration {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Specifies if values should be kept as strings (<code>true</code>), or if
|
* Specifies if values should be kept as strings (<code>true</code>), or if
|
||||||
* they should try to be guessed into JSON values (numeric, boolean, string)
|
* they should try to be guessed into JSON values (numeric, boolean, string).
|
||||||
*/
|
*/
|
||||||
protected boolean keepStrings;
|
protected boolean keepStrings;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The maximum nesting depth when parsing a document.
|
* The maximum nesting depth when parsing an object.
|
||||||
*/
|
*/
|
||||||
protected int maxNestingDepth;
|
protected int maxNestingDepth;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new ParserConfiguration with default settings.
|
||||||
|
*/
|
||||||
public ParserConfiguration() {
|
public ParserConfiguration() {
|
||||||
this.keepStrings = false;
|
this.keepStrings = false;
|
||||||
this.maxNestingDepth = DEFAULT_MAXIMUM_NESTING_DEPTH;
|
this.maxNestingDepth = DEFAULT_MAXIMUM_NESTING_DEPTH;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new ParserConfiguration with the specified settings.
|
||||||
|
*
|
||||||
|
* @param keepStrings A boolean indicating whether to preserve strings during parsing.
|
||||||
|
* @param maxNestingDepth An integer representing the maximum allowed nesting depth.
|
||||||
|
*/
|
||||||
protected ParserConfiguration(final boolean keepStrings, final int maxNestingDepth) {
|
protected ParserConfiguration(final boolean keepStrings, final int maxNestingDepth) {
|
||||||
this.keepStrings = keepStrings;
|
this.keepStrings = keepStrings;
|
||||||
this.maxNestingDepth = maxNestingDepth;
|
this.maxNestingDepth = maxNestingDepth;
|
||||||
@@ -57,7 +66,7 @@ public class ParserConfiguration {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* When parsing the XML into JSONML, specifies if values should be kept as strings (<code>true</code>), or if
|
* 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)
|
* they should try to be guessed into JSON values (numeric, boolean, string).
|
||||||
*
|
*
|
||||||
* @return The <code>keepStrings</code> configuration value.
|
* @return The <code>keepStrings</code> configuration value.
|
||||||
*/
|
*/
|
||||||
@@ -69,22 +78,21 @@ public class ParserConfiguration {
|
|||||||
* When parsing the XML into JSONML, specifies if values should be kept as strings (<code>true</code>), or if
|
* 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)
|
* they should try to be guessed into JSON values (numeric, boolean, string)
|
||||||
*
|
*
|
||||||
* @param newVal
|
* @param newVal new value to use for the <code>keepStrings</code> configuration option.
|
||||||
* new value to use for the <code>keepStrings</code> configuration option.
|
|
||||||
* @param <T> the type of the configuration object
|
* @param <T> the type of the configuration object
|
||||||
*
|
|
||||||
* @return The existing configuration will not be modified. A new configuration is returned.
|
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||||
*/
|
*/
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
public <T extends ParserConfiguration> T withKeepStrings(final boolean newVal) {
|
public <T extends ParserConfiguration> T withKeepStrings(final boolean newVal) {
|
||||||
T newConfig = (T)this.clone();
|
T newConfig = (T) this.clone();
|
||||||
newConfig.keepStrings = newVal;
|
newConfig.keepStrings = newVal;
|
||||||
return newConfig;
|
return newConfig;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The maximum nesting depth that the parser will descend before throwing an exception
|
* The maximum nesting depth that the parser will descend before throwing an exception
|
||||||
* when parsing the XML into JSONML.
|
* when parsing an object (e.g. Map, Collection) into JSON-related objects.
|
||||||
|
*
|
||||||
* @return the maximum nesting depth set for this configuration
|
* @return the maximum nesting depth set for this configuration
|
||||||
*/
|
*/
|
||||||
public int getMaxNestingDepth() {
|
public int getMaxNestingDepth() {
|
||||||
@@ -93,18 +101,19 @@ public class ParserConfiguration {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines the maximum nesting depth that the parser will descend before throwing an exception
|
* 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
|
* when parsing an object (e.g. Map, Collection) into JSON-related objects.
|
||||||
* will throw a JsonException if the maximum depth is reached.
|
* 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,
|
* 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.
|
* 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
|
* @param maxNestingDepth the maximum nesting depth allowed to the XML parser
|
||||||
* @param <T> the type of the configuration object
|
* @param <T> the type of the configuration object
|
||||||
*
|
|
||||||
* @return The existing configuration will not be modified. A new configuration is returned.
|
* @return The existing configuration will not be modified. A new configuration is returned.
|
||||||
*/
|
*/
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
public <T extends ParserConfiguration> T withMaxNestingDepth(int maxNestingDepth) {
|
public <T extends ParserConfiguration> T withMaxNestingDepth(int maxNestingDepth) {
|
||||||
T newConfig = (T)this.clone();
|
T newConfig = (T) this.clone();
|
||||||
|
|
||||||
if (maxNestingDepth > UNDEFINED_MAXIMUM_NESTING_DEPTH) {
|
if (maxNestingDepth > UNDEFINED_MAXIMUM_NESTING_DEPTH) {
|
||||||
newConfig.maxNestingDepth = maxNestingDepth;
|
newConfig.maxNestingDepth = maxNestingDepth;
|
||||||
|
|||||||
@@ -13,6 +13,13 @@ import java.util.Properties;
|
|||||||
* @version 2015-05-05
|
* @version 2015-05-05
|
||||||
*/
|
*/
|
||||||
public class Property {
|
public class Property {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new Property object.
|
||||||
|
*/
|
||||||
|
public Property() {
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Converts a property file object into a JSONObject. The property file object is a table of name value pairs.
|
* Converts a property file object into a JSONObject. The property file object is a table of name value pairs.
|
||||||
* @param properties java.util.Properties
|
* @param properties java.util.Properties
|
||||||
|
|||||||
92
src/main/java/org/json/StringBuilderWriter.java
Normal file
92
src/main/java/org/json/StringBuilderWriter.java
Normal file
@@ -0,0 +1,92 @@
|
|||||||
|
package org.json;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.Writer;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Performance optimised alternative for {@link java.io.StringWriter}
|
||||||
|
* using internally a {@link StringBuilder} instead of a {@link StringBuffer}.
|
||||||
|
*/
|
||||||
|
public class StringBuilderWriter extends Writer {
|
||||||
|
private final StringBuilder builder;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a new string builder writer using the default initial string-builder buffer size.
|
||||||
|
*/
|
||||||
|
public StringBuilderWriter() {
|
||||||
|
builder = new StringBuilder();
|
||||||
|
lock = builder;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a new string builder writer using the specified initial string-builder buffer size.
|
||||||
|
*
|
||||||
|
* @param initialSize The number of {@code char} values that will fit into this buffer
|
||||||
|
* before it is automatically expanded
|
||||||
|
*
|
||||||
|
* @throws IllegalArgumentException If {@code initialSize} is negative
|
||||||
|
*/
|
||||||
|
public StringBuilderWriter(int initialSize) {
|
||||||
|
builder = new StringBuilder(initialSize);
|
||||||
|
lock = builder;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void write(int c) {
|
||||||
|
builder.append((char) c);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void write(char[] cbuf, int offset, int length) {
|
||||||
|
if ((offset < 0) || (offset > cbuf.length) || (length < 0) ||
|
||||||
|
((offset + length) > cbuf.length) || ((offset + length) < 0)) {
|
||||||
|
throw new IndexOutOfBoundsException();
|
||||||
|
} else if (length == 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
builder.append(cbuf, offset, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void write(String str) {
|
||||||
|
builder.append(str);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void write(String str, int offset, int length) {
|
||||||
|
builder.append(str, offset, offset + length);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public StringBuilderWriter append(CharSequence csq) {
|
||||||
|
write(String.valueOf(csq));
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public StringBuilderWriter append(CharSequence csq, int start, int end) {
|
||||||
|
if (csq == null) {
|
||||||
|
csq = "null";
|
||||||
|
}
|
||||||
|
return append(csq.subSequence(start, end));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public StringBuilderWriter append(char c) {
|
||||||
|
write(c);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return builder.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void flush() {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void close() throws IOException {
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -10,10 +10,6 @@ import java.math.BigDecimal;
|
|||||||
import java.math.BigInteger;
|
import java.math.BigInteger;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
|
|
||||||
import static org.json.NumberConversionUtil.potentialNumber;
|
|
||||||
import static org.json.NumberConversionUtil.stringToNumber;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This provides static methods to convert an XML text into a JSONObject, and to
|
* This provides static methods to convert an XML text into a JSONObject, and to
|
||||||
* covert a JSONObject into an XML text.
|
* covert a JSONObject into an XML text.
|
||||||
@@ -24,6 +20,12 @@ import static org.json.NumberConversionUtil.stringToNumber;
|
|||||||
@SuppressWarnings("boxing")
|
@SuppressWarnings("boxing")
|
||||||
public class XML {
|
public class XML {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new XML object.
|
||||||
|
*/
|
||||||
|
public XML() {
|
||||||
|
}
|
||||||
|
|
||||||
/** The Character '&'. */
|
/** The Character '&'. */
|
||||||
public static final Character AMP = '&';
|
public static final Character AMP = '&';
|
||||||
|
|
||||||
@@ -56,6 +58,9 @@ public class XML {
|
|||||||
*/
|
*/
|
||||||
public static final String NULL_ATTR = "xsi:nil";
|
public static final String NULL_ATTR = "xsi:nil";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents the XML attribute name for specifying type information.
|
||||||
|
*/
|
||||||
public static final String TYPE_ATTR = "xsi:type";
|
public static final String TYPE_ATTR = "xsi:type";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -490,6 +495,76 @@ public class XML {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* direct copy of {@link JSONObject#stringToNumber(String)} to maintain Android support.
|
||||||
|
*/
|
||||||
|
private static Number stringToNumber(final String val) throws NumberFormatException {
|
||||||
|
char initial = val.charAt(0);
|
||||||
|
if ((initial >= '0' && initial <= '9') || initial == '-') {
|
||||||
|
// decimal representation
|
||||||
|
if (isDecimalNotation(val)) {
|
||||||
|
// Use a BigDecimal all the time so we keep the original
|
||||||
|
// representation. BigDecimal doesn't support -0.0, ensure we
|
||||||
|
// keep that by forcing a decimal.
|
||||||
|
try {
|
||||||
|
BigDecimal bd = new BigDecimal(val);
|
||||||
|
if(initial == '-' && BigDecimal.ZERO.compareTo(bd)==0) {
|
||||||
|
return Double.valueOf(-0.0);
|
||||||
|
}
|
||||||
|
return bd;
|
||||||
|
} catch (NumberFormatException retryAsDouble) {
|
||||||
|
// this is to support "Hex Floats" like this: 0x1.0P-1074
|
||||||
|
try {
|
||||||
|
Double d = Double.valueOf(val);
|
||||||
|
if(d.isNaN() || d.isInfinite()) {
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
return d;
|
||||||
|
} catch (NumberFormatException ignore) {
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// block items like 00 01 etc. Java number parsers treat these as Octal.
|
||||||
|
if(initial == '0' && val.length() > 1) {
|
||||||
|
char at1 = val.charAt(1);
|
||||||
|
if(at1 >= '0' && at1 <= '9') {
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
} else if (initial == '-' && val.length() > 2) {
|
||||||
|
char at1 = val.charAt(1);
|
||||||
|
char at2 = val.charAt(2);
|
||||||
|
if(at1 == '0' && at2 >= '0' && at2 <= '9') {
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// integer representation.
|
||||||
|
// This will narrow any values to the smallest reasonable Object representation
|
||||||
|
// (Integer, Long, or BigInteger)
|
||||||
|
|
||||||
|
// BigInteger down conversion: We use a similar bitLength compare as
|
||||||
|
// BigInteger#intValueExact uses. Increases GC, but objects hold
|
||||||
|
// only what they need. i.e. Less runtime overhead if the value is
|
||||||
|
// long lived.
|
||||||
|
BigInteger bi = new BigInteger(val);
|
||||||
|
if(bi.bitLength() <= 31){
|
||||||
|
return Integer.valueOf(bi.intValue());
|
||||||
|
}
|
||||||
|
if(bi.bitLength() <= 63){
|
||||||
|
return Long.valueOf(bi.longValue());
|
||||||
|
}
|
||||||
|
return bi;
|
||||||
|
}
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* direct copy of {@link JSONObject#isDecimalNotation(String)} to maintain Android support.
|
||||||
|
*/
|
||||||
|
private static boolean isDecimalNotation(final String val) {
|
||||||
|
return val.indexOf('.') > -1 || val.indexOf('e') > -1
|
||||||
|
|| val.indexOf('E') > -1 || "-0".equals(val);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method tries to convert the given string value to the target object
|
* This method tries to convert the given string value to the target object
|
||||||
@@ -534,7 +609,8 @@ public class XML {
|
|||||||
* produced, then the value will just be a string.
|
* produced, then the value will just be a string.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
if (potentialNumber(string)) {
|
char initial = string.charAt(0);
|
||||||
|
if ((initial >= '0' && initial <= '9') || initial == '-') {
|
||||||
try {
|
try {
|
||||||
return stringToNumber(string);
|
return stringToNumber(string);
|
||||||
} catch (Exception ignore) {
|
} catch (Exception ignore) {
|
||||||
@@ -543,11 +619,6 @@ public class XML {
|
|||||||
return string;
|
return string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert a well-formed (but not necessarily valid) XML string into a
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
* JSONObject. Some information may be lost in this transformation because
|
* JSONObject. Some information may be lost in this transformation because
|
||||||
@@ -966,5 +1037,4 @@ public class XML {
|
|||||||
}
|
}
|
||||||
return sb.toString();
|
return sb.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -352,9 +352,20 @@ public class XMLParserConfiguration extends ParserConfiguration {
|
|||||||
return clonedConfiguration;
|
return clonedConfiguration;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the parser should automatically close empty XML tags.
|
||||||
|
*
|
||||||
|
* @return {@code true} if empty XML tags should be automatically closed, {@code false} otherwise.
|
||||||
|
*/
|
||||||
public boolean isCloseEmptyTag() {
|
public boolean isCloseEmptyTag() {
|
||||||
return this.closeEmptyTag;
|
return this.closeEmptyTag;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the parser should trim white spaces from XML content.
|
||||||
|
*
|
||||||
|
* @return {@code true} if white spaces should be trimmed, {@code false} otherwise.
|
||||||
|
*/
|
||||||
public boolean shouldTrimWhiteSpace() {
|
public boolean shouldTrimWhiteSpace() {
|
||||||
return this.shouldTrimWhiteSpace;
|
return this.shouldTrimWhiteSpace;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -42,5 +42,12 @@ Public Domain.
|
|||||||
* @param <T> return type of convert method
|
* @param <T> return type of convert method
|
||||||
*/
|
*/
|
||||||
public interface XMLXsiTypeConverter<T> {
|
public interface XMLXsiTypeConverter<T> {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts an XML xsi:type attribute value to the specified type {@code T}.
|
||||||
|
*
|
||||||
|
* @param value The string representation of the XML xsi:type attribute value to be converted.
|
||||||
|
* @return An object of type {@code T} representing the converted value.
|
||||||
|
*/
|
||||||
T convert(String value);
|
T convert(String value);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,174 +0,0 @@
|
|||||||
package org.json;
|
|
||||||
|
|
||||||
import org.junit.Test;
|
|
||||||
|
|
||||||
import java.math.BigDecimal;
|
|
||||||
import java.math.BigInteger;
|
|
||||||
|
|
||||||
import static org.junit.Assert.*;
|
|
||||||
|
|
||||||
public class NumberConversionUtilTest {
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseDecimalFractionNumbersWithMultipleLeadingZeros(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("00.10d");
|
|
||||||
assertEquals("Do not match", 0.10d, number.doubleValue(),0.0d);
|
|
||||||
assertEquals("Do not match", 0.10f, number.floatValue(),0.0f);
|
|
||||||
assertEquals("Do not match", 0, number.longValue(),0);
|
|
||||||
assertEquals("Do not match", 0, number.intValue(),0);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseDecimalFractionNumbersWithSingleLeadingZero(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("0.10d");
|
|
||||||
assertEquals("Do not match", 0.10d, number.doubleValue(),0.0d);
|
|
||||||
assertEquals("Do not match", 0.10f, number.floatValue(),0.0f);
|
|
||||||
assertEquals("Do not match", 0, number.longValue(),0);
|
|
||||||
assertEquals("Do not match", 0, number.intValue(),0);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseDecimalFractionNumbersWithZerosAfterDecimalPoint(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("0.010d");
|
|
||||||
assertEquals("Do not match", 0.010d, number.doubleValue(),0.0d);
|
|
||||||
assertEquals("Do not match", 0.010f, number.floatValue(),0.0f);
|
|
||||||
assertEquals("Do not match", 0, number.longValue(),0);
|
|
||||||
assertEquals("Do not match", 0, number.intValue(),0);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseMixedDecimalFractionNumbersWithMultipleLeadingZeros(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("00200.10d");
|
|
||||||
assertEquals("Do not match", 200.10d, number.doubleValue(),0.0d);
|
|
||||||
assertEquals("Do not match", 200.10f, number.floatValue(),0.0f);
|
|
||||||
assertEquals("Do not match", 200, number.longValue(),0);
|
|
||||||
assertEquals("Do not match", 200, number.intValue(),0);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseMixedDecimalFractionNumbersWithoutLeadingZero(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("200.10d");
|
|
||||||
assertEquals("Do not match", 200.10d, number.doubleValue(),0.0d);
|
|
||||||
assertEquals("Do not match", 200.10f, number.floatValue(),0.0f);
|
|
||||||
assertEquals("Do not match", 200, number.longValue(),0);
|
|
||||||
assertEquals("Do not match", 200, number.intValue(),0);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseMixedDecimalFractionNumbersWithZerosAfterDecimalPoint(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("200.010d");
|
|
||||||
assertEquals("Do not match", 200.010d, number.doubleValue(),0.0d);
|
|
||||||
assertEquals("Do not match", 200.010f, number.floatValue(),0.0f);
|
|
||||||
assertEquals("Do not match", 200, number.longValue(),0);
|
|
||||||
assertEquals("Do not match", 200, number.intValue(),0);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseNegativeDecimalFractionNumbersWithMultipleLeadingZeros(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("-00.10d");
|
|
||||||
assertEquals("Do not match", -0.10d, number.doubleValue(),0.0d);
|
|
||||||
assertEquals("Do not match", -0.10f, number.floatValue(),0.0f);
|
|
||||||
assertEquals("Do not match", -0, number.longValue(),0);
|
|
||||||
assertEquals("Do not match", -0, number.intValue(),0);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseNegativeDecimalFractionNumbersWithSingleLeadingZero(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("-0.10d");
|
|
||||||
assertEquals("Do not match", -0.10d, number.doubleValue(),0.0d);
|
|
||||||
assertEquals("Do not match", -0.10f, number.floatValue(),0.0f);
|
|
||||||
assertEquals("Do not match", -0, number.longValue(),0);
|
|
||||||
assertEquals("Do not match", -0, number.intValue(),0);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseNegativeDecimalFractionNumbersWithZerosAfterDecimalPoint(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("-0.010d");
|
|
||||||
assertEquals("Do not match", -0.010d, number.doubleValue(),0.0d);
|
|
||||||
assertEquals("Do not match", -0.010f, number.floatValue(),0.0f);
|
|
||||||
assertEquals("Do not match", -0, number.longValue(),0);
|
|
||||||
assertEquals("Do not match", -0, number.intValue(),0);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseNegativeMixedDecimalFractionNumbersWithMultipleLeadingZeros(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("-00200.10d");
|
|
||||||
assertEquals("Do not match", -200.10d, number.doubleValue(),0.0d);
|
|
||||||
assertEquals("Do not match", -200.10f, number.floatValue(),0.0f);
|
|
||||||
assertEquals("Do not match", -200, number.longValue(),0);
|
|
||||||
assertEquals("Do not match", -200, number.intValue(),0);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseNegativeMixedDecimalFractionNumbersWithoutLeadingZero(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("-200.10d");
|
|
||||||
assertEquals("Do not match", -200.10d, number.doubleValue(),0.0d);
|
|
||||||
assertEquals("Do not match", -200.10f, number.floatValue(),0.0f);
|
|
||||||
assertEquals("Do not match", -200, number.longValue(),0);
|
|
||||||
assertEquals("Do not match", -200, number.intValue(),0);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseNegativeMixedDecimalFractionNumbersWithZerosAfterDecimalPoint(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("-200.010d");
|
|
||||||
assertEquals("Do not match", -200.010d, number.doubleValue(),0.0d);
|
|
||||||
assertEquals("Do not match", -200.010f, number.floatValue(),0.0f);
|
|
||||||
assertEquals("Do not match", -200, number.longValue(),0);
|
|
||||||
assertEquals("Do not match", -200, number.intValue(),0);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseNumbersWithExponents(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("23.45e7");
|
|
||||||
assertEquals("Do not match", 23.45e7d, number.doubleValue(),0.0d);
|
|
||||||
assertEquals("Do not match", 23.45e7f, number.floatValue(),0.0f);
|
|
||||||
assertEquals("Do not match", 2.345E8, number.longValue(),0);
|
|
||||||
assertEquals("Do not match", 2.345E8, number.intValue(),0);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseNegativeNumbersWithExponents(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("-23.45e7");
|
|
||||||
assertEquals("Do not match", -23.45e7d, number.doubleValue(),0.0d);
|
|
||||||
assertEquals("Do not match", -23.45e7f, number.floatValue(),0.0f);
|
|
||||||
assertEquals("Do not match", -2.345E8, number.longValue(),0);
|
|
||||||
assertEquals("Do not match", -2.345E8, number.intValue(),0);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseBigDecimal(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("19007199254740993.35481234487103587486413587843213584");
|
|
||||||
assertTrue(number instanceof BigDecimal);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseBigInteger(){
|
|
||||||
Number number = NumberConversionUtil.stringToNumber("1900719925474099335481234487103587486413587843213584");
|
|
||||||
assertTrue(number instanceof BigInteger);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldIdentifyPotentialNumber(){
|
|
||||||
assertTrue("Does not identify as number", NumberConversionUtil.potentialNumber("112.123"));
|
|
||||||
assertTrue("Does not identify as number", NumberConversionUtil.potentialNumber("112e123"));
|
|
||||||
assertTrue("Does not identify as number", NumberConversionUtil.potentialNumber("-112.123"));
|
|
||||||
assertTrue("Does not identify as number", NumberConversionUtil.potentialNumber("-112e23"));
|
|
||||||
assertFalse("Does not identify as not number", NumberConversionUtil.potentialNumber("--112.123"));
|
|
||||||
assertFalse("Does not identify as not number", NumberConversionUtil.potentialNumber("-a112.123"));
|
|
||||||
assertFalse("Does not identify as not number", NumberConversionUtil.potentialNumber("a112.123"));
|
|
||||||
assertFalse("Does not identify as not number", NumberConversionUtil.potentialNumber("e112.123"));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test(expected = NumberFormatException.class)
|
|
||||||
public void shouldExpectExceptionWhenNumberIsNotFormatted(){
|
|
||||||
NumberConversionUtil.stringToNumber("112.aa123");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
@@ -24,14 +24,13 @@ public class CDLTest {
|
|||||||
* String of lines where the column names are in the first row,
|
* String of lines where the column names are in the first row,
|
||||||
* and all subsequent rows are values. All keys and values should be legal.
|
* and all subsequent rows are values. All keys and values should be legal.
|
||||||
*/
|
*/
|
||||||
String lines = new String(
|
private static final String LINES = "Col 1, Col 2, \tCol 3, Col 4, Col 5, Col 6, Col 7\n" +
|
||||||
"Col 1, Col 2, \tCol 3, Col 4, Col 5, Col 6, Col 7\n" +
|
|
||||||
"val1, val2, val3, val4, val5, val6, val7\n" +
|
"val1, val2, val3, val4, val5, val6, val7\n" +
|
||||||
"1, 2, 3, 4\t, 5, 6, 7\n" +
|
"1, 2, 3, 4\t, 5, 6, 7\n" +
|
||||||
"true, false, true, true, false, false, false\n" +
|
"true, false, true, true, false, false, false\n" +
|
||||||
"0.23, 57.42, 5e27, -234.879, 2.34e5, 0.0, 9e-3\n" +
|
"0.23, 57.42, 5e27, -234.879, 2.34e5, 0.0, 9e-3\n" +
|
||||||
"\"va\tl1\", \"v\bal2\", \"val3\", \"val\f4\", \"val5\", va\'l6, val7\n"
|
"\"va\tl1\", \"v\bal2\", \"val3\", \"val\f4\", \"val5\", \"va'l6\", val7\n";
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* CDL.toJSONArray() adds all values as strings, with no filtering or
|
* CDL.toJSONArray() adds all values as strings, with no filtering or
|
||||||
@@ -39,12 +38,54 @@ public class CDLTest {
|
|||||||
* values all must be quoted in the cases where the JSONObject parsing
|
* values all must be quoted in the cases where the JSONObject parsing
|
||||||
* might normally convert the value into a non-string.
|
* might normally convert the value into a non-string.
|
||||||
*/
|
*/
|
||||||
String expectedLines = new String(
|
private static final String EXPECTED_LINES =
|
||||||
"[{Col 1:val1, Col 2:val2, Col 3:val3, Col 4:val4, Col 5:val5, Col 6:val6, Col 7:val7}, "+
|
"[ " +
|
||||||
"{Col 1:\"1\", Col 2:\"2\", Col 3:\"3\", Col 4:\"4\", Col 5:\"5\", Col 6:\"6\", Col 7:\"7\"}, "+
|
"{" +
|
||||||
"{Col 1:\"true\", Col 2:\"false\", Col 3:\"true\", Col 4:\"true\", Col 5:\"false\", Col 6:\"false\", Col 7:\"false\"}, "+
|
"\"Col 1\":\"val1\", " +
|
||||||
"{Col 1:\"0.23\", Col 2:\"57.42\", Col 3:\"5e27\", Col 4:\"-234.879\", Col 5:\"2.34e5\", Col 6:\"0.0\", Col 7:\"9e-3\"}, "+
|
"\"Col 2\":\"val2\", " +
|
||||||
"{Col 1:\"va\tl1\", Col 2:\"v\bal2\", Col 3:val3, Col 4:\"val\f4\", Col 5:val5, Col 6:va\'l6, Col 7:val7}]");
|
"\"Col 3\":\"val3\", " +
|
||||||
|
"\"Col 4\":\"val4\", " +
|
||||||
|
"\"Col 5\":\"val5\", " +
|
||||||
|
"\"Col 6\":\"val6\", " +
|
||||||
|
"\"Col 7\":\"val7\"" +
|
||||||
|
"}, " +
|
||||||
|
" {" +
|
||||||
|
"\"Col 1\":\"1\", " +
|
||||||
|
"\"Col 2\":\"2\", " +
|
||||||
|
"\"Col 3\":\"3\", " +
|
||||||
|
"\"Col 4\":\"4\", " +
|
||||||
|
"\"Col 5\":\"5\", " +
|
||||||
|
"\"Col 6\":\"6\", " +
|
||||||
|
"\"Col 7\":\"7\"" +
|
||||||
|
"}, " +
|
||||||
|
" {" +
|
||||||
|
"\"Col 1\":\"true\", " +
|
||||||
|
"\"Col 2\":\"false\", " +
|
||||||
|
"\"Col 3\":\"true\", " +
|
||||||
|
"\"Col 4\":\"true\", " +
|
||||||
|
"\"Col 5\":\"false\", " +
|
||||||
|
"\"Col 6\":\"false\", " +
|
||||||
|
"\"Col 7\":\"false\"" +
|
||||||
|
"}, " +
|
||||||
|
"{" +
|
||||||
|
"\"Col 1\":\"0.23\", " +
|
||||||
|
"\"Col 2\":\"57.42\", " +
|
||||||
|
"\"Col 3\":\"5e27\", " +
|
||||||
|
"\"Col 4\":\"-234.879\", " +
|
||||||
|
"\"Col 5\":\"2.34e5\", " +
|
||||||
|
"\"Col 6\":\"0.0\", " +
|
||||||
|
"\"Col 7\":\"9e-3\"" +
|
||||||
|
"}, " +
|
||||||
|
"{" +
|
||||||
|
"\"Col 1\":\"va\tl1\", " +
|
||||||
|
"\"Col 2\":\"v\bal2\", " +
|
||||||
|
"\"Col 3\":\"val3\", " +
|
||||||
|
"\"Col 4\":\"val\f4\", " +
|
||||||
|
"\"Col 5\":\"val5\", " +
|
||||||
|
"\"Col 6\":\"va'l6\", " +
|
||||||
|
"\"Col 7\":\"val7\"" +
|
||||||
|
"}" +
|
||||||
|
"]";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Attempts to create a JSONArray from a null string.
|
* Attempts to create a JSONArray from a null string.
|
||||||
@@ -194,8 +235,7 @@ public class CDLTest {
|
|||||||
public void emptyString() {
|
public void emptyString() {
|
||||||
String emptyStr = "";
|
String emptyStr = "";
|
||||||
JSONArray jsonArray = CDL.toJSONArray(emptyStr);
|
JSONArray jsonArray = CDL.toJSONArray(emptyStr);
|
||||||
assertTrue("CDL should return null when the input string is empty",
|
assertNull("CDL should return null when the input string is empty", jsonArray);
|
||||||
jsonArray == null);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -254,7 +294,7 @@ public class CDLTest {
|
|||||||
jsonObject.put("Col \r1", "V1");
|
jsonObject.put("Col \r1", "V1");
|
||||||
// \r will be filtered from value
|
// \r will be filtered from value
|
||||||
jsonObject.put("Col 2", "V2\r");
|
jsonObject.put("Col 2", "V2\r");
|
||||||
assertTrue("expected length should be 1",jsonArray.length() == 1);
|
assertEquals("expected length should be 1", 1, jsonArray.length());
|
||||||
String cdlStr = CDL.toString(jsonArray);
|
String cdlStr = CDL.toString(jsonArray);
|
||||||
jsonObject = jsonArray.getJSONObject(0);
|
jsonObject = jsonArray.getJSONObject(0);
|
||||||
assertTrue(cdlStr.contains("\"Col 1\""));
|
assertTrue(cdlStr.contains("\"Col 1\""));
|
||||||
@@ -268,8 +308,15 @@ public class CDLTest {
|
|||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void textToJSONArray() {
|
public void textToJSONArray() {
|
||||||
JSONArray jsonArray = CDL.toJSONArray(this.lines);
|
JSONArray jsonArray = CDL.toJSONArray(LINES);
|
||||||
JSONArray expectedJsonArray = new JSONArray(this.expectedLines);
|
JSONArray expectedJsonArray = new JSONArray(EXPECTED_LINES);
|
||||||
|
Util.compareActualVsExpectedJsonArrays(jsonArray, expectedJsonArray);
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void textToJSONArrayPipeDelimited() {
|
||||||
|
char delimiter = '|';
|
||||||
|
JSONArray jsonArray = CDL.toJSONArray(LINES.replaceAll(",", String.valueOf(delimiter)), delimiter);
|
||||||
|
JSONArray expectedJsonArray = new JSONArray(EXPECTED_LINES);
|
||||||
Util.compareActualVsExpectedJsonArrays(jsonArray, expectedJsonArray);
|
Util.compareActualVsExpectedJsonArrays(jsonArray, expectedJsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -279,11 +326,11 @@ public class CDLTest {
|
|||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void jsonArrayToJSONArray() {
|
public void jsonArrayToJSONArray() {
|
||||||
String nameArrayStr = "[Col1, Col2]";
|
String nameArrayStr = "[\"Col1\", \"Col2\"]";
|
||||||
String values = "V1, V2";
|
String values = "V1, V2";
|
||||||
JSONArray nameJSONArray = new JSONArray(nameArrayStr);
|
JSONArray nameJSONArray = new JSONArray(nameArrayStr);
|
||||||
JSONArray jsonArray = CDL.toJSONArray(nameJSONArray, values);
|
JSONArray jsonArray = CDL.toJSONArray(nameJSONArray, values);
|
||||||
JSONArray expectedJsonArray = new JSONArray("[{Col1:V1,Col2:V2}]");
|
JSONArray expectedJsonArray = new JSONArray("[{\"Col1\":\"V1\",\"Col2\":\"V2\"}]");
|
||||||
Util.compareActualVsExpectedJsonArrays(jsonArray, expectedJsonArray);
|
Util.compareActualVsExpectedJsonArrays(jsonArray, expectedJsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -293,10 +340,24 @@ public class CDLTest {
|
|||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void textToJSONArrayAndBackToString() {
|
public void textToJSONArrayAndBackToString() {
|
||||||
JSONArray jsonArray = CDL.toJSONArray(this.lines);
|
JSONArray jsonArray = CDL.toJSONArray(LINES);
|
||||||
String jsonStr = CDL.toString(jsonArray);
|
String jsonStr = CDL.toString(jsonArray);
|
||||||
JSONArray finalJsonArray = CDL.toJSONArray(jsonStr);
|
JSONArray finalJsonArray = CDL.toJSONArray(jsonStr);
|
||||||
JSONArray expectedJsonArray = new JSONArray(this.expectedLines);
|
JSONArray expectedJsonArray = new JSONArray(EXPECTED_LINES);
|
||||||
|
Util.compareActualVsExpectedJsonArrays(finalJsonArray, expectedJsonArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a JSONArray from a string of lines,
|
||||||
|
* then convert to string and then back to JSONArray
|
||||||
|
* with a custom delimiter
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void textToJSONArrayAndBackToStringCustomDelimiter() {
|
||||||
|
JSONArray jsonArray = CDL.toJSONArray(LINES, ',');
|
||||||
|
String jsonStr = CDL.toString(jsonArray, ';');
|
||||||
|
JSONArray finalJsonArray = CDL.toJSONArray(jsonStr, ';');
|
||||||
|
JSONArray expectedJsonArray = new JSONArray(EXPECTED_LINES);
|
||||||
Util.compareActualVsExpectedJsonArrays(finalJsonArray, expectedJsonArray);
|
Util.compareActualVsExpectedJsonArrays(finalJsonArray, expectedJsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -259,6 +259,11 @@ public class JSONArrayTest {
|
|||||||
jsonArray.length(),
|
jsonArray.length(),
|
||||||
len);
|
len);
|
||||||
|
|
||||||
|
// collection as object
|
||||||
|
@SuppressWarnings("RedundantCast")
|
||||||
|
Object myListAsObject = (Object) myList;
|
||||||
|
jsonArray.putAll(myListAsObject);
|
||||||
|
|
||||||
for (int i = 0; i < myList.size(); i++) {
|
for (int i = 0; i < myList.size(); i++) {
|
||||||
assertEquals("collection elements should be equal",
|
assertEquals("collection elements should be equal",
|
||||||
myList.get(i),
|
myList.get(i),
|
||||||
@@ -471,11 +476,16 @@ public class JSONArrayTest {
|
|||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void unquotedText() {
|
public void unquotedText() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration();
|
||||||
|
if (jsonParserConfiguration.isStrictMode()) {
|
||||||
|
System.out.println("Skipping JSONArrayTest unquotedText() when strictMode default is true");
|
||||||
|
} else {
|
||||||
String str = "[value1, something!, (parens), foo@bar.com, 23, 23+45]";
|
String str = "[value1, something!, (parens), foo@bar.com, 23, 23+45]";
|
||||||
JSONArray jsonArray = new JSONArray(str);
|
JSONArray jsonArray = new JSONArray(str);
|
||||||
List<Object> expected = Arrays.asList("value1", "something!", "(parens)", "foo@bar.com", 23, "23+45");
|
List<Object> expected = Arrays.asList("value1", "something!", "(parens)", "foo@bar.com", 23, "23+45");
|
||||||
assertEquals(expected, jsonArray.toList());
|
assertEquals(expected, jsonArray.toList());
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Exercise JSONArray.join() by converting a JSONArray into a
|
* Exercise JSONArray.join() by converting a JSONArray into a
|
||||||
@@ -685,8 +695,8 @@ public class JSONArrayTest {
|
|||||||
|
|
||||||
String jsonArrayStr =
|
String jsonArrayStr =
|
||||||
"["+
|
"["+
|
||||||
"hello,"+
|
"\"hello\","+
|
||||||
"world"+
|
"\"world\""+
|
||||||
"]";
|
"]";
|
||||||
// 2
|
// 2
|
||||||
jsonArray.put(new JSONArray(jsonArrayStr));
|
jsonArray.put(new JSONArray(jsonArrayStr));
|
||||||
@@ -763,8 +773,8 @@ public class JSONArrayTest {
|
|||||||
|
|
||||||
String jsonArrayStr =
|
String jsonArrayStr =
|
||||||
"["+
|
"["+
|
||||||
"hello,"+
|
"\"hello\","+
|
||||||
"world"+
|
"\"world\""+
|
||||||
"]";
|
"]";
|
||||||
// 2
|
// 2
|
||||||
jsonArray.put(2, new JSONArray(jsonArrayStr));
|
jsonArray.put(2, new JSONArray(jsonArrayStr));
|
||||||
|
|||||||
@@ -625,7 +625,7 @@ public class JSONMLTest {
|
|||||||
"\"subValue\","+
|
"\"subValue\","+
|
||||||
"{\"svAttr\":\"svValue\"},"+
|
"{\"svAttr\":\"svValue\"},"+
|
||||||
"\"abc\""+
|
"\"abc\""+
|
||||||
"],"+
|
"]"+
|
||||||
"],"+
|
"],"+
|
||||||
"[\"value\",3],"+
|
"[\"value\",3],"+
|
||||||
"[\"value\",4.1],"+
|
"[\"value\",4.1],"+
|
||||||
@@ -709,7 +709,7 @@ public class JSONMLTest {
|
|||||||
@Test
|
@Test
|
||||||
public void testToJSONArray_jsonOutput() {
|
public void testToJSONArray_jsonOutput() {
|
||||||
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
||||||
final String expectedJsonString = "[\"root\",[\"id\",1],[\"id\",1],[\"id\",0],[\"id\",0],[\"item\",{\"id\":1}],[\"title\",true]]";
|
final String expectedJsonString = "[\"root\",[\"id\",\"01\"],[\"id\",1],[\"id\",\"00\"],[\"id\",0],[\"item\",{\"id\":\"01\"}],[\"title\",true]]";
|
||||||
final JSONArray actualJsonOutput = JSONML.toJSONArray(originalXml, false);
|
final JSONArray actualJsonOutput = JSONML.toJSONArray(originalXml, false);
|
||||||
assertEquals(expectedJsonString, actualJsonOutput.toString());
|
assertEquals(expectedJsonString, actualJsonOutput.toString());
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,100 +0,0 @@
|
|||||||
package org.json.junit;
|
|
||||||
|
|
||||||
import org.json.JSONObject;
|
|
||||||
import org.junit.Test;
|
|
||||||
|
|
||||||
import java.math.BigDecimal;
|
|
||||||
import java.math.BigInteger;
|
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
|
||||||
|
|
||||||
public class JSONObjectDecimalTest {
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseDecimalNumberThatStartsWithDecimalPoint(){
|
|
||||||
JSONObject jsonObject = new JSONObject("{value:0.50}");
|
|
||||||
assertEquals("Float not recognized", 0.5f, jsonObject.getFloat("value"), 0.0f);
|
|
||||||
assertEquals("Float not recognized", 0.5f, jsonObject.optFloat("value"), 0.0f);
|
|
||||||
assertEquals("Float not recognized", 0.5f, jsonObject.optFloatObject("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", 0.5d, jsonObject.optDouble("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", 0.5d, jsonObject.optDoubleObject("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", 0.5d, jsonObject.getDouble("value"), 0.0f);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.optLong("value"), 0);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.getLong("value"), 0);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.optLongObject("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.optInt("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.getInt("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.optIntegerObject("value"), 0);
|
|
||||||
assertEquals("Number not recognized", 0, jsonObject.getNumber("value").intValue(), 0);
|
|
||||||
assertEquals("Number not recognized", 0, jsonObject.getNumber("value").longValue(), 0);
|
|
||||||
assertEquals("BigDecimal not recognized", 0, BigDecimal.valueOf(.5).compareTo(jsonObject.getBigDecimal("value")));
|
|
||||||
assertEquals("BigInteger not recognized",0, BigInteger.valueOf(0).compareTo(jsonObject.getBigInteger("value")));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseNegativeDecimalNumberThatStartsWithDecimalPoint(){
|
|
||||||
JSONObject jsonObject = new JSONObject("{value:-.50}");
|
|
||||||
assertEquals("Float not recognized", -0.5f, jsonObject.getFloat("value"), 0.0f);
|
|
||||||
assertEquals("Float not recognized", -0.5f, jsonObject.optFloat("value"), 0.0f);
|
|
||||||
assertEquals("Float not recognized", -0.5f, jsonObject.optFloatObject("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", -0.5d, jsonObject.optDouble("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", -0.5d, jsonObject.optDoubleObject("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", -0.5d, jsonObject.getDouble("value"), 0.0f);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.optLong("value"), 0);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.getLong("value"), 0);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.optLongObject("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.optInt("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.getInt("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.optIntegerObject("value"), 0);
|
|
||||||
assertEquals("Number not recognized", 0, jsonObject.getNumber("value").intValue(), 0);
|
|
||||||
assertEquals("Number not recognized", 0, jsonObject.getNumber("value").longValue(), 0);
|
|
||||||
assertEquals("BigDecimal not recognized", 0, BigDecimal.valueOf(-.5).compareTo(jsonObject.getBigDecimal("value")));
|
|
||||||
assertEquals("BigInteger not recognized",0, BigInteger.valueOf(0).compareTo(jsonObject.getBigInteger("value")));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseDecimalNumberThatHasZeroBeforeWithDecimalPoint(){
|
|
||||||
JSONObject jsonObject = new JSONObject("{value:00.050}");
|
|
||||||
assertEquals("Float not recognized", 0.05f, jsonObject.getFloat("value"), 0.0f);
|
|
||||||
assertEquals("Float not recognized", 0.05f, jsonObject.optFloat("value"), 0.0f);
|
|
||||||
assertEquals("Float not recognized", 0.05f, jsonObject.optFloatObject("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", 0.05d, jsonObject.optDouble("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", 0.05d, jsonObject.optDoubleObject("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", 0.05d, jsonObject.getDouble("value"), 0.0f);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.optLong("value"), 0);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.getLong("value"), 0);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.optLongObject("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.optInt("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.getInt("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.optIntegerObject("value"), 0);
|
|
||||||
assertEquals("Number not recognized", 0, jsonObject.getNumber("value").intValue(), 0);
|
|
||||||
assertEquals("Number not recognized", 0, jsonObject.getNumber("value").longValue(), 0);
|
|
||||||
assertEquals("BigDecimal not recognized", 0, BigDecimal.valueOf(.05).compareTo(jsonObject.getBigDecimal("value")));
|
|
||||||
assertEquals("BigInteger not recognized",0, BigInteger.valueOf(0).compareTo(jsonObject.getBigInteger("value")));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseNegativeDecimalNumberThatHasZeroBeforeWithDecimalPoint(){
|
|
||||||
JSONObject jsonObject = new JSONObject("{value:-00.050}");
|
|
||||||
assertEquals("Float not recognized", -0.05f, jsonObject.getFloat("value"), 0.0f);
|
|
||||||
assertEquals("Float not recognized", -0.05f, jsonObject.optFloat("value"), 0.0f);
|
|
||||||
assertEquals("Float not recognized", -0.05f, jsonObject.optFloatObject("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", -0.05d, jsonObject.optDouble("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", -0.05d, jsonObject.optDoubleObject("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", -0.05d, jsonObject.getDouble("value"), 0.0f);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.optLong("value"), 0);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.getLong("value"), 0);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.optLongObject("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.optInt("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.getInt("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.optIntegerObject("value"), 0);
|
|
||||||
assertEquals("Number not recognized", 0, jsonObject.getNumber("value").intValue(), 0);
|
|
||||||
assertEquals("Number not recognized", 0, jsonObject.getNumber("value").longValue(), 0);
|
|
||||||
assertEquals("BigDecimal not recognized", 0, BigDecimal.valueOf(-.05).compareTo(jsonObject.getBigDecimal("value")));
|
|
||||||
assertEquals("BigInteger not recognized",0, BigInteger.valueOf(0).compareTo(jsonObject.getBigInteger("value")));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
@@ -23,22 +23,18 @@ public class JSONObjectNumberTest {
|
|||||||
@Parameters(name = "{index}: {0}")
|
@Parameters(name = "{index}: {0}")
|
||||||
public static Collection<Object[]> data() {
|
public static Collection<Object[]> data() {
|
||||||
return Arrays.asList(new Object[][]{
|
return Arrays.asList(new Object[][]{
|
||||||
{"{value:0050}", 1},
|
{"{\"value\":50}", 1},
|
||||||
{"{value:0050.0000}", 1},
|
{"{\"value\":50.0}", 1},
|
||||||
{"{value:-0050}", -1},
|
{"{\"value\":5e1}", 1},
|
||||||
{"{value:-0050.0000}", -1},
|
{"{\"value\":5E1}", 1},
|
||||||
{"{value:50.0}", 1},
|
{"{\"value\":5e1}", 1},
|
||||||
{"{value:5e1}", 1},
|
{"{\"value\":\"50\"}", 1},
|
||||||
{"{value:5E1}", 1},
|
{"{\"value\":-50}", -1},
|
||||||
{"{value:5e1}", 1},
|
{"{\"value\":-50.0}", -1},
|
||||||
{"{value:'50'}", 1},
|
{"{\"value\":-5e1}", -1},
|
||||||
{"{value:-50}", -1},
|
{"{\"value\":-5E1}", -1},
|
||||||
{"{value:-50.0}", -1},
|
{"{\"value\":-5e1}", -1},
|
||||||
{"{value:-5e1}", -1},
|
{"{\"value\":\"-50\"}", -1}
|
||||||
{"{value:-0005e1}", -1},
|
|
||||||
{"{value:-5E1}", -1},
|
|
||||||
{"{value:-5e1}", -1},
|
|
||||||
{"{value:'-50'}", -1}
|
|
||||||
// JSON does not support octal or hex numbers;
|
// JSON does not support octal or hex numbers;
|
||||||
// see https://stackoverflow.com/a/52671839/6323312
|
// see https://stackoverflow.com/a/52671839/6323312
|
||||||
// "{value:062}", // octal 50
|
// "{value:062}", // octal 50
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ package org.json.junit;
|
|||||||
Public Domain.
|
Public Domain.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static java.lang.Double.NaN;
|
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
import static org.junit.Assert.assertFalse;
|
import static org.junit.Assert.assertFalse;
|
||||||
import static org.junit.Assert.assertNotEquals;
|
import static org.junit.Assert.assertNotEquals;
|
||||||
@@ -57,6 +56,7 @@ import org.json.junit.data.RecursiveBeanEquals;
|
|||||||
import org.json.junit.data.Singleton;
|
import org.json.junit.data.Singleton;
|
||||||
import org.json.junit.data.SingletonEnum;
|
import org.json.junit.data.SingletonEnum;
|
||||||
import org.json.junit.data.WeirdList;
|
import org.json.junit.data.WeirdList;
|
||||||
|
import org.junit.After;
|
||||||
import org.junit.Ignore;
|
import org.junit.Ignore;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
@@ -76,6 +76,14 @@ public class JSONObjectTest {
|
|||||||
*/
|
*/
|
||||||
static final Pattern NUMBER_PATTERN = Pattern.compile("-?(?:0|[1-9]\\d*)(?:\\.\\d+)?(?:[eE][+-]?\\d+)?");
|
static final Pattern NUMBER_PATTERN = Pattern.compile("-?(?:0|[1-9]\\d*)(?:\\.\\d+)?(?:[eE][+-]?\\d+)?");
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void tearDown() {
|
||||||
|
SingletonEnum.getInstance().setSomeInt(0);
|
||||||
|
SingletonEnum.getInstance().setSomeString(null);
|
||||||
|
Singleton.getInstance().setSomeInt(0);
|
||||||
|
Singleton.getInstance().setSomeString(null);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests that the similar method is working as expected.
|
* Tests that the similar method is working as expected.
|
||||||
*/
|
*/
|
||||||
@@ -210,6 +218,10 @@ public class JSONObjectTest {
|
|||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void unquotedText() {
|
public void unquotedText() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration();
|
||||||
|
if (jsonParserConfiguration.isStrictMode()) {
|
||||||
|
System.out.println("Skipping JSONObjectTest unquotedText() when strictMode default is true");
|
||||||
|
} else {
|
||||||
String str = "{key1:value1, key2:42, 1.2 : 3.4, -7e5 : something!}";
|
String str = "{key1:value1, key2:42, 1.2 : 3.4, -7e5 : something!}";
|
||||||
JSONObject jsonObject = new JSONObject(str);
|
JSONObject jsonObject = new JSONObject(str);
|
||||||
String textStr = jsonObject.toString();
|
String textStr = jsonObject.toString();
|
||||||
@@ -223,6 +235,7 @@ public class JSONObjectTest {
|
|||||||
assertTrue("expected something!", textStr.contains("\"something!\""));
|
assertTrue("expected something!", textStr.contains("\"something!\""));
|
||||||
Util.checkJSONObjectMaps(jsonObject);
|
Util.checkJSONObjectMaps(jsonObject);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testLongFromString(){
|
public void testLongFromString(){
|
||||||
@@ -786,7 +799,7 @@ public class JSONObjectTest {
|
|||||||
jsonObject.accumulate("myArray", -23.45e7);
|
jsonObject.accumulate("myArray", -23.45e7);
|
||||||
// include an unsupported object for coverage
|
// include an unsupported object for coverage
|
||||||
try {
|
try {
|
||||||
jsonObject.accumulate("myArray", NaN);
|
jsonObject.accumulate("myArray", Double.NaN);
|
||||||
fail("Expected exception");
|
fail("Expected exception");
|
||||||
} catch (JSONException ignored) {}
|
} catch (JSONException ignored) {}
|
||||||
|
|
||||||
@@ -818,7 +831,7 @@ public class JSONObjectTest {
|
|||||||
jsonObject.append("myArray", -23.45e7);
|
jsonObject.append("myArray", -23.45e7);
|
||||||
// include an unsupported object for coverage
|
// include an unsupported object for coverage
|
||||||
try {
|
try {
|
||||||
jsonObject.append("myArray", NaN);
|
jsonObject.append("myArray", Double.NaN);
|
||||||
fail("Expected exception");
|
fail("Expected exception");
|
||||||
} catch (JSONException ignored) {}
|
} catch (JSONException ignored) {}
|
||||||
|
|
||||||
@@ -843,7 +856,7 @@ public class JSONObjectTest {
|
|||||||
public void jsonObjectDoubleToString() {
|
public void jsonObjectDoubleToString() {
|
||||||
String [] expectedStrs = {"1", "1", "-23.4", "-2.345E68", "null", "null" };
|
String [] expectedStrs = {"1", "1", "-23.4", "-2.345E68", "null", "null" };
|
||||||
Double [] doubles = { 1.0, 00001.00000, -23.4, -23.45e67,
|
Double [] doubles = { 1.0, 00001.00000, -23.4, -23.45e67,
|
||||||
NaN, Double.NEGATIVE_INFINITY };
|
Double.NaN, Double.NEGATIVE_INFINITY };
|
||||||
for (int i = 0; i < expectedStrs.length; ++i) {
|
for (int i = 0; i < expectedStrs.length; ++i) {
|
||||||
String actualStr = JSONObject.doubleToString(doubles[i]);
|
String actualStr = JSONObject.doubleToString(doubles[i]);
|
||||||
assertTrue("value expected ["+expectedStrs[i]+
|
assertTrue("value expected ["+expectedStrs[i]+
|
||||||
@@ -898,11 +911,11 @@ public class JSONObjectTest {
|
|||||||
assertTrue("opt doubleKey should be double",
|
assertTrue("opt doubleKey should be double",
|
||||||
jsonObject.optDouble("doubleKey") == -23.45e7);
|
jsonObject.optDouble("doubleKey") == -23.45e7);
|
||||||
assertTrue("opt doubleKey with Default should be double",
|
assertTrue("opt doubleKey with Default should be double",
|
||||||
jsonObject.optDouble("doubleStrKey", NaN) == 1);
|
jsonObject.optDouble("doubleStrKey", Double.NaN) == 1);
|
||||||
assertTrue("opt doubleKey should be Double",
|
assertTrue("opt doubleKey should be Double",
|
||||||
Double.valueOf(-23.45e7).equals(jsonObject.optDoubleObject("doubleKey")));
|
Double.valueOf(-23.45e7).equals(jsonObject.optDoubleObject("doubleKey")));
|
||||||
assertTrue("opt doubleKey with Default should be Double",
|
assertTrue("opt doubleKey with Default should be Double",
|
||||||
Double.valueOf(1).equals(jsonObject.optDoubleObject("doubleStrKey", NaN)));
|
Double.valueOf(1).equals(jsonObject.optDoubleObject("doubleStrKey", Double.NaN)));
|
||||||
assertTrue("opt negZeroKey should be a Double",
|
assertTrue("opt negZeroKey should be a Double",
|
||||||
jsonObject.opt("negZeroKey") instanceof Double);
|
jsonObject.opt("negZeroKey") instanceof Double);
|
||||||
assertTrue("get negZeroKey should be a Double",
|
assertTrue("get negZeroKey should be a Double",
|
||||||
@@ -1061,112 +1074,54 @@ public class JSONObjectTest {
|
|||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void jsonInvalidNumberValues() {
|
public void jsonInvalidNumberValues() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration();
|
||||||
|
if (jsonParserConfiguration.isStrictMode()) {
|
||||||
|
System.out.println("Skipping JSONObjectTest jsonInvalidNumberValues() when strictMode default is true");
|
||||||
|
} else {
|
||||||
// Number-notations supported by Java and invalid as JSON
|
// Number-notations supported by Java and invalid as JSON
|
||||||
String str =
|
String str =
|
||||||
"{"+
|
"{" +
|
||||||
"\"hexNumber\":-0x123,"+
|
"\"hexNumber\":-0x123," +
|
||||||
"\"tooManyZeros\":00,"+
|
"\"tooManyZeros\":00," +
|
||||||
"\"negativeInfinite\":-Infinity,"+
|
"\"negativeInfinite\":-Infinity," +
|
||||||
"\"negativeNaN\":-NaN,"+
|
"\"negativeNaN\":-NaN," +
|
||||||
"\"negativeNaNWithLeadingZeros\":-00NaN,"+
|
"\"negativeFraction\":-.01," +
|
||||||
"\"negativeFraction\":-.01,"+
|
"\"tooManyZerosFraction\":00.001," +
|
||||||
"\"tooManyZerosFraction\":00.001,"+
|
"\"negativeHexFloat\":-0x1.fffp1," +
|
||||||
"\"negativeHexFloat\":-0x1.fffp1,"+
|
"\"hexFloat\":0x1.0P-1074," +
|
||||||
"\"hexFloat\":0x1.0P-1074,"+
|
"\"floatIdentifier\":0.1f," +
|
||||||
"\"floatIdentifier\":0.1f,"+
|
"\"doubleIdentifier\":0.1d" +
|
||||||
"\"doubleIdentifier\":0.1d,"+
|
|
||||||
"\"doubleIdentifierWithMultipleLeadingZerosBeforeDecimal\":0000000.1d,"+
|
|
||||||
"\"negativeDoubleIdentifierWithMultipleLeadingZerosBeforeDecimal\":-0000000.1d,"+
|
|
||||||
"\"doubleIdentifierWithMultipleLeadingZerosAfterDecimal\":0000000.0001d,"+
|
|
||||||
"\"negativeDoubleIdentifierWithMultipleLeadingZerosAfterDecimal\":-0000000.0001d,"+
|
|
||||||
"\"integerWithLeadingZeros\":000900,"+
|
|
||||||
"\"integerWithAllZeros\":00000,"+
|
|
||||||
"\"compositeWithLeadingZeros\":00800.90d,"+
|
|
||||||
"\"decimalPositiveWithoutNumberBeforeDecimalPoint\":.90,"+
|
|
||||||
"}";
|
"}";
|
||||||
JSONObject jsonObject = new JSONObject(str);
|
JSONObject jsonObject = new JSONObject(str);
|
||||||
Object obj;
|
Object obj;
|
||||||
obj = jsonObject.get( "hexNumber" );
|
obj = jsonObject.get("hexNumber");
|
||||||
assertFalse( "hexNumber must not be a number (should throw exception!?)",
|
assertFalse("hexNumber must not be a number (should throw exception!?)",
|
||||||
obj instanceof Number );
|
obj instanceof Number);
|
||||||
assertTrue("hexNumber currently evaluates to string",
|
assertTrue("hexNumber currently evaluates to string",
|
||||||
obj.equals("-0x123"));
|
obj.equals("-0x123"));
|
||||||
assertTrue( "tooManyZeros currently evaluates to string",
|
assertTrue("tooManyZeros currently evaluates to string",
|
||||||
jsonObject.get( "tooManyZeros" ).equals(0));
|
jsonObject.get("tooManyZeros").equals("00"));
|
||||||
obj = jsonObject.get("negativeInfinite");
|
obj = jsonObject.get("negativeInfinite");
|
||||||
assertTrue( "negativeInfinite currently evaluates to string",
|
assertTrue("negativeInfinite currently evaluates to string",
|
||||||
obj.equals("-Infinity"));
|
obj.equals("-Infinity"));
|
||||||
obj = jsonObject.get("negativeNaN");
|
obj = jsonObject.get("negativeNaN");
|
||||||
assertTrue( "negativeNaN currently evaluates to string",
|
assertTrue("negativeNaN currently evaluates to string",
|
||||||
obj.equals("-NaN"));
|
obj.equals("-NaN"));
|
||||||
obj = jsonObject.get("negativeNaNWithLeadingZeros");
|
assertTrue("negativeFraction currently evaluates to double -0.01",
|
||||||
assertTrue( "negativeNaNWithLeadingZeros currently evaluates to string",
|
jsonObject.get("negativeFraction").equals(BigDecimal.valueOf(-0.01)));
|
||||||
obj.equals("-00NaN"));
|
assertTrue("tooManyZerosFraction currently evaluates to double 0.001",
|
||||||
assertTrue( "negativeFraction currently evaluates to double -0.01",
|
jsonObject.optLong("tooManyZerosFraction") == 0);
|
||||||
jsonObject.get( "negativeFraction" ).equals(BigDecimal.valueOf(-0.01)));
|
assertTrue("negativeHexFloat currently evaluates to double -3.99951171875",
|
||||||
assertTrue( "tooManyZerosFraction currently evaluates to double 0.001",
|
jsonObject.get("negativeHexFloat").equals(Double.valueOf(-3.99951171875)));
|
||||||
jsonObject.get( "tooManyZerosFraction" ).equals(BigDecimal.valueOf(0.001)));
|
|
||||||
assertTrue( "tooManyZerosFraction currently evaluates to double 0.001",
|
|
||||||
jsonObject.getLong( "tooManyZerosFraction" )==0);
|
|
||||||
assertTrue( "tooManyZerosFraction currently evaluates to double 0.001",
|
|
||||||
jsonObject.optLong( "tooManyZerosFraction" )==0);
|
|
||||||
assertTrue( "negativeHexFloat currently evaluates to double -3.99951171875",
|
|
||||||
jsonObject.get( "negativeHexFloat" ).equals(Double.valueOf(-3.99951171875)));
|
|
||||||
assertTrue("hexFloat currently evaluates to double 4.9E-324",
|
assertTrue("hexFloat currently evaluates to double 4.9E-324",
|
||||||
jsonObject.get("hexFloat").equals(Double.valueOf(4.9E-324)));
|
jsonObject.get("hexFloat").equals(Double.valueOf(4.9E-324)));
|
||||||
assertTrue("floatIdentifier currently evaluates to double 0.1",
|
assertTrue("floatIdentifier currently evaluates to double 0.1",
|
||||||
jsonObject.get("floatIdentifier").equals(Double.valueOf(0.1)));
|
jsonObject.get("floatIdentifier").equals(Double.valueOf(0.1)));
|
||||||
assertTrue("doubleIdentifier currently evaluates to double 0.1",
|
assertTrue("doubleIdentifier currently evaluates to double 0.1",
|
||||||
jsonObject.get("doubleIdentifier").equals(Double.valueOf(0.1)));
|
jsonObject.get("doubleIdentifier").equals(Double.valueOf(0.1)));
|
||||||
assertTrue("doubleIdentifierWithMultipleLeadingZerosBeforeDecimal currently evaluates to double 0.1",
|
|
||||||
jsonObject.get("doubleIdentifierWithMultipleLeadingZerosBeforeDecimal").equals(Double.valueOf(0.1)));
|
|
||||||
assertTrue("negativeDoubleIdentifierWithMultipleLeadingZerosBeforeDecimal currently evaluates to double -0.1",
|
|
||||||
jsonObject.get("negativeDoubleIdentifierWithMultipleLeadingZerosBeforeDecimal").equals(Double.valueOf(-0.1)));
|
|
||||||
assertTrue("doubleIdentifierWithMultipleLeadingZerosAfterDecimal currently evaluates to double 0.0001",
|
|
||||||
jsonObject.get("doubleIdentifierWithMultipleLeadingZerosAfterDecimal").equals(Double.valueOf(0.0001)));
|
|
||||||
assertTrue("doubleIdentifierWithMultipleLeadingZerosAfterDecimal currently evaluates to double 0.0001",
|
|
||||||
jsonObject.get("doubleIdentifierWithMultipleLeadingZerosAfterDecimal").equals(Double.valueOf(0.0001)));
|
|
||||||
assertTrue("negativeDoubleIdentifierWithMultipleLeadingZerosAfterDecimal currently evaluates to double -0.0001",
|
|
||||||
jsonObject.get("negativeDoubleIdentifierWithMultipleLeadingZerosAfterDecimal").equals(Double.valueOf(-0.0001)));
|
|
||||||
assertTrue("Integer does not evaluate to 900",
|
|
||||||
jsonObject.get("integerWithLeadingZeros").equals(900));
|
|
||||||
assertTrue("Integer does not evaluate to 900",
|
|
||||||
jsonObject.getInt("integerWithLeadingZeros")==900);
|
|
||||||
assertTrue("Integer does not evaluate to 900",
|
|
||||||
jsonObject.optInt("integerWithLeadingZeros")==900);
|
|
||||||
assertTrue("Integer does not evaluate to 0",
|
|
||||||
jsonObject.get("integerWithAllZeros").equals(0));
|
|
||||||
assertTrue("Integer does not evaluate to 0",
|
|
||||||
jsonObject.getInt("integerWithAllZeros")==0);
|
|
||||||
assertTrue("Integer does not evaluate to 0",
|
|
||||||
jsonObject.optInt("integerWithAllZeros")==0);
|
|
||||||
assertTrue("Double does not evaluate to 800.90",
|
|
||||||
jsonObject.get("compositeWithLeadingZeros").equals(800.90));
|
|
||||||
assertTrue("Double does not evaluate to 800.90",
|
|
||||||
jsonObject.getDouble("compositeWithLeadingZeros")==800.9d);
|
|
||||||
assertTrue("Integer does not evaluate to 800",
|
|
||||||
jsonObject.optInt("compositeWithLeadingZeros")==800);
|
|
||||||
assertTrue("Long does not evaluate to 800.90",
|
|
||||||
jsonObject.getLong("compositeWithLeadingZeros")==800);
|
|
||||||
assertTrue("Long does not evaluate to 800.90",
|
|
||||||
jsonObject.optLong("compositeWithLeadingZeros")==800);
|
|
||||||
assertEquals("Get long of decimalPositiveWithoutNumberBeforeDecimalPoint does not match",
|
|
||||||
0.9d,jsonObject.getDouble("decimalPositiveWithoutNumberBeforeDecimalPoint"), 0.0d);
|
|
||||||
assertEquals("Get long of decimalPositiveWithoutNumberBeforeDecimalPoint does not match",
|
|
||||||
0.9d,jsonObject.optDouble("decimalPositiveWithoutNumberBeforeDecimalPoint"), 0.0d);
|
|
||||||
assertEquals("Get long of decimalPositiveWithoutNumberBeforeDecimalPoint does not match",
|
|
||||||
0.0d,jsonObject.optLong("decimalPositiveWithoutNumberBeforeDecimalPoint"), 0.0d);
|
|
||||||
|
|
||||||
assertEquals("Get long of doubleIdentifierWithMultipleLeadingZerosAfterDecimal does not match",
|
|
||||||
0.0001d,jsonObject.getDouble("doubleIdentifierWithMultipleLeadingZerosAfterDecimal"), 0.0d);
|
|
||||||
assertEquals("Get long of doubleIdentifierWithMultipleLeadingZerosAfterDecimal does not match",
|
|
||||||
0.0001d,jsonObject.optDouble("doubleIdentifierWithMultipleLeadingZerosAfterDecimal"), 0.0d);
|
|
||||||
assertEquals("Get long of doubleIdentifierWithMultipleLeadingZerosAfterDecimal does not match",
|
|
||||||
0.0d, jsonObject.getLong("doubleIdentifierWithMultipleLeadingZerosAfterDecimal") , 0.0d);
|
|
||||||
assertEquals("Get long of doubleIdentifierWithMultipleLeadingZerosAfterDecimal does not match",
|
|
||||||
0.0d,jsonObject.optLong("doubleIdentifierWithMultipleLeadingZerosAfterDecimal"), 0.0d);
|
|
||||||
Util.checkJSONObjectMaps(jsonObject);
|
Util.checkJSONObjectMaps(jsonObject);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests how JSONObject get[type] handles incorrect types
|
* Tests how JSONObject get[type] handles incorrect types
|
||||||
@@ -1583,7 +1538,7 @@ public class JSONObjectTest {
|
|||||||
"{"+
|
"{"+
|
||||||
"\"trueKey\":true,"+
|
"\"trueKey\":true,"+
|
||||||
"\"falseKey\":false,"+
|
"\"falseKey\":false,"+
|
||||||
"\"stringKey\":\"hello world!\","+
|
"\"stringKey\":\"hello world!\""+
|
||||||
"}";
|
"}";
|
||||||
JSONObject jsonObject2 = new JSONObject(str);
|
JSONObject jsonObject2 = new JSONObject(str);
|
||||||
names = JSONObject.getNames(jsonObject2);
|
names = JSONObject.getNames(jsonObject2);
|
||||||
@@ -1678,7 +1633,7 @@ public class JSONObjectTest {
|
|||||||
"{"+
|
"{"+
|
||||||
"\"trueKey\":true,"+
|
"\"trueKey\":true,"+
|
||||||
"\"falseKey\":false,"+
|
"\"falseKey\":false,"+
|
||||||
"\"stringKey\":\"hello world!\","+
|
"\"stringKey\":\"hello world!\""+
|
||||||
"}";
|
"}";
|
||||||
|
|
||||||
JSONObject jsonObject = new JSONObject(str);
|
JSONObject jsonObject = new JSONObject(str);
|
||||||
@@ -2257,16 +2212,74 @@ public class JSONObjectTest {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void jsonObjectParseControlCharacterEOFAssertExceptionMessage(){
|
||||||
|
char c = '\0';
|
||||||
|
final String source = "{\"key\":\"" + c + "\"}";
|
||||||
|
try {
|
||||||
|
JSONObject jo = new JSONObject(source);
|
||||||
|
fail("JSONException should be thrown");
|
||||||
|
} catch (JSONException ex) {
|
||||||
|
assertEquals("Unterminated string. " + "Character with int code 0" +
|
||||||
|
" is not allowed within a quoted string. at 8 [character 9 line 1]", ex.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void jsonObjectParseControlCharacterNewLineAssertExceptionMessage(){
|
||||||
|
char[] chars = {'\n', '\r'};
|
||||||
|
for( char c : chars) {
|
||||||
|
final String source = "{\"key\":\"" + c + "\"}";
|
||||||
|
try {
|
||||||
|
JSONObject jo = new JSONObject(source);
|
||||||
|
fail("JSONException should be thrown");
|
||||||
|
} catch (JSONException ex) {
|
||||||
|
assertEquals("Unterminated string. " + "Character with int code " + (int) c +
|
||||||
|
" is not allowed within a quoted string. at 9 [character 0 line 2]", ex.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void jsonObjectParseUTF8EncodingAssertExceptionMessage(){
|
||||||
|
String c = "\\u123x";
|
||||||
|
final String source = "{\"key\":\"" + c + "\"}";
|
||||||
|
try {
|
||||||
|
JSONObject jo = new JSONObject(source);
|
||||||
|
fail("JSONException should be thrown");
|
||||||
|
} catch (JSONException ex) {
|
||||||
|
assertEquals("Illegal escape. \\u must be followed by a 4 digit hexadecimal number. " +
|
||||||
|
"\\123x is not valid. at 14 [character 15 line 1]", ex.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void jsonObjectParseIllegalEscapeAssertExceptionMessage(){
|
||||||
|
String c = "\\x";
|
||||||
|
final String source = "{\"key\":\"" + c + "\"}";
|
||||||
|
try {
|
||||||
|
JSONObject jo = new JSONObject(source);
|
||||||
|
fail("JSONException should be thrown");
|
||||||
|
} catch (JSONException ex) {
|
||||||
|
assertEquals("Illegal escape. Escape sequence " + c + " is not valid." +
|
||||||
|
" at 10 [character 11 line 1]", ex.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Explore how JSONObject handles parsing errors.
|
* Explore how JSONObject handles parsing errors.
|
||||||
*/
|
*/
|
||||||
@SuppressWarnings({"boxing", "unused"})
|
@SuppressWarnings({"boxing", "unused"})
|
||||||
@Test
|
@Test
|
||||||
public void jsonObjectParsingErrors() {
|
public void jsonObjectParsingErrors() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration();
|
||||||
|
if (jsonParserConfiguration.isStrictMode()) {
|
||||||
|
System.out.println("Skipping JSONObjectTest jaonObjectParsingErrors() when strictMode default is true");
|
||||||
|
} else {
|
||||||
try {
|
try {
|
||||||
// does not start with '{'
|
// does not start with '{'
|
||||||
String str = "abc";
|
String str = "abc";
|
||||||
assertNull("Expected an exception",new JSONObject(str));
|
assertNull("Expected an exception", new JSONObject(str));
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expecting an exception message",
|
assertEquals("Expecting an exception message",
|
||||||
"A JSONObject text must begin with '{' at 1 [character 2 line 1]",
|
"A JSONObject text must begin with '{' at 1 [character 2 line 1]",
|
||||||
@@ -2275,7 +2288,7 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// does not end with '}'
|
// does not end with '}'
|
||||||
String str = "{";
|
String str = "{";
|
||||||
assertNull("Expected an exception",new JSONObject(str));
|
assertNull("Expected an exception", new JSONObject(str));
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expecting an exception message",
|
assertEquals("Expecting an exception message",
|
||||||
"A JSONObject text must end with '}' at 1 [character 2 line 1]",
|
"A JSONObject text must end with '}' at 1 [character 2 line 1]",
|
||||||
@@ -2284,7 +2297,7 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// key with no ':'
|
// key with no ':'
|
||||||
String str = "{\"myKey\" = true}";
|
String str = "{\"myKey\" = true}";
|
||||||
assertNull("Expected an exception",new JSONObject(str));
|
assertNull("Expected an exception", new JSONObject(str));
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expecting an exception message",
|
assertEquals("Expecting an exception message",
|
||||||
"Expected a ':' after a key at 10 [character 11 line 1]",
|
"Expected a ':' after a key at 10 [character 11 line 1]",
|
||||||
@@ -2293,7 +2306,7 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// entries with no ',' separator
|
// entries with no ',' separator
|
||||||
String str = "{\"myKey\":true \"myOtherKey\":false}";
|
String str = "{\"myKey\":true \"myOtherKey\":false}";
|
||||||
assertNull("Expected an exception",new JSONObject(str));
|
assertNull("Expected an exception", new JSONObject(str));
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expecting an exception message",
|
assertEquals("Expecting an exception message",
|
||||||
"Expected a ',' or '}' at 15 [character 16 line 1]",
|
"Expected a ',' or '}' at 15 [character 16 line 1]",
|
||||||
@@ -2302,7 +2315,7 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// key is a nested map
|
// key is a nested map
|
||||||
String str = "{{\"foo\": \"bar\"}: \"baz\"}";
|
String str = "{{\"foo\": \"bar\"}: \"baz\"}";
|
||||||
assertNull("Expected an exception",new JSONObject(str));
|
assertNull("Expected an exception", new JSONObject(str));
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expecting an exception message",
|
assertEquals("Expecting an exception message",
|
||||||
"Missing value at 1 [character 2 line 1]",
|
"Missing value at 1 [character 2 line 1]",
|
||||||
@@ -2311,7 +2324,7 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// key is a nested array containing a map
|
// key is a nested array containing a map
|
||||||
String str = "{\"a\": 1, [{\"foo\": \"bar\"}]: \"baz\"}";
|
String str = "{\"a\": 1, [{\"foo\": \"bar\"}]: \"baz\"}";
|
||||||
assertNull("Expected an exception",new JSONObject(str));
|
assertNull("Expected an exception", new JSONObject(str));
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expecting an exception message",
|
assertEquals("Expecting an exception message",
|
||||||
"Missing value at 9 [character 10 line 1]",
|
"Missing value at 9 [character 10 line 1]",
|
||||||
@@ -2320,7 +2333,7 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// key contains }
|
// key contains }
|
||||||
String str = "{foo}: 2}";
|
String str = "{foo}: 2}";
|
||||||
assertNull("Expected an exception",new JSONObject(str));
|
assertNull("Expected an exception", new JSONObject(str));
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expecting an exception message",
|
assertEquals("Expecting an exception message",
|
||||||
"Expected a ':' after a key at 5 [character 6 line 1]",
|
"Expected a ':' after a key at 5 [character 6 line 1]",
|
||||||
@@ -2329,7 +2342,7 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// key contains ]
|
// key contains ]
|
||||||
String str = "{foo]: 2}";
|
String str = "{foo]: 2}";
|
||||||
assertNull("Expected an exception",new JSONObject(str));
|
assertNull("Expected an exception", new JSONObject(str));
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expecting an exception message",
|
assertEquals("Expecting an exception message",
|
||||||
"Expected a ':' after a key at 5 [character 6 line 1]",
|
"Expected a ':' after a key at 5 [character 6 line 1]",
|
||||||
@@ -2338,7 +2351,7 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// \0 after ,
|
// \0 after ,
|
||||||
String str = "{\"myKey\":true, \0\"myOtherKey\":false}";
|
String str = "{\"myKey\":true, \0\"myOtherKey\":false}";
|
||||||
assertNull("Expected an exception",new JSONObject(str));
|
assertNull("Expected an exception", new JSONObject(str));
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expecting an exception message",
|
assertEquals("Expecting an exception message",
|
||||||
"A JSONObject text must end with '}' at 15 [character 16 line 1]",
|
"A JSONObject text must end with '}' at 15 [character 16 line 1]",
|
||||||
@@ -2379,7 +2392,7 @@ public class JSONObjectTest {
|
|||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
// invalid numberToString()
|
// invalid numberToString()
|
||||||
JSONObject.numberToString((Number)null);
|
JSONObject.numberToString((Number) null);
|
||||||
fail("Expected an exception");
|
fail("Expected an exception");
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertEquals("Expecting an exception message",
|
assertEquals("Expecting an exception message",
|
||||||
@@ -2398,7 +2411,7 @@ public class JSONObjectTest {
|
|||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
// test validity of invalid double
|
// test validity of invalid double
|
||||||
JSONObject.testValidity(NaN);
|
JSONObject.testValidity(Double.NaN);
|
||||||
fail("Expected an exception");
|
fail("Expected an exception");
|
||||||
} catch (JSONException e) {
|
} catch (JSONException e) {
|
||||||
assertTrue("", true);
|
assertTrue("", true);
|
||||||
@@ -2413,10 +2426,10 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// test exception message when including a duplicate key (level 0)
|
// test exception message when including a duplicate key (level 0)
|
||||||
String str = "{\n"
|
String str = "{\n"
|
||||||
+" \"attr01\":\"value-01\",\n"
|
+ " \"attr01\":\"value-01\",\n"
|
||||||
+" \"attr02\":\"value-02\",\n"
|
+ " \"attr02\":\"value-02\",\n"
|
||||||
+" \"attr03\":\"value-03\",\n"
|
+ " \"attr03\":\"value-03\",\n"
|
||||||
+" \"attr03\":\"value-04\"\n"
|
+ " \"attr03\":\"value-04\"\n"
|
||||||
+ "}";
|
+ "}";
|
||||||
new JSONObject(str);
|
new JSONObject(str);
|
||||||
fail("Expected an exception");
|
fail("Expected an exception");
|
||||||
@@ -2428,13 +2441,13 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// test exception message when including a duplicate key (level 0) holding an object
|
// test exception message when including a duplicate key (level 0) holding an object
|
||||||
String str = "{\n"
|
String str = "{\n"
|
||||||
+" \"attr01\":\"value-01\",\n"
|
+ " \"attr01\":\"value-01\",\n"
|
||||||
+" \"attr02\":\"value-02\",\n"
|
+ " \"attr02\":\"value-02\",\n"
|
||||||
+" \"attr03\":\"value-03\",\n"
|
+ " \"attr03\":\"value-03\",\n"
|
||||||
+" \"attr03\": {"
|
+ " \"attr03\": {"
|
||||||
+" \"attr04-01\":\"value-04-01\",n"
|
+ " \"attr04-01\":\"value-04-01\",n"
|
||||||
+" \"attr04-02\":\"value-04-02\",n"
|
+ " \"attr04-02\":\"value-04-02\",n"
|
||||||
+" \"attr04-03\":\"value-04-03\"n"
|
+ " \"attr04-03\":\"value-04-03\"n"
|
||||||
+ " }\n"
|
+ " }\n"
|
||||||
+ "}";
|
+ "}";
|
||||||
new JSONObject(str);
|
new JSONObject(str);
|
||||||
@@ -2447,15 +2460,15 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// test exception message when including a duplicate key (level 0) holding an array
|
// test exception message when including a duplicate key (level 0) holding an array
|
||||||
String str = "{\n"
|
String str = "{\n"
|
||||||
+" \"attr01\":\"value-01\",\n"
|
+ " \"attr01\":\"value-01\",\n"
|
||||||
+" \"attr02\":\"value-02\",\n"
|
+ " \"attr02\":\"value-02\",\n"
|
||||||
+" \"attr03\":\"value-03\",\n"
|
+ " \"attr03\":\"value-03\",\n"
|
||||||
+" \"attr03\": [\n"
|
+ " \"attr03\": [\n"
|
||||||
+" {"
|
+ " {"
|
||||||
+" \"attr04-01\":\"value-04-01\",n"
|
+ " \"attr04-01\":\"value-04-01\",n"
|
||||||
+" \"attr04-02\":\"value-04-02\",n"
|
+ " \"attr04-02\":\"value-04-02\",n"
|
||||||
+" \"attr04-03\":\"value-04-03\"n"
|
+ " \"attr04-03\":\"value-04-03\"n"
|
||||||
+" }\n"
|
+ " }\n"
|
||||||
+ " ]\n"
|
+ " ]\n"
|
||||||
+ "}";
|
+ "}";
|
||||||
new JSONObject(str);
|
new JSONObject(str);
|
||||||
@@ -2468,14 +2481,14 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// test exception message when including a duplicate key (level 1)
|
// test exception message when including a duplicate key (level 1)
|
||||||
String str = "{\n"
|
String str = "{\n"
|
||||||
+" \"attr01\":\"value-01\",\n"
|
+ " \"attr01\":\"value-01\",\n"
|
||||||
+" \"attr02\":\"value-02\",\n"
|
+ " \"attr02\":\"value-02\",\n"
|
||||||
+" \"attr03\":\"value-03\",\n"
|
+ " \"attr03\":\"value-03\",\n"
|
||||||
+" \"attr04\": {\n"
|
+ " \"attr04\": {\n"
|
||||||
+" \"attr04-01\":\"value04-01\",\n"
|
+ " \"attr04-01\":\"value04-01\",\n"
|
||||||
+" \"attr04-02\":\"value04-02\",\n"
|
+ " \"attr04-02\":\"value04-02\",\n"
|
||||||
+" \"attr04-03\":\"value04-03\",\n"
|
+ " \"attr04-03\":\"value04-03\",\n"
|
||||||
+" \"attr04-03\":\"value04-04\"\n"
|
+ " \"attr04-03\":\"value04-04\"\n"
|
||||||
+ " }\n"
|
+ " }\n"
|
||||||
+ "}";
|
+ "}";
|
||||||
new JSONObject(str);
|
new JSONObject(str);
|
||||||
@@ -2488,19 +2501,19 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// test exception message when including a duplicate key (level 1) holding an object
|
// test exception message when including a duplicate key (level 1) holding an object
|
||||||
String str = "{\n"
|
String str = "{\n"
|
||||||
+" \"attr01\":\"value-01\",\n"
|
+ " \"attr01\":\"value-01\",\n"
|
||||||
+" \"attr02\":\"value-02\",\n"
|
+ " \"attr02\":\"value-02\",\n"
|
||||||
+" \"attr03\":\"value-03\",\n"
|
+ " \"attr03\":\"value-03\",\n"
|
||||||
+" \"attr04\": {\n"
|
+ " \"attr04\": {\n"
|
||||||
+" \"attr04-01\":\"value04-01\",\n"
|
+ " \"attr04-01\":\"value04-01\",\n"
|
||||||
+" \"attr04-02\":\"value04-02\",\n"
|
+ " \"attr04-02\":\"value04-02\",\n"
|
||||||
+" \"attr04-03\":\"value04-03\",\n"
|
+ " \"attr04-03\":\"value04-03\",\n"
|
||||||
+" \"attr04-03\": {\n"
|
+ " \"attr04-03\": {\n"
|
||||||
+" \"attr04-04-01\":\"value04-04-01\",\n"
|
+ " \"attr04-04-01\":\"value04-04-01\",\n"
|
||||||
+" \"attr04-04-02\":\"value04-04-02\",\n"
|
+ " \"attr04-04-02\":\"value04-04-02\",\n"
|
||||||
+" \"attr04-04-03\":\"value04-04-03\",\n"
|
+ " \"attr04-04-03\":\"value04-04-03\",\n"
|
||||||
+" }\n"
|
+ " }\n"
|
||||||
+" }\n"
|
+ " }\n"
|
||||||
+ "}";
|
+ "}";
|
||||||
new JSONObject(str);
|
new JSONObject(str);
|
||||||
fail("Expected an exception");
|
fail("Expected an exception");
|
||||||
@@ -2512,21 +2525,21 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// test exception message when including a duplicate key (level 1) holding an array
|
// test exception message when including a duplicate key (level 1) holding an array
|
||||||
String str = "{\n"
|
String str = "{\n"
|
||||||
+" \"attr01\":\"value-01\",\n"
|
+ " \"attr01\":\"value-01\",\n"
|
||||||
+" \"attr02\":\"value-02\",\n"
|
+ " \"attr02\":\"value-02\",\n"
|
||||||
+" \"attr03\":\"value-03\",\n"
|
+ " \"attr03\":\"value-03\",\n"
|
||||||
+" \"attr04\": {\n"
|
+ " \"attr04\": {\n"
|
||||||
+" \"attr04-01\":\"value04-01\",\n"
|
+ " \"attr04-01\":\"value04-01\",\n"
|
||||||
+" \"attr04-02\":\"value04-02\",\n"
|
+ " \"attr04-02\":\"value04-02\",\n"
|
||||||
+" \"attr04-03\":\"value04-03\",\n"
|
+ " \"attr04-03\":\"value04-03\",\n"
|
||||||
+" \"attr04-03\": [\n"
|
+ " \"attr04-03\": [\n"
|
||||||
+" {\n"
|
+ " {\n"
|
||||||
+" \"attr04-04-01\":\"value04-04-01\",\n"
|
+ " \"attr04-04-01\":\"value04-04-01\",\n"
|
||||||
+" \"attr04-04-02\":\"value04-04-02\",\n"
|
+ " \"attr04-04-02\":\"value04-04-02\",\n"
|
||||||
+" \"attr04-04-03\":\"value04-04-03\",\n"
|
+ " \"attr04-04-03\":\"value04-04-03\",\n"
|
||||||
+" }\n"
|
+ " }\n"
|
||||||
+" ]\n"
|
+ " ]\n"
|
||||||
+" }\n"
|
+ " }\n"
|
||||||
+ "}";
|
+ "}";
|
||||||
new JSONObject(str);
|
new JSONObject(str);
|
||||||
fail("Expected an exception");
|
fail("Expected an exception");
|
||||||
@@ -2538,14 +2551,14 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// test exception message when including a duplicate key in object (level 0) within an array
|
// test exception message when including a duplicate key in object (level 0) within an array
|
||||||
String str = "[\n"
|
String str = "[\n"
|
||||||
+" {\n"
|
+ " {\n"
|
||||||
+" \"attr01\":\"value-01\",\n"
|
+ " \"attr01\":\"value-01\",\n"
|
||||||
+" \"attr02\":\"value-02\"\n"
|
+ " \"attr02\":\"value-02\"\n"
|
||||||
+" },\n"
|
+ " },\n"
|
||||||
+" {\n"
|
+ " {\n"
|
||||||
+" \"attr01\":\"value-01\",\n"
|
+ " \"attr01\":\"value-01\",\n"
|
||||||
+" \"attr01\":\"value-02\"\n"
|
+ " \"attr01\":\"value-02\"\n"
|
||||||
+" }\n"
|
+ " }\n"
|
||||||
+ "]";
|
+ "]";
|
||||||
new JSONArray(str);
|
new JSONArray(str);
|
||||||
fail("Expected an exception");
|
fail("Expected an exception");
|
||||||
@@ -2557,20 +2570,20 @@ public class JSONObjectTest {
|
|||||||
try {
|
try {
|
||||||
// test exception message when including a duplicate key in object (level 1) within an array
|
// test exception message when including a duplicate key in object (level 1) within an array
|
||||||
String str = "[\n"
|
String str = "[\n"
|
||||||
+" {\n"
|
+ " {\n"
|
||||||
+" \"attr01\":\"value-01\",\n"
|
+ " \"attr01\":\"value-01\",\n"
|
||||||
+" \"attr02\": {\n"
|
+ " \"attr02\": {\n"
|
||||||
+" \"attr02-01\":\"value-02-01\",\n"
|
+ " \"attr02-01\":\"value-02-01\",\n"
|
||||||
+" \"attr02-02\":\"value-02-02\"\n"
|
+ " \"attr02-02\":\"value-02-02\"\n"
|
||||||
+" }\n"
|
+ " }\n"
|
||||||
+" },\n"
|
+ " },\n"
|
||||||
+" {\n"
|
+ " {\n"
|
||||||
+" \"attr01\":\"value-01\",\n"
|
+ " \"attr01\":\"value-01\",\n"
|
||||||
+" \"attr02\": {\n"
|
+ " \"attr02\": {\n"
|
||||||
+" \"attr02-01\":\"value-02-01\",\n"
|
+ " \"attr02-01\":\"value-02-01\",\n"
|
||||||
+" \"attr02-01\":\"value-02-02\"\n"
|
+ " \"attr02-01\":\"value-02-02\"\n"
|
||||||
+" }\n"
|
+ " }\n"
|
||||||
+" }\n"
|
+ " }\n"
|
||||||
+ "]";
|
+ "]";
|
||||||
new JSONArray(str);
|
new JSONArray(str);
|
||||||
fail("Expected an exception");
|
fail("Expected an exception");
|
||||||
@@ -2580,6 +2593,7 @@ public class JSONObjectTest {
|
|||||||
e.getMessage());
|
e.getMessage());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Confirm behavior when putOnce() is called with null parameters
|
* Confirm behavior when putOnce() is called with null parameters
|
||||||
@@ -3802,6 +3816,43 @@ public class JSONObjectTest {
|
|||||||
new JSONObject(map1);
|
new JSONObject(map1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void clarifyCurrentBehavior() {
|
||||||
|
// Behavior documented in #653 optLong vs getLong inconsistencies
|
||||||
|
// This problem still exists.
|
||||||
|
// Internally, both number_1 and number_2 are stored as strings. This is reasonable since they are parsed as strings.
|
||||||
|
// However, getLong and optLong should return similar results
|
||||||
|
JSONObject json = new JSONObject("{\"number_1\":\"01234\", \"number_2\": \"332211\"}");
|
||||||
|
assertEquals(json.getLong("number_1"), 1234L);
|
||||||
|
assertEquals(json.optLong("number_1"), 0); //THIS VALUE IS NOT RETURNED AS A NUMBER
|
||||||
|
assertEquals(json.getLong("number_2"), 332211L);
|
||||||
|
assertEquals(json.optLong("number_2"), 332211L);
|
||||||
|
|
||||||
|
// Behavior documented in #826 JSONObject parsing 0-led numeric strings as ints
|
||||||
|
// After reverting the code, personId is stored as a string, and the behavior is as expected
|
||||||
|
String personId = "\"0123\"";
|
||||||
|
JSONObject j1 = new JSONObject("{\"personId\": " + personId + "}");
|
||||||
|
assertEquals(j1.getString("personId"), "0123");
|
||||||
|
|
||||||
|
// Also #826. Here is input with missing quotes. Because of the leading zero, it should not be parsed as a number.
|
||||||
|
// This example was mentioned in the same ticket
|
||||||
|
// After reverting the code, personId is stored as a string, and the behavior is as expected
|
||||||
|
JSONObject j2 = new JSONObject("{\"personId\":\"0123\"}");
|
||||||
|
assertEquals(j2.getString("personId"), "0123");
|
||||||
|
|
||||||
|
// Behavior uncovered while working on the code
|
||||||
|
// All of the values are stored as strings except for hex4, which is stored as a number. This is probably incorrect
|
||||||
|
JSONObject j3 = new JSONObject("{ " +
|
||||||
|
"\"hex1\": \"010e4\", \"hex2\": \"00f0\", \"hex3\": \"0011\", " +
|
||||||
|
"\"hex4\": 00e0, \"hex5\": \"00f0\", \"hex6\": \"0011\" }");
|
||||||
|
assertEquals(j3.getString("hex1"), "010e4");
|
||||||
|
assertEquals(j3.getString("hex2"), "00f0");
|
||||||
|
assertEquals(j3.getString("hex3"), "0011");
|
||||||
|
assertEquals(j3.getLong("hex4"), 0, .1);
|
||||||
|
assertEquals(j3.getString("hex5"), "00f0");
|
||||||
|
assertEquals(j3.getString("hex6"), "0011");
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Method to build nested map of max maxDepth
|
* Method to build nested map of max maxDepth
|
||||||
*
|
*
|
||||||
|
|||||||
560
src/test/java/org/json/junit/JSONParserConfigurationTest.java
Normal file
560
src/test/java/org/json/junit/JSONParserConfigurationTest.java
Normal file
@@ -0,0 +1,560 @@
|
|||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
import org.json.JSONArray;
|
||||||
|
import org.json.JSONException;
|
||||||
|
import org.json.JSONObject;
|
||||||
|
import org.json.JSONParserConfiguration;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.nio.file.Files;
|
||||||
|
import java.nio.file.Paths;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
|
public class JSONParserConfigurationTest {
|
||||||
|
private static final String TEST_SOURCE = "{\"key\": \"value1\", \"key\": \"value2\"}";
|
||||||
|
|
||||||
|
@Test(expected = JSONException.class)
|
||||||
|
public void testThrowException() {
|
||||||
|
new JSONObject(TEST_SOURCE);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testOverwrite() {
|
||||||
|
JSONObject jsonObject = new JSONObject(TEST_SOURCE,
|
||||||
|
new JSONParserConfiguration().withOverwriteDuplicateKey(true));
|
||||||
|
|
||||||
|
assertEquals("duplicate key should be overwritten", "value2", jsonObject.getString("key"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void verifyDuplicateKeyThenMaxDepth() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withOverwriteDuplicateKey(true)
|
||||||
|
.withMaxNestingDepth(42);
|
||||||
|
|
||||||
|
assertEquals(42, jsonParserConfiguration.getMaxNestingDepth());
|
||||||
|
assertTrue(jsonParserConfiguration.isOverwriteDuplicateKey());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void verifyMaxDepthThenDuplicateKey() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withMaxNestingDepth(42)
|
||||||
|
.withOverwriteDuplicateKey(true);
|
||||||
|
|
||||||
|
assertTrue(jsonParserConfiguration.isOverwriteDuplicateKey());
|
||||||
|
assertEquals(42, jsonParserConfiguration.getMaxNestingDepth());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidInput_testStrictModeTrue_shouldThrowJsonException() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
List<String> strictModeInputTestCases = getNonCompliantJSONArrayList();
|
||||||
|
// this is a lot easier to debug when things stop working
|
||||||
|
for (int i = 0; i < strictModeInputTestCases.size(); ++i) {
|
||||||
|
String testCase = strictModeInputTestCases.get(i);
|
||||||
|
try {
|
||||||
|
JSONArray jsonArray = new JSONArray(testCase, jsonParserConfiguration);
|
||||||
|
String s = jsonArray.toString();
|
||||||
|
String msg = "Expected an exception, but got: " + s + " Noncompliant Array index: " + i;
|
||||||
|
fail(msg);
|
||||||
|
} catch (Exception e) {
|
||||||
|
// its all good
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidInputObjects_testStrictModeTrue_shouldThrowJsonException() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
List<String> strictModeInputTestCases = getNonCompliantJSONObjectList();
|
||||||
|
// this is a lot easier to debug when things stop working
|
||||||
|
for (int i = 0; i < strictModeInputTestCases.size(); ++i) {
|
||||||
|
String testCase = strictModeInputTestCases.get(i);
|
||||||
|
try {
|
||||||
|
JSONObject jsonObject = new JSONObject(testCase, jsonParserConfiguration);
|
||||||
|
String s = jsonObject.toString();
|
||||||
|
String msg = "Expected an exception, but got: " + s + " Noncompliant Array index: " + i;
|
||||||
|
fail(msg);
|
||||||
|
} catch (Exception e) {
|
||||||
|
// its all good
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenEmptyArray_testStrictModeTrue_shouldNotThrowJsonException() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
String testCase = "[]";
|
||||||
|
JSONArray jsonArray = new JSONArray(testCase, jsonParserConfiguration);
|
||||||
|
assertEquals(testCase, jsonArray.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenEmptyObject_testStrictModeTrue_shouldNotThrowJsonException() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
String testCase = "{}";
|
||||||
|
JSONObject jsonObject = new JSONObject(testCase, jsonParserConfiguration);
|
||||||
|
assertEquals(testCase, jsonObject.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenValidNestedArray_testStrictModeTrue_shouldNotThrowJsonException() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
|
||||||
|
String testCase = "[[\"c\"], [10.2], [true, false, true]]";
|
||||||
|
|
||||||
|
JSONArray jsonArray = new JSONArray(testCase, jsonParserConfiguration);
|
||||||
|
JSONArray arrayShouldContainStringAt0 = jsonArray.getJSONArray(0);
|
||||||
|
JSONArray arrayShouldContainNumberAt0 = jsonArray.getJSONArray(1);
|
||||||
|
JSONArray arrayShouldContainBooleanAt0 = jsonArray.getJSONArray(2);
|
||||||
|
|
||||||
|
assertTrue(arrayShouldContainStringAt0.get(0) instanceof String);
|
||||||
|
assertTrue(arrayShouldContainNumberAt0.get(0) instanceof Number);
|
||||||
|
assertTrue(arrayShouldContainBooleanAt0.get(0) instanceof Boolean);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenValidNestedObject_testStrictModeTrue_shouldNotThrowJsonException() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
|
||||||
|
String testCase = "{\"a0\":[\"c\"], \"a1\":[10.2], \"a2\":[true, false, true]}";
|
||||||
|
|
||||||
|
JSONObject jsonObject = new JSONObject(testCase, jsonParserConfiguration);
|
||||||
|
JSONArray arrayShouldContainStringAt0 = jsonObject.getJSONArray("a0");
|
||||||
|
JSONArray arrayShouldContainNumberAt0 = jsonObject.getJSONArray("a1");
|
||||||
|
JSONArray arrayShouldContainBooleanAt0 = jsonObject.getJSONArray("a2");
|
||||||
|
|
||||||
|
assertTrue(arrayShouldContainStringAt0.get(0) instanceof String);
|
||||||
|
assertTrue(arrayShouldContainNumberAt0.get(0) instanceof Number);
|
||||||
|
assertTrue(arrayShouldContainBooleanAt0.get(0) instanceof Boolean);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenValidEmptyArrayInsideArray_testStrictModeTrue_shouldNotThrowJsonException(){
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
String testCase = "[[]]";
|
||||||
|
JSONArray jsonArray = new JSONArray(testCase, jsonParserConfiguration);
|
||||||
|
assertEquals(testCase, jsonArray.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenValidEmptyArrayInsideObject_testStrictModeTrue_shouldNotThrowJsonException(){
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
String testCase = "{\"a0\":[]}";
|
||||||
|
JSONObject jsonObject = new JSONObject(testCase, jsonParserConfiguration);
|
||||||
|
assertEquals(testCase, jsonObject.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenValidEmptyArrayInsideArray_testStrictModeFalse_shouldNotThrowJsonException(){
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(false);
|
||||||
|
String testCase = "[[]]";
|
||||||
|
JSONArray jsonArray = new JSONArray(testCase, jsonParserConfiguration);
|
||||||
|
assertEquals(testCase, jsonArray.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenValidEmptyArrayInsideObject_testStrictModeFalse_shouldNotThrowJsonException(){
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(false);
|
||||||
|
String testCase = "{\"a0\":[]}";
|
||||||
|
JSONObject jsonObject = new JSONObject(testCase, jsonParserConfiguration);
|
||||||
|
assertEquals(testCase, jsonObject.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidStringArray_testStrictModeTrue_shouldThrowJsonException() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
String testCase = "[badString]";
|
||||||
|
JSONException je = assertThrows(JSONException.class, () -> new JSONArray(testCase, jsonParserConfiguration));
|
||||||
|
assertEquals("Strict mode error: Value 'badString' is not surrounded by quotes at 10 [character 11 line 1]",
|
||||||
|
je.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidStringObject_testStrictModeTrue_shouldThrowJsonException() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
String testCase = "{\"a0\":badString}";
|
||||||
|
JSONException je = assertThrows(JSONException.class, () -> new JSONObject(testCase, jsonParserConfiguration));
|
||||||
|
assertEquals("Strict mode error: Value 'badString' is not surrounded by quotes at 15 [character 16 line 1]",
|
||||||
|
je.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void allowNullArrayInStrictMode() {
|
||||||
|
String expected = "[null]";
|
||||||
|
JSONArray jsonArray = new JSONArray(expected, new JSONParserConfiguration().withStrictMode(true));
|
||||||
|
assertEquals(expected, jsonArray.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void allowNullObjectInStrictMode() {
|
||||||
|
String expected = "{\"a0\":null}";
|
||||||
|
JSONObject jsonObject = new JSONObject(expected, new JSONParserConfiguration().withStrictMode(true));
|
||||||
|
assertEquals(expected, jsonObject.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNumericArray() {
|
||||||
|
String expected = "[10]";
|
||||||
|
JSONArray jsonArray = new JSONArray(expected, new JSONParserConfiguration().withStrictMode(true));
|
||||||
|
assertEquals(expected, jsonArray.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNumericObject() {
|
||||||
|
String expected = "{\"a0\":10}";
|
||||||
|
JSONObject jsonObject = new JSONObject(expected, new JSONParserConfiguration().withStrictMode(true));
|
||||||
|
assertEquals(expected, jsonObject.toString());
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void givenCompliantJSONArrayFile_testStrictModeTrue_shouldNotThrowAnyException() throws IOException {
|
||||||
|
try (Stream<String> lines = Files.lines(Paths.get("src/test/resources/compliantJsonArray.json"))) {
|
||||||
|
String compliantJsonArrayAsString = lines.collect(Collectors.joining());
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
new JSONArray(compliantJsonArrayAsString, jsonParserConfiguration);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenCompliantJSONObjectFile_testStrictModeTrue_shouldNotThrowAnyException() throws IOException {
|
||||||
|
try (Stream<String> lines = Files.lines(Paths.get("src/test/resources/compliantJsonObject.json"))) {
|
||||||
|
String compliantJsonObjectAsString = lines.collect(Collectors.joining());
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
|
||||||
|
new JSONObject(compliantJsonObjectAsString, jsonParserConfiguration);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidInputArrays_testStrictModeFalse_shouldNotThrowAnyException() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(false);
|
||||||
|
|
||||||
|
List<String> strictModeInputTestCases = getNonCompliantJSONArrayList();
|
||||||
|
|
||||||
|
// this is a lot easier to debug when things stop working
|
||||||
|
for (int i = 0; i < strictModeInputTestCases.size(); ++i) {
|
||||||
|
String testCase = strictModeInputTestCases.get(i);
|
||||||
|
try {
|
||||||
|
JSONArray jsonArray = new JSONArray(testCase, jsonParserConfiguration);
|
||||||
|
} catch (Exception e) {
|
||||||
|
System.out.println("Unexpected exception: " + e.getMessage() + " Noncompliant Array index: " + i);
|
||||||
|
fail(String.format("Noncompliant array index: %d", i));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidInputObjects_testStrictModeFalse_shouldNotThrowAnyException() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(false);
|
||||||
|
|
||||||
|
List<String> strictModeInputTestCases = getNonCompliantJSONObjectList();
|
||||||
|
|
||||||
|
// this is a lot easier to debug when things stop working
|
||||||
|
for (int i = 0; i < strictModeInputTestCases.size(); ++i) {
|
||||||
|
String testCase = strictModeInputTestCases.get(i);
|
||||||
|
try {
|
||||||
|
JSONObject jsonObject = new JSONObject(testCase, jsonParserConfiguration);
|
||||||
|
} catch (Exception e) {
|
||||||
|
System.out.println("Unexpected exception: " + e.getMessage() + " Noncompliant Array index: " + i);
|
||||||
|
fail(String.format("Noncompliant array index: %d", i));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidInputArray_testStrictModeTrue_shouldThrowInvalidCharacterErrorMessage() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
String testCase = "[1,2];[3,4]";
|
||||||
|
JSONException je = assertThrows("expected non-compliant array but got instead: " + testCase,
|
||||||
|
JSONException.class, () -> new JSONArray(testCase, jsonParserConfiguration));
|
||||||
|
assertEquals("Strict mode error: Unparsed characters found at end of input text at 6 [character 7 line 1]",
|
||||||
|
je.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidInputObject_testStrictModeTrue_shouldThrowInvalidCharacterErrorMessage() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
String testCase = "{\"a0\":[1,2];\"a1\":[3,4]}";
|
||||||
|
JSONException je = assertThrows("expected non-compliant array but got instead: " + testCase,
|
||||||
|
JSONException.class, () -> new JSONObject(testCase, jsonParserConfiguration));
|
||||||
|
assertEquals("Strict mode error: Invalid character ';' found at 12 [character 13 line 1]", je.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidInputArrayWithNumericStrings_testStrictModeTrue_shouldThrowInvalidCharacterErrorMessage() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
String testCase = "[\"1\",\"2\"];[3,4]";
|
||||||
|
JSONException je = assertThrows("expected non-compliant array but got instead: " + testCase,
|
||||||
|
JSONException.class, () -> new JSONArray(testCase, jsonParserConfiguration));
|
||||||
|
assertEquals("Strict mode error: Unparsed characters found at end of input text at 10 [character 11 line 1]",
|
||||||
|
je.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidInputObjectWithNumericStrings_testStrictModeTrue_shouldThrowInvalidCharacterErrorMessage() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
String testCase = "{\"a0\":[\"1\",\"2\"];\"a1\":[3,4]}";
|
||||||
|
JSONException je = assertThrows("expected non-compliant array but got instead: " + testCase,
|
||||||
|
JSONException.class, () -> new JSONObject(testCase, jsonParserConfiguration));
|
||||||
|
assertEquals("Strict mode error: Invalid character ';' found at 16 [character 17 line 1]", je.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidInputArray_testStrictModeTrue_shouldThrowValueNotSurroundedByQuotesErrorMessage() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
String testCase = "[{\"test\": implied}]";
|
||||||
|
JSONException je = assertThrows("expected non-compliant array but got instead: " + testCase,
|
||||||
|
JSONException.class, () -> new JSONArray(testCase, jsonParserConfiguration));
|
||||||
|
assertEquals("Strict mode error: Value 'implied' is not surrounded by quotes at 17 [character 18 line 1]",
|
||||||
|
je.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidInputObject_testStrictModeTrue_shouldThrowValueNotSurroundedByQuotesErrorMessage() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
String testCase = "{\"a0\":{\"test\": implied}]}";
|
||||||
|
JSONException je = assertThrows("expected non-compliant array but got instead: " + testCase,
|
||||||
|
JSONException.class, () -> new JSONObject(testCase, jsonParserConfiguration));
|
||||||
|
assertEquals("Strict mode error: Value 'implied' is not surrounded by quotes at 22 [character 23 line 1]",
|
||||||
|
je.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidInputArray_testStrictModeFalse_shouldNotThrowAnyException() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(false);
|
||||||
|
String testCase = "[{\"test\": implied}]";
|
||||||
|
new JSONArray(testCase, jsonParserConfiguration);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidInputObject_testStrictModeFalse_shouldNotThrowAnyException() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(false);
|
||||||
|
String testCase = "{\"a0\":{\"test\": implied}}";
|
||||||
|
new JSONObject(testCase, jsonParserConfiguration);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenNonCompliantQuotesArray_testStrictModeTrue_shouldThrowJsonExceptionWithConcreteErrorDescription() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
|
||||||
|
String testCaseOne = "[\"abc', \"test\"]";
|
||||||
|
String testCaseTwo = "['abc\", \"test\"]";
|
||||||
|
String testCaseThree = "['abc']";
|
||||||
|
String testCaseFour = "[{'testField': \"testValue\"}]";
|
||||||
|
|
||||||
|
JSONException jeOne = assertThrows(JSONException.class,
|
||||||
|
() -> new JSONArray(testCaseOne, jsonParserConfiguration));
|
||||||
|
JSONException jeTwo = assertThrows(JSONException.class,
|
||||||
|
() -> new JSONArray(testCaseTwo, jsonParserConfiguration));
|
||||||
|
JSONException jeThree = assertThrows(JSONException.class,
|
||||||
|
() -> new JSONArray(testCaseThree, jsonParserConfiguration));
|
||||||
|
JSONException jeFour = assertThrows(JSONException.class,
|
||||||
|
() -> new JSONArray(testCaseFour, jsonParserConfiguration));
|
||||||
|
|
||||||
|
assertEquals(
|
||||||
|
"Expected a ',' or ']' at 10 [character 11 line 1]",
|
||||||
|
jeOne.getMessage());
|
||||||
|
assertEquals(
|
||||||
|
"Strict mode error: Single quoted strings are not allowed at 2 [character 3 line 1]",
|
||||||
|
jeTwo.getMessage());
|
||||||
|
assertEquals(
|
||||||
|
"Strict mode error: Single quoted strings are not allowed at 2 [character 3 line 1]",
|
||||||
|
jeThree.getMessage());
|
||||||
|
assertEquals(
|
||||||
|
"Strict mode error: Single quoted strings are not allowed at 3 [character 4 line 1]",
|
||||||
|
jeFour.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenNonCompliantQuotesObject_testStrictModeTrue_shouldThrowJsonExceptionWithConcreteErrorDescription() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
|
||||||
|
String testCaseOne = "{\"abc': \"test\"}";
|
||||||
|
String testCaseTwo = "{'abc\": \"test\"}";
|
||||||
|
String testCaseThree = "{\"a\":'abc'}";
|
||||||
|
String testCaseFour = "{'testField': \"testValue\"}";
|
||||||
|
|
||||||
|
JSONException jeOne = assertThrows(JSONException.class,
|
||||||
|
() -> new JSONObject(testCaseOne, jsonParserConfiguration));
|
||||||
|
JSONException jeTwo = assertThrows(JSONException.class,
|
||||||
|
() -> new JSONObject(testCaseTwo, jsonParserConfiguration));
|
||||||
|
JSONException jeThree = assertThrows(JSONException.class,
|
||||||
|
() -> new JSONObject(testCaseThree, jsonParserConfiguration));
|
||||||
|
JSONException jeFour = assertThrows(JSONException.class,
|
||||||
|
() -> new JSONObject(testCaseFour, jsonParserConfiguration));
|
||||||
|
|
||||||
|
assertEquals(
|
||||||
|
"Expected a ':' after a key at 10 [character 11 line 1]",
|
||||||
|
jeOne.getMessage());
|
||||||
|
assertEquals(
|
||||||
|
"Strict mode error: Single quoted strings are not allowed at 2 [character 3 line 1]",
|
||||||
|
jeTwo.getMessage());
|
||||||
|
assertEquals(
|
||||||
|
"Strict mode error: Single quoted strings are not allowed at 6 [character 7 line 1]",
|
||||||
|
jeThree.getMessage());
|
||||||
|
assertEquals(
|
||||||
|
"Strict mode error: Single quoted strings are not allowed at 2 [character 3 line 1]",
|
||||||
|
jeFour.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenUnbalancedQuotesArray_testStrictModeFalse_shouldThrowJsonException() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(false);
|
||||||
|
|
||||||
|
String testCaseOne = "[\"abc', \"test\"]";
|
||||||
|
String testCaseTwo = "['abc\", \"test\"]";
|
||||||
|
|
||||||
|
JSONException jeOne = assertThrows(JSONException.class,
|
||||||
|
() -> new JSONArray(testCaseOne, jsonParserConfiguration));
|
||||||
|
JSONException jeTwo = assertThrows(JSONException.class,
|
||||||
|
() -> new JSONArray(testCaseTwo, jsonParserConfiguration));
|
||||||
|
|
||||||
|
assertEquals("Expected a ',' or ']' at 10 [character 11 line 1]", jeOne.getMessage());
|
||||||
|
assertEquals("Unterminated string. Character with int code 0 is not allowed within a quoted string. at 15 [character 16 line 1]", jeTwo.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenUnbalancedQuotesObject_testStrictModeFalse_shouldThrowJsonException() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(false);
|
||||||
|
|
||||||
|
String testCaseOne = "{\"abc': \"test\"}";
|
||||||
|
String testCaseTwo = "{'abc\": \"test\"}";
|
||||||
|
|
||||||
|
JSONException jeOne = assertThrows(JSONException.class,
|
||||||
|
() -> new JSONObject(testCaseOne, jsonParserConfiguration));
|
||||||
|
JSONException jeTwo = assertThrows(JSONException.class,
|
||||||
|
() -> new JSONObject(testCaseTwo, jsonParserConfiguration));
|
||||||
|
|
||||||
|
assertEquals("Expected a ':' after a key at 10 [character 11 line 1]", jeOne.getMessage());
|
||||||
|
assertEquals("Unterminated string. Character with int code 0 is not allowed within a quoted string. at 15 [character 16 line 1]", jeTwo.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidInputArray_testStrictModeTrue_shouldThrowKeyNotSurroundedByQuotesErrorMessage() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
|
||||||
|
String testCase = "[{test: implied}]";
|
||||||
|
JSONException je = assertThrows("expected non-compliant array but got instead: " + testCase,
|
||||||
|
JSONException.class, () -> new JSONArray(testCase, jsonParserConfiguration));
|
||||||
|
|
||||||
|
assertEquals("Strict mode error: Value 'test' is not surrounded by quotes at 6 [character 7 line 1]",
|
||||||
|
je.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenInvalidInputObject_testStrictModeTrue_shouldThrowKeyNotSurroundedByQuotesErrorMessage() {
|
||||||
|
JSONParserConfiguration jsonParserConfiguration = new JSONParserConfiguration()
|
||||||
|
.withStrictMode(true);
|
||||||
|
|
||||||
|
String testCase = "{test: implied}";
|
||||||
|
JSONException je = assertThrows("expected non-compliant json but got instead: " + testCase,
|
||||||
|
JSONException.class, () -> new JSONObject(testCase, jsonParserConfiguration));
|
||||||
|
|
||||||
|
assertEquals("Strict mode error: Value 'test' is not surrounded by quotes at 5 [character 6 line 1]",
|
||||||
|
je.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method contains short but focused use-case samples and is exclusively used to test strictMode unit tests in
|
||||||
|
* this class.
|
||||||
|
*
|
||||||
|
* @return List with JSON strings.
|
||||||
|
*/
|
||||||
|
private List<String> getNonCompliantJSONArrayList() {
|
||||||
|
return Arrays.asList(
|
||||||
|
"[1],",
|
||||||
|
"[1,]",
|
||||||
|
"[,]",
|
||||||
|
"[,,]",
|
||||||
|
"[[1],\"sa\",[2]]a",
|
||||||
|
"[1],\"dsa\": \"test\"",
|
||||||
|
"[[a]]",
|
||||||
|
"[]asdf",
|
||||||
|
"[]]",
|
||||||
|
"[]}",
|
||||||
|
"[][",
|
||||||
|
"[]{",
|
||||||
|
"[],",
|
||||||
|
"[]:",
|
||||||
|
"[],[",
|
||||||
|
"[],{",
|
||||||
|
"[1,2];[3,4]",
|
||||||
|
"[test]",
|
||||||
|
"[{'testSingleQuote': 'testSingleQuote'}]",
|
||||||
|
"[1, 2,3]:[4,5]",
|
||||||
|
"[{test: implied}]",
|
||||||
|
"[{\"test\": implied}]",
|
||||||
|
"[{\"number\":\"7990154836330\",\"color\":'c'},{\"number\":8784148854580,\"color\":RosyBrown},{\"number\":\"5875770107113\",\"color\":\"DarkSeaGreen\"}]",
|
||||||
|
"[{test: \"implied\"}]");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method contains short but focused use-case samples and is exclusively used to test strictMode unit tests in
|
||||||
|
* this class.
|
||||||
|
*
|
||||||
|
* @return List with JSON strings.
|
||||||
|
*/
|
||||||
|
private List<String> getNonCompliantJSONObjectList() {
|
||||||
|
return Arrays.asList(
|
||||||
|
"{\"a\":1},",
|
||||||
|
"{\"a\":1,}",
|
||||||
|
"{\"a0\":[1],\"a1\":\"sa\",\"a2\":[2]}a",
|
||||||
|
"{\"a\":1},\"dsa\": \"test\"",
|
||||||
|
"{\"a\":[a]}",
|
||||||
|
"{}asdf",
|
||||||
|
"{}}",
|
||||||
|
"{}]",
|
||||||
|
"{}{",
|
||||||
|
"{}[",
|
||||||
|
"{},",
|
||||||
|
"{}:",
|
||||||
|
"{},{",
|
||||||
|
"{},[",
|
||||||
|
"{\"a0\":[1,2];\"a1\":[3,4]}",
|
||||||
|
"{\"a\":test}",
|
||||||
|
"{a:{'testSingleQuote': 'testSingleQuote'}}",
|
||||||
|
"{\"a0\":1, \"a1\":2,\"a2\":3}:{\"a3\":4,\"a4\":5}",
|
||||||
|
"{\"a\":{test: implied}}",
|
||||||
|
"{a:{\"test\": implied}}",
|
||||||
|
"{a:[{\"number\":\"7990154836330\",\"color\":'c'},{\"number\":8784148854580,\"color\":RosyBrown},{\"number\":\"5875770107113\",\"color\":\"DarkSeaGreen\"}]}",
|
||||||
|
"{a:{test: \"implied\"}}"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -384,8 +384,7 @@ public class JSONPointerTest {
|
|||||||
String str = "{"+
|
String str = "{"+
|
||||||
"\"string\\\\\\\\Key\":\"hello world!\","+
|
"\"string\\\\\\\\Key\":\"hello world!\","+
|
||||||
|
|
||||||
"\"\\\\\":\"slash test\"," +
|
"\"\\\\\":\"slash test\"" +
|
||||||
"}"+
|
|
||||||
"}";
|
"}";
|
||||||
JSONObject jsonObject = new JSONObject(str);
|
JSONObject jsonObject = new JSONObject(str);
|
||||||
//Summary of issue: When a KEY in the jsonObject is "\\\\" --> it's held
|
//Summary of issue: When a KEY in the jsonObject is "\\\\" --> it's held
|
||||||
|
|||||||
@@ -319,6 +319,22 @@ public class JSONStringTest {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testEnumJSONString() {
|
||||||
|
JSONObject jsonObject = new JSONObject();
|
||||||
|
jsonObject.put("key", MyEnum.MY_ENUM);
|
||||||
|
assertEquals("{\"key\":\"myJsonString\"}", jsonObject.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
private enum MyEnum implements JSONString {
|
||||||
|
MY_ENUM;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toJSONString() {
|
||||||
|
return "\"myJsonString\"";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A JSONString that returns a valid JSON string value.
|
* A JSONString that returns a valid JSON string value.
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -1,55 +0,0 @@
|
|||||||
package org.json.junit;
|
|
||||||
|
|
||||||
import org.json.JSONObject;
|
|
||||||
import org.junit.Test;
|
|
||||||
|
|
||||||
import java.math.BigDecimal;
|
|
||||||
import java.math.BigInteger;
|
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
|
||||||
|
|
||||||
public class JsonNumberZeroTest {
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseNegativeZeroValueWithMultipleZeroDigit(){
|
|
||||||
JSONObject jsonObject = new JSONObject("{value:-0000}");
|
|
||||||
assertEquals("Float not recognized", -0f, jsonObject.getFloat("value"), 0.0f);
|
|
||||||
assertEquals("Float not recognized", -0f, jsonObject.optFloat("value"), 0.0f);
|
|
||||||
assertEquals("Float not recognized", -0f, jsonObject.optFloatObject("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", -0d, jsonObject.optDouble("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", -0.0d, jsonObject.optDoubleObject("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", -0.0d, jsonObject.getDouble("value"), 0.0f);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.optLong("value"), 0);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.getLong("value"), 0);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.optLongObject("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.optInt("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.getInt("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.optIntegerObject("value"), 0);
|
|
||||||
assertEquals("Number not recognized", 0, jsonObject.getNumber("value").intValue(), 0);
|
|
||||||
assertEquals("Number not recognized", 0, jsonObject.getNumber("value").longValue(), 0);
|
|
||||||
assertEquals("BigDecimal not recognized", 0, BigDecimal.valueOf(-0).compareTo(jsonObject.getBigDecimal("value")));
|
|
||||||
assertEquals("BigInteger not recognized",0, BigInteger.valueOf(0).compareTo(jsonObject.getBigInteger("value")));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldParseZeroValueWithMultipleZeroDigit(){
|
|
||||||
JSONObject jsonObject = new JSONObject("{value:0000}");
|
|
||||||
assertEquals("Float not recognized", 0f, jsonObject.getFloat("value"), 0.0f);
|
|
||||||
assertEquals("Float not recognized", 0f, jsonObject.optFloat("value"), 0.0f);
|
|
||||||
assertEquals("Float not recognized", 0f, jsonObject.optFloatObject("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", 0d, jsonObject.optDouble("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", 0.0d, jsonObject.optDoubleObject("value"), 0.0f);
|
|
||||||
assertEquals("Double not recognized", 0.0d, jsonObject.getDouble("value"), 0.0f);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.optLong("value"), 0);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.getLong("value"), 0);
|
|
||||||
assertEquals("Long not recognized", 0, jsonObject.optLongObject("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.optInt("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.getInt("value"), 0);
|
|
||||||
assertEquals("Integer not recognized", 0, jsonObject.optIntegerObject("value"), 0);
|
|
||||||
assertEquals("Number not recognized", 0, jsonObject.getNumber("value").intValue(), 0);
|
|
||||||
assertEquals("Number not recognized", 0, jsonObject.getNumber("value").longValue(), 0);
|
|
||||||
assertEquals("BigDecimal not recognized", 0, BigDecimal.valueOf(-0).compareTo(jsonObject.getBigDecimal("value")));
|
|
||||||
assertEquals("BigInteger not recognized",0, BigInteger.valueOf(0).compareTo(jsonObject.getBigInteger("value")));
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
60
src/test/java/org/json/junit/StringBuilderWriterTest.java
Normal file
60
src/test/java/org/json/junit/StringBuilderWriterTest.java
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
|
||||||
|
import org.json.StringBuilderWriter;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
public class StringBuilderWriterTest {
|
||||||
|
private StringBuilderWriter writer;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() {
|
||||||
|
writer = new StringBuilderWriter();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testWriteChar() {
|
||||||
|
writer.write('a');
|
||||||
|
assertEquals("a", writer.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testWriteCharArray() {
|
||||||
|
char[] chars = {'a', 'b', 'c'};
|
||||||
|
writer.write(chars, 0, 3);
|
||||||
|
assertEquals("abc", writer.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testWriteString() {
|
||||||
|
writer.write("hello");
|
||||||
|
assertEquals("hello", writer.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testWriteStringWithOffsetAndLength() {
|
||||||
|
writer.write("hello world", 6, 5);
|
||||||
|
assertEquals("world", writer.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAppendCharSequence() {
|
||||||
|
writer.append("hello");
|
||||||
|
assertEquals("hello", writer.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAppendCharSequenceWithStartAndEnd() {
|
||||||
|
CharSequence csq = "hello world";
|
||||||
|
writer.append(csq, 6, 11);
|
||||||
|
assertEquals("world", writer.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAppendChar() {
|
||||||
|
writer.append('a');
|
||||||
|
assertEquals("a", writer.toString());
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -270,9 +270,9 @@ public class XMLConfigurationTest {
|
|||||||
|
|
||||||
String expectedStr =
|
String expectedStr =
|
||||||
"{\"addresses\":{\"address\":{\"street\":\"[CDATA[Baker street 5]\","+
|
"{\"addresses\":{\"address\":{\"street\":\"[CDATA[Baker street 5]\","+
|
||||||
"\"name\":\"Joe Tester\",\"NothingHere\":\"\",TrueValue:true,\n"+
|
"\"name\":\"Joe Tester\",\"NothingHere\":\"\",\"TrueValue\":true,\n"+
|
||||||
"\"FalseValue\":false,\"NullValue\":null,\"PositiveValue\":42,\n"+
|
"\"FalseValue\":false,\"NullValue\":null,\"PositiveValue\":42,\n"+
|
||||||
"\"NegativeValue\":-23,\"DoubleValue\":-23.45,\"Nan\":-23x.45,\n"+
|
"\"NegativeValue\":-23,\"DoubleValue\":-23.45,\"Nan\":\"-23x.45\",\n"+
|
||||||
"\"ArrayOfNum\":\"1, 2, 3, 4.1, 5.2\"\n"+
|
"\"ArrayOfNum\":\"1, 2, 3, 4.1, 5.2\"\n"+
|
||||||
"},\"xsi:noNamespaceSchemaLocation\":"+
|
"},\"xsi:noNamespaceSchemaLocation\":"+
|
||||||
"\"test.xsd\",\"xmlns:xsi\":\"http://www.w3.org/2001/"+
|
"\"test.xsd\",\"xmlns:xsi\":\"http://www.w3.org/2001/"+
|
||||||
@@ -761,7 +761,7 @@ public class XMLConfigurationTest {
|
|||||||
@Test
|
@Test
|
||||||
public void testToJSONArray_jsonOutput() {
|
public void testToJSONArray_jsonOutput() {
|
||||||
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
||||||
final JSONObject expected = new JSONObject("{\"root\":{\"item\":{\"id\":1},\"id\":[1,1,0,0],\"title\":true}}");
|
final JSONObject expected = new JSONObject("{\"root\":{\"item\":{\"id\":\"01\"},\"id\":[\"01\",1,\"00\",0],\"title\":true}}");
|
||||||
final JSONObject actualJsonOutput = XML.toJSONObject(originalXml,
|
final JSONObject actualJsonOutput = XML.toJSONObject(originalXml,
|
||||||
new XMLParserConfiguration().withKeepStrings(false));
|
new XMLParserConfiguration().withKeepStrings(false));
|
||||||
Util.compareActualVsExpectedJsonObjects(actualJsonOutput,expected);
|
Util.compareActualVsExpectedJsonObjects(actualJsonOutput,expected);
|
||||||
|
|||||||
@@ -267,9 +267,9 @@ public class XMLTest {
|
|||||||
|
|
||||||
String expectedStr =
|
String expectedStr =
|
||||||
"{\"addresses\":{\"address\":{\"street\":\"[CDATA[Baker street 5]\","+
|
"{\"addresses\":{\"address\":{\"street\":\"[CDATA[Baker street 5]\","+
|
||||||
"\"name\":\"Joe Tester\",\"NothingHere\":\"\",TrueValue:true,\n"+
|
"\"name\":\"Joe Tester\",\"NothingHere\":\"\",\"TrueValue\":true,\n"+
|
||||||
"\"FalseValue\":false,\"NullValue\":null,\"PositiveValue\":42,\n"+
|
"\"FalseValue\":false,\"NullValue\":null,\"PositiveValue\":42,\n"+
|
||||||
"\"NegativeValue\":-23,\"DoubleValue\":-23.45,\"Nan\":-23x.45,\n"+
|
"\"NegativeValue\":-23,\"DoubleValue\":-23.45,\"Nan\":\"-23x.45\",\n"+
|
||||||
"\"ArrayOfNum\":\"1, 2, 3, 4.1, 5.2\"\n"+
|
"\"ArrayOfNum\":\"1, 2, 3, 4.1, 5.2\"\n"+
|
||||||
"},\"xsi:noNamespaceSchemaLocation\":"+
|
"},\"xsi:noNamespaceSchemaLocation\":"+
|
||||||
"\"test.xsd\",\"xmlns:xsi\":\"http://www.w3.org/2001/"+
|
"\"test.xsd\",\"xmlns:xsi\":\"http://www.w3.org/2001/"+
|
||||||
@@ -791,7 +791,7 @@ public class XMLTest {
|
|||||||
@Test
|
@Test
|
||||||
public void testToJSONArray_jsonOutput() {
|
public void testToJSONArray_jsonOutput() {
|
||||||
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
||||||
final JSONObject expectedJson = new JSONObject("{\"root\":{\"item\":{\"id\":1},\"id\":[1,1,0,0],\"title\":true}}");
|
final JSONObject expectedJson = new JSONObject("{\"root\":{\"item\":{\"id\":\"01\"},\"id\":[\"01\",1,\"00\",0],\"title\":true}}");
|
||||||
final JSONObject actualJsonOutput = XML.toJSONObject(originalXml, false);
|
final JSONObject actualJsonOutput = XML.toJSONObject(originalXml, false);
|
||||||
|
|
||||||
Util.compareActualVsExpectedJsonObjects(actualJsonOutput,expectedJson);
|
Util.compareActualVsExpectedJsonObjects(actualJsonOutput,expectedJson);
|
||||||
@@ -1180,7 +1180,7 @@ public class XMLTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void shouldCreateExplicitEndTagWithEmptyValueWhenConfigured(){
|
public void shouldCreateExplicitEndTagWithEmptyValueWhenConfigured(){
|
||||||
String jsonString = "{outer:{innerOne:\"\", innerTwo:\"two\"}}";
|
String jsonString = "{\"outer\":{\"innerOne\":\"\", \"innerTwo\":\"two\"}}";
|
||||||
JSONObject jsonObject = new JSONObject(jsonString);
|
JSONObject jsonObject = new JSONObject(jsonString);
|
||||||
String expectedXmlString = "<encloser><outer><innerOne></innerOne><innerTwo>two</innerTwo></outer></encloser>";
|
String expectedXmlString = "<encloser><outer><innerOne></innerOne><innerTwo>two</innerTwo></outer></encloser>";
|
||||||
String xmlForm = XML.toString(jsonObject,"encloser", new XMLParserConfiguration().withCloseEmptyTag(true));
|
String xmlForm = XML.toString(jsonObject,"encloser", new XMLParserConfiguration().withCloseEmptyTag(true));
|
||||||
@@ -1191,7 +1191,7 @@ public class XMLTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void shouldNotCreateExplicitEndTagWithEmptyValueWhenNotConfigured(){
|
public void shouldNotCreateExplicitEndTagWithEmptyValueWhenNotConfigured(){
|
||||||
String jsonString = "{outer:{innerOne:\"\", innerTwo:\"two\"}}";
|
String jsonString = "{\"outer\":{\"innerOne\":\"\", \"innerTwo\":\"two\"}}";
|
||||||
JSONObject jsonObject = new JSONObject(jsonString);
|
JSONObject jsonObject = new JSONObject(jsonString);
|
||||||
String expectedXmlString = "<encloser><outer><innerOne/><innerTwo>two</innerTwo></outer></encloser>";
|
String expectedXmlString = "<encloser><outer><innerOne/><innerTwo>two</innerTwo></outer></encloser>";
|
||||||
String xmlForm = XML.toString(jsonObject,"encloser", new XMLParserConfiguration().withCloseEmptyTag(false));
|
String xmlForm = XML.toString(jsonObject,"encloser", new XMLParserConfiguration().withCloseEmptyTag(false));
|
||||||
@@ -1397,6 +1397,35 @@ public class XMLTest {
|
|||||||
Util.compareActualVsExpectedJsonObjects(actualJson,expectedJson);
|
Util.compareActualVsExpectedJsonObjects(actualJson,expectedJson);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void clarifyCurrentBehavior() {
|
||||||
|
|
||||||
|
// Behavior documented in #826
|
||||||
|
// After reverting the code, amount is stored as numeric, and phone is stored as string
|
||||||
|
String str1 =
|
||||||
|
" <datatypes>\n" +
|
||||||
|
" <telephone>0123456789</telephone>\n" +
|
||||||
|
" <amount>0.1230</amount>\n" +
|
||||||
|
" <boolean>true</boolean>\n" +
|
||||||
|
" </datatypes>";
|
||||||
|
JSONObject jsonObject1 = XML.toJSONObject(str1,
|
||||||
|
new XMLParserConfiguration().withKeepStrings(false));
|
||||||
|
assertEquals(jsonObject1.getJSONObject("datatypes").getFloat("amount"), 0.123, .1);
|
||||||
|
assertEquals(jsonObject1.getJSONObject("datatypes").getString("telephone"), "0123456789");
|
||||||
|
|
||||||
|
|
||||||
|
// Behavior documented in #852
|
||||||
|
// After reverting the code, value is still stored as a number. This is due to how XML.isDecimalNotation() works
|
||||||
|
// and is probably a bug. JSONObject has a similar problem.
|
||||||
|
String str2 = "<color> <color_type>primary</color_type> <value>008E97</value> </color>";
|
||||||
|
JSONObject jsonObject2 = XML.toJSONObject(str2);
|
||||||
|
assertEquals(jsonObject2.getJSONObject("color").getLong("value"), 0e897, .1);
|
||||||
|
|
||||||
|
// Workaround for now is to use keepStrings
|
||||||
|
JSONObject jsonObject3 = XML.toJSONObject(str2, new XMLParserConfiguration().withKeepStrings(true));
|
||||||
|
assertEquals(jsonObject3.getJSONObject("color").getString("value"), "008E97");
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
317
src/test/resources/compliantJsonArray.json
Normal file
317
src/test/resources/compliantJsonArray.json
Normal file
@@ -0,0 +1,317 @@
|
|||||||
|
[
|
||||||
|
{
|
||||||
|
"_id": "6606c27d2ab4a0102d49420a",
|
||||||
|
"index": 0,
|
||||||
|
"guid": "441331fb-84d1-4873-a649-3814621a0370",
|
||||||
|
"isActive": true,
|
||||||
|
"balance": "$2,691.63",
|
||||||
|
"picture": "http://example.abc/32x32",
|
||||||
|
"age": 26,
|
||||||
|
"eyeColor": "blue",
|
||||||
|
"name": "abc",
|
||||||
|
"gender": "female",
|
||||||
|
"company": "example",
|
||||||
|
"email": "abc@def.com",
|
||||||
|
"phone": "+1 (123) 456-7890",
|
||||||
|
"address": "123 Main St",
|
||||||
|
"about": "Laborum magna tempor officia irure cillum nulla incididunt Lorem dolor veniam elit cupidatat amet. Veniam veniam exercitation nulla consectetur officia esse ex sunt nulla nisi ea cillum nisi reprehenderit. Qui aliquip reprehenderit aliqua aliquip aliquip anim sit magna nostrud dolore veniam velit elit aliquip.\r\n",
|
||||||
|
"registered": "2016-07-22T03:18:11 -01:00",
|
||||||
|
"latitude": -21.544934,
|
||||||
|
"longitude": 72.765495,
|
||||||
|
"tags": [
|
||||||
|
"consectetur",
|
||||||
|
"minim",
|
||||||
|
"sunt",
|
||||||
|
"in",
|
||||||
|
"ut",
|
||||||
|
"velit",
|
||||||
|
"anim"
|
||||||
|
],
|
||||||
|
"friends": [
|
||||||
|
{
|
||||||
|
"id": 0,
|
||||||
|
"name": "abc def"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 1,
|
||||||
|
"name": "ghi jkl"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 2,
|
||||||
|
"name": "mno pqr"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"greeting": "Hello, abc! You have 10 unread messages.",
|
||||||
|
"favoriteFruit": "banana"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"_id": "6606c27d0a45df5121fb765f",
|
||||||
|
"index": 1,
|
||||||
|
"guid": "fd774715-de85-44b9-b498-c214d8f68d9f",
|
||||||
|
"isActive": true,
|
||||||
|
"balance": "$2,713.96",
|
||||||
|
"picture": "http://placehold.it/32x32",
|
||||||
|
"age": 27,
|
||||||
|
"eyeColor": "green",
|
||||||
|
"name": "def",
|
||||||
|
"gender": "female",
|
||||||
|
"company": "sample",
|
||||||
|
"email": "def@abc.com",
|
||||||
|
"phone": "+1 (123) 456-78910",
|
||||||
|
"address": "1234 Main St",
|
||||||
|
"about": "Ea id cupidatat eiusmod culpa. Nulla consequat esse elit enim et pariatur eiusmod ipsum. Consequat eu non reprehenderit in.\r\n",
|
||||||
|
"registered": "2015-04-06T07:54:22 -01:00",
|
||||||
|
"latitude": 83.512347,
|
||||||
|
"longitude": -9.368739,
|
||||||
|
"tags": [
|
||||||
|
"excepteur",
|
||||||
|
"non",
|
||||||
|
"nostrud",
|
||||||
|
"laboris",
|
||||||
|
"laboris",
|
||||||
|
"qui",
|
||||||
|
"aute"
|
||||||
|
],
|
||||||
|
"friends": [
|
||||||
|
{
|
||||||
|
"id": 0,
|
||||||
|
"name": "sample example"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 1,
|
||||||
|
"name": "test name"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 2,
|
||||||
|
"name": "aaa aaaa"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"greeting": "Hello, test! You have 7 unread messages.",
|
||||||
|
"favoriteFruit": "apple"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"_id": "6606c27dfb3a0e4e7e7183d3",
|
||||||
|
"index": 2,
|
||||||
|
"guid": "688b0c36-98e0-4ee7-86b8-863638d79b5f",
|
||||||
|
"isActive": false,
|
||||||
|
"balance": "$3,514.35",
|
||||||
|
"picture": "http://placehold.it/32x32",
|
||||||
|
"age": 32,
|
||||||
|
"eyeColor": "green",
|
||||||
|
"name": "test",
|
||||||
|
"gender": "female",
|
||||||
|
"company": "test",
|
||||||
|
"email": "test@test.com",
|
||||||
|
"phone": "+1 (123) 456-7890",
|
||||||
|
"address": "123 Main St",
|
||||||
|
"about": "Mollit officia adipisicing ex nisi non Lorem sunt quis est. Irure exercitation duis ipsum qui ullamco eu ea commodo occaecat minim proident. Incididunt nostrud ex cupidatat eiusmod mollit anim irure culpa. Labore voluptate voluptate labore nisi sit eu. Dolor sit proident velit dolor deserunt labore sit ipsum incididunt eiusmod reprehenderit voluptate. Duis anim velit officia laboris consequat officia dolor sint dolor nisi ex.\r\n",
|
||||||
|
"registered": "2021-11-02T12:50:05 -00:00",
|
||||||
|
"latitude": -82.969939,
|
||||||
|
"longitude": 86.415645,
|
||||||
|
"tags": [
|
||||||
|
"aliquip",
|
||||||
|
"et",
|
||||||
|
"est",
|
||||||
|
"nulla",
|
||||||
|
"nulla",
|
||||||
|
"tempor",
|
||||||
|
"adipisicing"
|
||||||
|
],
|
||||||
|
"friends": [
|
||||||
|
{
|
||||||
|
"id": 0,
|
||||||
|
"name": "test"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 1,
|
||||||
|
"name": "sample"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 2,
|
||||||
|
"name": "example"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"greeting": "Hello, test! You have 1 unread messages.",
|
||||||
|
"favoriteFruit": "strawberry"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"_id": "6606c27d204bc2327fc9ba23",
|
||||||
|
"index": 3,
|
||||||
|
"guid": "be970cba-306e-4cbd-be08-c265a43a61fa",
|
||||||
|
"isActive": true,
|
||||||
|
"balance": "$3,691.63",
|
||||||
|
"picture": "http://placehold.it/32x32",
|
||||||
|
"age": 35,
|
||||||
|
"eyeColor": "brown",
|
||||||
|
"name": "another test",
|
||||||
|
"gender": "male",
|
||||||
|
"company": "TEST",
|
||||||
|
"email": "anothertest@anothertest.com",
|
||||||
|
"phone": "+1 (321) 987-6543",
|
||||||
|
"address": "123 Example Main St",
|
||||||
|
"about": "Do proident consectetur minim quis. In adipisicing culpa Lorem fugiat cillum exercitation velit velit. Non voluptate laboris deserunt veniam et sint consectetur irure aliqua quis eiusmod consectetur elit id. Ex sint do anim Lorem excepteur eu nulla.\r\n",
|
||||||
|
"registered": "2020-06-25T04:55:25 -01:00",
|
||||||
|
"latitude": 63.614955,
|
||||||
|
"longitude": -109.299405,
|
||||||
|
"tags": [
|
||||||
|
"irure",
|
||||||
|
"esse",
|
||||||
|
"non",
|
||||||
|
"mollit",
|
||||||
|
"laborum",
|
||||||
|
"adipisicing",
|
||||||
|
"ad"
|
||||||
|
],
|
||||||
|
"friends": [
|
||||||
|
{
|
||||||
|
"id": 0,
|
||||||
|
"name": "test"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 1,
|
||||||
|
"name": "sample"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 2,
|
||||||
|
"name": "example"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"greeting": "Hello, another test! You have 5 unread messages.",
|
||||||
|
"favoriteFruit": "apple"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"_id": "6606c27df63eb5f390cb9989",
|
||||||
|
"index": 4,
|
||||||
|
"guid": "2c3e5115-758d-468e-99c5-c9afa26e1f9f",
|
||||||
|
"isActive": true,
|
||||||
|
"balance": "$1,047.20",
|
||||||
|
"picture": "http://test.it/32x32",
|
||||||
|
"age": 30,
|
||||||
|
"eyeColor": "green",
|
||||||
|
"name": "Test Name",
|
||||||
|
"gender": "female",
|
||||||
|
"company": "test",
|
||||||
|
"email": "testname@testname.com",
|
||||||
|
"phone": "+1 (999) 999-9999",
|
||||||
|
"address": "999 Test Main St",
|
||||||
|
"about": "Voluptate exercitation tempor consectetur velit magna ea occaecat cupidatat consectetur anim aute. Aliquip est aute ipsum laboris non irure qui consectetur tempor quis do ea Lorem. Cupidatat exercitation ad culpa aliqua amet commodo mollit reprehenderit exercitation adipisicing amet et laborum pariatur.\r\n",
|
||||||
|
"registered": "2023-01-19T02:43:18 -00:00",
|
||||||
|
"latitude": 14.15208,
|
||||||
|
"longitude": 170.411535,
|
||||||
|
"tags": [
|
||||||
|
"dolor",
|
||||||
|
"qui",
|
||||||
|
"cupidatat",
|
||||||
|
"aliqua",
|
||||||
|
"laboris",
|
||||||
|
"reprehenderit",
|
||||||
|
"sint"
|
||||||
|
],
|
||||||
|
"friends": [
|
||||||
|
{
|
||||||
|
"id": 0,
|
||||||
|
"name": "test"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 1,
|
||||||
|
"name": "sample"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 2,
|
||||||
|
"name": "example"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"greeting": "Hello, test! You have 6 unread messages.",
|
||||||
|
"favoriteFruit": "apple"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"_id": "6606c27d01d19fa29853d59c",
|
||||||
|
"index": 5,
|
||||||
|
"guid": "816cda74-5d4b-498f-9724-20f340d5f5bf",
|
||||||
|
"isActive": false,
|
||||||
|
"balance": "$2,628.74",
|
||||||
|
"picture": "http://testing.it/32x32",
|
||||||
|
"age": 28,
|
||||||
|
"eyeColor": "green",
|
||||||
|
"name": "Testing",
|
||||||
|
"gender": "female",
|
||||||
|
"company": "test",
|
||||||
|
"email": "testing@testing.com",
|
||||||
|
"phone": "+1 (888) 888-8888",
|
||||||
|
"address": "123 Main St",
|
||||||
|
"about": "Cupidatat non ut nulla qui excepteur in minim non et nulla fugiat. Dolor quis laborum occaecat veniam dolor ullamco deserunt amet veniam dolor quis proident tempor laboris. In cillum duis ut quis. Aliqua cupidatat magna proident velit tempor veniam et consequat laborum ex dolore qui. Incididunt deserunt magna minim Lorem consectetur.\r\n",
|
||||||
|
"registered": "2017-10-14T11:14:08 -01:00",
|
||||||
|
"latitude": -5.345728,
|
||||||
|
"longitude": -9.706491,
|
||||||
|
"tags": [
|
||||||
|
"officia",
|
||||||
|
"velit",
|
||||||
|
"laboris",
|
||||||
|
"qui",
|
||||||
|
"cupidatat",
|
||||||
|
"cupidatat",
|
||||||
|
"ad"
|
||||||
|
],
|
||||||
|
"friends": [
|
||||||
|
{
|
||||||
|
"id": 0,
|
||||||
|
"name": "test"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 1,
|
||||||
|
"name": "sample"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 2,
|
||||||
|
"name": "example"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"greeting": "Hello, testing! You have 2 unread messages.",
|
||||||
|
"favoriteFruit": "strawberry"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"_id": "6606c27d803003cede1d6deb",
|
||||||
|
"index": 6,
|
||||||
|
"guid": "4ee550bc-0920-4104-b3ce-ebf9db6a803f",
|
||||||
|
"isActive": true,
|
||||||
|
"balance": "$1,709.31",
|
||||||
|
"picture": "http://sample.it/32x32",
|
||||||
|
"age": 31,
|
||||||
|
"eyeColor": "blue",
|
||||||
|
"name": "Sample Name",
|
||||||
|
"gender": "female",
|
||||||
|
"company": "Sample",
|
||||||
|
"email": "sample@sample.com",
|
||||||
|
"phone": "+1 (777) 777-7777",
|
||||||
|
"address": "123 Main St",
|
||||||
|
"about": "Lorem ex proident ipsum ullamco velit sit nisi eiusmod cillum. Id tempor irure culpa nisi sit non qui veniam non ut. Aliquip reprehenderit excepteur mollit quis excepteur ex sit. Quis do eu veniam do ullamco occaecat eu cupidatat nisi laborum tempor minim fugiat pariatur. Ex in nulla ex velit.\r\n",
|
||||||
|
"registered": "2019-04-08T03:54:36 -01:00",
|
||||||
|
"latitude": -70.660321,
|
||||||
|
"longitude": 71.547525,
|
||||||
|
"tags": [
|
||||||
|
"consequat",
|
||||||
|
"veniam",
|
||||||
|
"pariatur",
|
||||||
|
"aliqua",
|
||||||
|
"cillum",
|
||||||
|
"eu",
|
||||||
|
"officia"
|
||||||
|
],
|
||||||
|
"friends": [
|
||||||
|
{
|
||||||
|
"id": 0,
|
||||||
|
"name": "Test"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 1,
|
||||||
|
"name": "Sample"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 2,
|
||||||
|
"name": "Example"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"greeting": "Hello, Sample! You have 6 unread messages.",
|
||||||
|
"favoriteFruit": "apple"
|
||||||
|
}
|
||||||
|
]
|
||||||
3703
src/test/resources/compliantJsonObject.json
Normal file
3703
src/test/resources/compliantJsonObject.json
Normal file
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user