mirror of
https://github.com/stleary/JSON-java.git
synced 2026-01-24 00:03:17 -05:00
Compare commits
167 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f9b5587c87 | ||
|
|
3d69990ab5 | ||
|
|
ba05e1a98c | ||
|
|
390d442054 | ||
|
|
3eb8a62af6 | ||
|
|
989cdb61bc | ||
|
|
8de0628bd1 | ||
|
|
569be9d19e | ||
|
|
3f97826462 | ||
|
|
d520210ea2 | ||
|
|
898288810f | ||
|
|
771c82c4eb | ||
|
|
d36066cf82 | ||
|
|
c1107fa987 | ||
|
|
cd631d970e | ||
|
|
f0289413d6 | ||
|
|
b4b39bb441 | ||
|
|
86253211c2 | ||
|
|
77c899d325 | ||
|
|
f164b8c597 | ||
|
|
6358b7f681 | ||
|
|
8550175556 | ||
|
|
72214f1b43 | ||
|
|
99c84fdf3a | ||
|
|
010e83b925 | ||
|
|
9865dbbebe | ||
|
|
4548696c8d | ||
|
|
f0308a3475 | ||
|
|
19dec1bb5f | ||
|
|
f2d20988de | ||
|
|
7915d8518f | ||
|
|
55b824d4c4 | ||
|
|
ac7806d060 | ||
|
|
86bb0a1a02 | ||
|
|
5ddb8c3d35 | ||
|
|
23ac2e7bca | ||
|
|
d7819a4fa2 | ||
|
|
7701f21839 | ||
|
|
ffd48afa42 | ||
|
|
abea194120 | ||
|
|
dcac3bc18e | ||
|
|
6d811607dd | ||
|
|
4d6de8c00a | ||
|
|
6dba7220e1 | ||
|
|
d4521696a9 | ||
|
|
e430db40aa | ||
|
|
7cbeb35498 | ||
|
|
aba82d9cc4 | ||
|
|
9ee10fdfc8 | ||
|
|
4a468d163a | ||
|
|
92991770ca | ||
|
|
09f35372d4 | ||
|
|
30f5b2de79 | ||
|
|
7f1cb8bf62 | ||
|
|
75419e3f25 | ||
|
|
097a401f3f | ||
|
|
5c4a7a1b1f | ||
|
|
b5f9febfe9 | ||
|
|
1a61af8255 | ||
|
|
11c29c366d | ||
|
|
a3742acf74 | ||
|
|
783d298f99 | ||
|
|
6c1bc0660a | ||
|
|
8ec822c575 | ||
|
|
1ceb70b525 | ||
|
|
c05d7058ff | ||
|
|
07a358449a | ||
|
|
7fe2fd95a5 | ||
|
|
b5b9f636ff | ||
|
|
c6ec2f0e4c | ||
|
|
1a2108efa2 | ||
|
|
caadcba30e | ||
|
|
ea842b437c | ||
|
|
a2a8240d0d | ||
|
|
1ab11d0802 | ||
|
|
5539722c69 | ||
|
|
98b79ae7bf | ||
|
|
04a4c5a3ec | ||
|
|
6007165c17 | ||
|
|
411f71137b | ||
|
|
e9117dbe5c | ||
|
|
7a85b514a9 | ||
|
|
82c8f486c5 | ||
|
|
006b29bda3 | ||
|
|
996d3a5fad | ||
|
|
de745e9c81 | ||
|
|
2374766018 | ||
|
|
1d0775cce7 | ||
|
|
d677a99f4e | ||
|
|
e6d37c469d | ||
|
|
f074bed732 | ||
|
|
e8f125fb6e | ||
|
|
ed183e6142 | ||
|
|
a86786a5f5 | ||
|
|
7c4f98c42c | ||
|
|
3894483560 | ||
|
|
be115059e9 | ||
|
|
2b41cf44b5 | ||
|
|
9a9efac2af | ||
|
|
134074aeaa | ||
|
|
8540bb80c0 | ||
|
|
4dfd779b1c | ||
|
|
7b2677ac5a | ||
|
|
29a7f4622d | ||
|
|
f346203cd6 | ||
|
|
b180dbedbc | ||
|
|
cca6d1020f | ||
|
|
af5f780d5b | ||
|
|
495cec9037 | ||
|
|
e4aa7f1308 | ||
|
|
56cb5f84c4 | ||
|
|
0cdc38ac24 | ||
|
|
d5277b126b | ||
|
|
228598ca84 | ||
|
|
0a6fb1d578 | ||
|
|
c4cd526c53 | ||
|
|
776b5ccb85 | ||
|
|
fb99c06bad | ||
|
|
bc09f90e90 | ||
|
|
c93014cb53 | ||
|
|
0e4a94d91d | ||
|
|
1a38879c90 | ||
|
|
4c8cac22a8 | ||
|
|
fe45fa9cfb | ||
|
|
79af389f7a | ||
|
|
1726b6cf55 | ||
|
|
beb2fb5706 | ||
|
|
ff921db783 | ||
|
|
61bb60e752 | ||
|
|
ef68cdf810 | ||
|
|
eaa5611ba3 | ||
|
|
dbb113176b | ||
|
|
16967f322e | ||
|
|
284a316838 | ||
|
|
4e8231c512 | ||
|
|
db0fde2a56 | ||
|
|
661114c50d | ||
|
|
ca88454f1c | ||
|
|
3e688afc66 | ||
|
|
becc1631e6 | ||
|
|
01727fd0ed | ||
|
|
74cd73f97c | ||
|
|
c29d4881e0 | ||
|
|
7c1b6531e7 | ||
|
|
db122e5d3a | ||
|
|
a309931d20 | ||
|
|
e27da22e05 | ||
|
|
af6d07cecb | ||
|
|
64093366b3 | ||
|
|
9b69ec49ad | ||
|
|
2c674be1b6 | ||
|
|
be33deb7d5 | ||
|
|
48089a4da7 | ||
|
|
a4e152f4f0 | ||
|
|
3dd8f2ecd5 | ||
|
|
bae0b0dac9 | ||
|
|
e563dbcaaa | ||
|
|
50dfcc59b3 | ||
|
|
b2943eb395 | ||
|
|
60662e2f83 | ||
|
|
2a4bc3420a | ||
|
|
b6ff0db984 | ||
|
|
c8a9e15a57 | ||
|
|
402db6ad84 | ||
|
|
4951ec48c8 | ||
|
|
8ce0019a5d | ||
|
|
3d524349a1 |
6
.github/workflows/codeql-analysis.yml
vendored
6
.github/workflows/codeql-analysis.yml
vendored
@@ -25,11 +25,11 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v2
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
# Initializes the CodeQL tools for scanning.
|
# Initializes the CodeQL tools for scanning.
|
||||||
- name: Initialize CodeQL
|
- name: Initialize CodeQL
|
||||||
uses: github/codeql-action/init@v1
|
uses: github/codeql-action/init@v2
|
||||||
with:
|
with:
|
||||||
languages: ${{ matrix.language }}
|
languages: ${{ matrix.language }}
|
||||||
# If you wish to specify custom queries, you can do so here or in a config file.
|
# If you wish to specify custom queries, you can do so here or in a config file.
|
||||||
@@ -40,4 +40,4 @@ jobs:
|
|||||||
- run: "mvn clean compile -Dmaven.test.skip=true -Dmaven.site.skip=true -Dmaven.javadoc.skip=true"
|
- run: "mvn clean compile -Dmaven.test.skip=true -Dmaven.site.skip=true -Dmaven.javadoc.skip=true"
|
||||||
|
|
||||||
- name: Perform CodeQL Analysis
|
- name: Perform CodeQL Analysis
|
||||||
uses: github/codeql-action/analyze@v1
|
uses: github/codeql-action/analyze@v2
|
||||||
|
|||||||
206
.github/workflows/pipeline.yml
vendored
206
.github/workflows/pipeline.yml
vendored
@@ -1,5 +1,5 @@
|
|||||||
# This workflow will build a Java project with Maven
|
# This workflow will build a Java project with Maven
|
||||||
# For more information see: https://help.github.com/actions/language-and-framework-guides/building-and-testing-java-with-maven
|
# For more information see: https://docs.github.com/en/actions/learn-github-actions or https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions
|
||||||
|
|
||||||
name: Java CI with Maven
|
name: Java CI with Maven
|
||||||
|
|
||||||
@@ -12,63 +12,217 @@ on:
|
|||||||
jobs:
|
jobs:
|
||||||
# old-school build and jar method. No tests run or compiled.
|
# old-school build and jar method. No tests run or compiled.
|
||||||
build-1_6:
|
build-1_6:
|
||||||
|
name: Java 1.6
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
# build for java 1.6, however don't run any tests
|
|
||||||
java: [ 1.6 ]
|
|
||||||
name: Java ${{ matrix.java }}
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v4
|
||||||
- name: Setup java
|
- name: Setup java
|
||||||
uses: actions/setup-java@v1
|
uses: actions/setup-java@v1
|
||||||
with:
|
with:
|
||||||
java-version: ${{ matrix.java }}
|
java-version: 1.6
|
||||||
- name: Compile Java ${{ matrix.java }}
|
- name: Compile Java 1.6
|
||||||
run: |
|
run: |
|
||||||
mkdir -p target/classes
|
mkdir -p target/classes
|
||||||
javac -d target/classes/ src/main/java/org/json/*.java
|
javac -version
|
||||||
- name: Create java ${{ matrix.java }} JAR
|
javac -source 1.6 -target 1.6 -d target/classes/ src/main/java/org/json/*.java
|
||||||
|
- name: Create java 1.6 JAR
|
||||||
run: |
|
run: |
|
||||||
jar cvf target/org.json.jar -C target/classes .
|
jar cvf target/org.json.jar -C target/classes .
|
||||||
- name: Upload Java ${{ matrix.java }} JAR
|
- name: Upload JAR 1.6
|
||||||
uses: actions/upload-artifact@v1
|
if: ${{ always() }}
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: Java ${{ matrix.java }} JAR
|
name: Create java 1.6 JAR
|
||||||
path: target/org.json.jar
|
path: target/*.jar
|
||||||
|
|
||||||
build:
|
build-8:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
strategy:
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
max-parallel: 1
|
||||||
matrix:
|
matrix:
|
||||||
# build against supported Java LTS versions:
|
# build against supported Java LTS versions:
|
||||||
java: [ 8, 11 ]
|
java: [ 8 ]
|
||||||
name: Java ${{ matrix.java }}
|
name: Java ${{ matrix.java }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v3
|
||||||
- name: Setup java
|
- name: Set up JDK ${{ matrix.java }}
|
||||||
uses: actions/setup-java@v1
|
uses: actions/setup-java@v3
|
||||||
with:
|
with:
|
||||||
|
distribution: 'temurin'
|
||||||
java-version: ${{ matrix.java }}
|
java-version: ${{ matrix.java }}
|
||||||
|
cache: 'maven'
|
||||||
- name: Compile Java ${{ matrix.java }}
|
- name: Compile Java ${{ matrix.java }}
|
||||||
run: mvn clean compile -Dmaven.compiler.source=${{ matrix.java }} -Dmaven.compiler.target=${{ matrix.java }} -Dmaven.test.skip=true -Dmaven.site.skip=true -Dmaven.javadoc.skip=true
|
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 }}
|
- name: Run Tests ${{ matrix.java }}
|
||||||
run: |
|
run: |
|
||||||
mvn test -Dmaven.compiler.source=${{ matrix.java }} -Dmaven.compiler.target=${{ matrix.java }}
|
mvn test -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }}
|
||||||
- name: Build Test Report ${{ matrix.java }}
|
- name: Build Test Report ${{ matrix.java }}
|
||||||
if: ${{ always() }}
|
if: ${{ always() }}
|
||||||
run: |
|
run: |
|
||||||
mvn surefire-report:report-only -Dmaven.compiler.source=${{ matrix.java }} -Dmaven.compiler.target=${{ matrix.java }}
|
mvn surefire-report:report-only -D maven.compiler.source=${{ matrix.java }} -D maven.compiler.target=${{ matrix.java }}
|
||||||
mvn site -DgenerateReports=false -Dmaven.compiler.source=${{ matrix.java }} -Dmaven.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 }}
|
- name: Upload Test Results ${{ matrix.java }}
|
||||||
if: ${{ always() }}
|
if: ${{ always() }}
|
||||||
uses: actions/upload-artifact@v1
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: Test Results ${{ matrix.java }}
|
name: Test Results ${{ matrix.java }}
|
||||||
path: target/surefire-reports/
|
path: target/surefire-reports/
|
||||||
- name: Upload Test Report ${{ matrix.java }}
|
- name: Upload Test Report ${{ matrix.java }}
|
||||||
if: ${{ always() }}
|
if: ${{ always() }}
|
||||||
uses: actions/upload-artifact@v1
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: Test Report ${{ matrix.java }}
|
name: Test Report ${{ matrix.java }}
|
||||||
path: target/site/
|
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 }}
|
||||||
|
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
|
||||||
|
|||||||
2
.gitignore
vendored
2
.gitignore
vendored
@@ -1,6 +1,8 @@
|
|||||||
# ignore eclipse project files
|
# ignore eclipse project files
|
||||||
.project
|
.project
|
||||||
.classpath
|
.classpath
|
||||||
|
# ignore vscode files
|
||||||
|
.vscode
|
||||||
# ignore Intellij Idea project files
|
# ignore Intellij Idea project files
|
||||||
.idea
|
.idea
|
||||||
*.iml
|
*.iml
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
# Contribution Guidelines
|
# Contribution Guidelines
|
||||||
|
|
||||||
Feel free to work on any issue with a #hacktoberfest label.
|
Feel free to work on any open issue, you don't need to ask permission first. This year, the hacktoberfest label will be added to any PR and associated issue during the month of October.
|
||||||
|
|
||||||
If you discover an issue you would like to work on, you can add a new issue to the list. If it meets our criteria, a hacktoberfest label will be added.
|
If you discover an issue you would like to work on, you can add a new issue to the list. If it meets our criteria, it will be available to work on (if not, it will be closed after review).
|
||||||
|
|
||||||
# Who is allowed to submit pull requests for this project?
|
# Who is allowed to submit pull requests for this project?
|
||||||
|
|
||||||
|
|||||||
37
README.md
37
README.md
@@ -7,8 +7,10 @@ JSON in Java [package org.json]
|
|||||||
===============================
|
===============================
|
||||||
|
|
||||||
[](https://mvnrepository.com/artifact/org.json/json)
|
[](https://mvnrepository.com/artifact/org.json/json)
|
||||||
|
[](https://github.com/stleary/JSON-java/actions/workflows/pipeline.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/20230618/json-20230618.jar)**
|
**[Click here if you just want the latest release jar file.](https://search.maven.org/remotecontent?filepath=org/json/json/20240205/json-20240303.jar)**
|
||||||
|
|
||||||
|
|
||||||
# Overview
|
# Overview
|
||||||
@@ -24,7 +26,8 @@ Project goals include:
|
|||||||
* No external dependencies
|
* No external dependencies
|
||||||
* Fast execution and low memory footprint
|
* Fast execution and low memory footprint
|
||||||
* Maintain backward compatibility
|
* Maintain backward compatibility
|
||||||
* Designed and tested to use on Java versions 1.6 - 1.11
|
* Designed and tested to use on Java versions 1.6 - 21
|
||||||
|
|
||||||
|
|
||||||
The files in this package implement JSON encoders and decoders. The package can also convert between JSON and XML, HTTP headers, Cookies, and CDL.
|
The files in this package implement JSON encoders and decoders. The package can also convert between JSON and XML, HTTP headers, Cookies, and CDL.
|
||||||
|
|
||||||
@@ -41,56 +44,56 @@ The org.json package can be built from the command line, Maven, and Gradle. The
|
|||||||
**Building from the command line**
|
**Building from the command line**
|
||||||
|
|
||||||
*Build the class files from the package root directory src/main/java*
|
*Build the class files from the package root directory src/main/java*
|
||||||
````
|
```shell
|
||||||
javac org/json/*.java
|
javac org/json/*.java
|
||||||
````
|
```
|
||||||
|
|
||||||
*Create the jar file in the current directory*
|
*Create the jar file in the current directory*
|
||||||
````
|
```shell
|
||||||
jar cf json-java.jar org/json/*.class
|
jar cf json-java.jar org/json/*.class
|
||||||
````
|
```
|
||||||
|
|
||||||
*Compile a program that uses the jar (see example code below)*
|
*Compile a program that uses the jar (see example code below)*
|
||||||
````
|
```shell
|
||||||
javac -cp .;json-java.jar Test.java (Windows)
|
javac -cp .;json-java.jar Test.java (Windows)
|
||||||
javac -cp .:json-java.jar Test.java (Unix Systems)
|
javac -cp .:json-java.jar Test.java (Unix Systems)
|
||||||
````
|
```
|
||||||
|
|
||||||
*Test file contents*
|
*Test file contents*
|
||||||
|
|
||||||
````
|
```java
|
||||||
import org.json.JSONObject;
|
import org.json.JSONObject;
|
||||||
public class Test {
|
public class Test {
|
||||||
public static void main(String args[]){
|
public static void main(String args[]){
|
||||||
JSONObject jo = new JSONObject("{ \"abc\" : \"def\" }");
|
JSONObject jo = new JSONObject("{ \"abc\" : \"def\" }");
|
||||||
System.out.println(jo.toString());
|
System.out.println(jo);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
````
|
```
|
||||||
|
|
||||||
*Execute the Test file*
|
*Execute the Test file*
|
||||||
````
|
```shell
|
||||||
java -cp .;json-java.jar Test (Windows)
|
java -cp .;json-java.jar Test (Windows)
|
||||||
java -cp .:json-java.jar Test (Unix Systems)
|
java -cp .:json-java.jar Test (Unix Systems)
|
||||||
````
|
```
|
||||||
|
|
||||||
*Expected output*
|
*Expected output*
|
||||||
|
|
||||||
````
|
```json
|
||||||
{"abc":"def"}
|
{"abc":"def"}
|
||||||
````
|
```
|
||||||
|
|
||||||
|
|
||||||
**Tools to build the package and execute the unit tests**
|
**Tools to build the package and execute the unit tests**
|
||||||
|
|
||||||
Execute the test suite with Maven:
|
Execute the test suite with Maven:
|
||||||
```
|
```shell
|
||||||
mvn clean test
|
mvn clean test
|
||||||
```
|
```
|
||||||
|
|
||||||
Execute the test suite with Gradlew:
|
Execute the test suite with Gradlew:
|
||||||
|
|
||||||
```
|
```shell
|
||||||
gradlew clean build test
|
gradlew clean build test
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
10
build.gradle
10
build.gradle
@@ -20,9 +20,9 @@ repositories {
|
|||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
testImplementation 'junit:junit:4.13.1'
|
testImplementation 'junit:junit:4.13.2'
|
||||||
testImplementation 'com.jayway.jsonpath:json-path:2.1.0'
|
testImplementation 'com.jayway.jsonpath:json-path:2.4.0'
|
||||||
testImplementation 'org.mockito:mockito-core:1.9.5'
|
testImplementation 'org.mockito:mockito-core:4.2.0'
|
||||||
}
|
}
|
||||||
|
|
||||||
subprojects {
|
subprojects {
|
||||||
@@ -30,9 +30,9 @@ subprojects {
|
|||||||
}
|
}
|
||||||
|
|
||||||
group = 'org.json'
|
group = 'org.json'
|
||||||
version = 'v20211205-SNAPSHOT'
|
version = 'v20230618-SNAPSHOT'
|
||||||
description = 'JSON in Java'
|
description = 'JSON in Java'
|
||||||
sourceCompatibility = '1.7'
|
sourceCompatibility = '1.8'
|
||||||
|
|
||||||
configurations.all {
|
configurations.all {
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,6 +5,12 @@ and artifactId "json". For example:
|
|||||||
[https://search.maven.org/search?q=g:org.json%20AND%20a:json&core=gav](https://search.maven.org/search?q=g:org.json%20AND%20a:json&core=gav)
|
[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)
|
||||||
|
|
||||||
~~~
|
~~~
|
||||||
|
20240303 Revert optLong/getLong changes, and recent commits.
|
||||||
|
|
||||||
|
20240205 Recent commits.
|
||||||
|
|
||||||
|
20231013 First release with minimum Java version 1.8. Recent commits, including fixes for CVE-2023-5072.
|
||||||
|
|
||||||
20230618 Final release with Java 1.6 compatibility. Future releases will require Java 1.8 or greater.
|
20230618 Final release with Java 1.6 compatibility. Future releases will require Java 1.8 or greater.
|
||||||
|
|
||||||
20230227 Fix for CVE-2022-45688 and recent commits
|
20230227 Fix for CVE-2022-45688 and recent commits
|
||||||
|
|||||||
79
pom.xml
79
pom.xml
@@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
<groupId>org.json</groupId>
|
<groupId>org.json</groupId>
|
||||||
<artifactId>json</artifactId>
|
<artifactId>json</artifactId>
|
||||||
<version>20230618</version>
|
<version>20240303</version>
|
||||||
<packaging>bundle</packaging>
|
<packaging>bundle</packaging>
|
||||||
|
|
||||||
<name>JSON in Java</name>
|
<name>JSON in Java</name>
|
||||||
@@ -15,18 +15,12 @@
|
|||||||
It also includes the capability to convert between JSON and XML, HTTP
|
It also includes the capability to convert between JSON and XML, HTTP
|
||||||
headers, Cookies, and CDL.
|
headers, Cookies, and CDL.
|
||||||
|
|
||||||
This is a reference implementation. There is a large number of JSON packages
|
This is a reference implementation. There are a large number of JSON packages
|
||||||
in Java. Perhaps someday the Java community will standardize on one. Until
|
in Java. Perhaps someday the Java community will standardize on one. Until
|
||||||
then, choose carefully.
|
then, choose carefully.
|
||||||
</description>
|
</description>
|
||||||
<url>https://github.com/douglascrockford/JSON-java</url>
|
<url>https://github.com/douglascrockford/JSON-java</url>
|
||||||
|
|
||||||
<parent>
|
|
||||||
<groupId>org.sonatype.oss</groupId>
|
|
||||||
<artifactId>oss-parent</artifactId>
|
|
||||||
<version>9</version>
|
|
||||||
</parent>
|
|
||||||
|
|
||||||
<scm>
|
<scm>
|
||||||
<url>https://github.com/douglascrockford/JSON-java.git</url>
|
<url>https://github.com/douglascrockford/JSON-java.git</url>
|
||||||
<connection>scm:git:git://github.com/douglascrockford/JSON-java.git</connection>
|
<connection>scm:git:git://github.com/douglascrockford/JSON-java.git</connection>
|
||||||
@@ -53,23 +47,36 @@
|
|||||||
</properties>
|
</properties>
|
||||||
|
|
||||||
|
|
||||||
|
<distributionManagement>
|
||||||
|
<repository>
|
||||||
|
<id>ossrh</id>
|
||||||
|
<name>Central Repository OSSRH</name>
|
||||||
|
<url>https://oss.sonatype.org/service/local/staging/deploy/maven2/</url>
|
||||||
|
</repository>
|
||||||
|
<snapshotRepository>
|
||||||
|
<id>ossrh</id>
|
||||||
|
<url>https://oss.sonatype.org/content/repositories/snapshots</url>
|
||||||
|
</snapshotRepository>
|
||||||
|
</distributionManagement>
|
||||||
|
|
||||||
|
|
||||||
<dependencies>
|
<dependencies>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>junit</groupId>
|
<groupId>junit</groupId>
|
||||||
<artifactId>junit</artifactId>
|
<artifactId>junit</artifactId>
|
||||||
<version>4.13.1</version>
|
<version>4.13.2</version>
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>com.jayway.jsonpath</groupId>
|
<groupId>com.jayway.jsonpath</groupId>
|
||||||
<artifactId>json-path</artifactId>
|
<artifactId>json-path</artifactId>
|
||||||
<version>2.1.0</version>
|
<version>2.4.0</version>
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.mockito</groupId>
|
<groupId>org.mockito</groupId>
|
||||||
<artifactId>mockito-core</artifactId>
|
<artifactId>mockito-core</artifactId>
|
||||||
<version>1.9.5</version>
|
<version>4.2.0</version>
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
@@ -79,7 +86,7 @@
|
|||||||
<plugin>
|
<plugin>
|
||||||
<groupId>org.apache.felix</groupId>
|
<groupId>org.apache.felix</groupId>
|
||||||
<artifactId>maven-bundle-plugin</artifactId>
|
<artifactId>maven-bundle-plugin</artifactId>
|
||||||
<version>3.0.1</version>
|
<version>5.1.9</version>
|
||||||
<extensions>true</extensions>
|
<extensions>true</extensions>
|
||||||
<configuration>
|
<configuration>
|
||||||
<instructions>
|
<instructions>
|
||||||
@@ -93,16 +100,19 @@
|
|||||||
<plugin>
|
<plugin>
|
||||||
<groupId>org.apache.maven.plugins</groupId>
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
<artifactId>maven-compiler-plugin</artifactId>
|
<artifactId>maven-compiler-plugin</artifactId>
|
||||||
<version>2.3.2</version>
|
<version>3.11.0</version>
|
||||||
<configuration>
|
<configuration>
|
||||||
<source>1.6</source>
|
<source>1.8</source>
|
||||||
<target>1.6</target>
|
<target>1.8</target>
|
||||||
|
<compilerArgs>
|
||||||
|
<arg>-Xlint:unchecked</arg>
|
||||||
|
</compilerArgs>
|
||||||
</configuration>
|
</configuration>
|
||||||
</plugin>
|
</plugin>
|
||||||
<plugin>
|
<plugin>
|
||||||
<groupId>org.apache.maven.plugins</groupId>
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
<artifactId>maven-source-plugin</artifactId>
|
<artifactId>maven-source-plugin</artifactId>
|
||||||
<version>2.1.2</version>
|
<version>3.3.0</version>
|
||||||
<executions>
|
<executions>
|
||||||
<execution>
|
<execution>
|
||||||
<id>attach-sources</id>
|
<id>attach-sources</id>
|
||||||
@@ -115,7 +125,7 @@
|
|||||||
<plugin>
|
<plugin>
|
||||||
<groupId>org.apache.maven.plugins</groupId>
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
<artifactId>maven-javadoc-plugin</artifactId>
|
<artifactId>maven-javadoc-plugin</artifactId>
|
||||||
<version>2.7</version>
|
<version>3.5.0</version>
|
||||||
<executions>
|
<executions>
|
||||||
<execution>
|
<execution>
|
||||||
<id>attach-javadocs</id>
|
<id>attach-javadocs</id>
|
||||||
@@ -131,7 +141,7 @@
|
|||||||
<plugin>
|
<plugin>
|
||||||
<groupId>org.apache.maven.plugins</groupId>
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
<artifactId>maven-gpg-plugin</artifactId>
|
<artifactId>maven-gpg-plugin</artifactId>
|
||||||
<version>1.5</version>
|
<version>1.6</version>
|
||||||
<executions>
|
<executions>
|
||||||
<execution>
|
<execution>
|
||||||
<id>sign-artifacts</id>
|
<id>sign-artifacts</id>
|
||||||
@@ -159,17 +169,34 @@
|
|||||||
<autoReleaseAfterClose>false</autoReleaseAfterClose>
|
<autoReleaseAfterClose>false</autoReleaseAfterClose>
|
||||||
</configuration>
|
</configuration>
|
||||||
</plugin>
|
</plugin>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.moditect</groupId>
|
||||||
|
<artifactId>moditect-maven-plugin</artifactId>
|
||||||
|
<version>1.0.0.Final</version>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<id>add-module-infos</id>
|
||||||
|
<phase>package</phase>
|
||||||
|
<goals>
|
||||||
|
<goal>add-module-info</goal>
|
||||||
|
</goals>
|
||||||
|
<configuration>
|
||||||
|
<jvmVersion>9</jvmVersion>
|
||||||
|
<module>
|
||||||
|
<moduleInfoSource>
|
||||||
|
module org.json {
|
||||||
|
exports org.json;
|
||||||
|
}
|
||||||
|
</moduleInfoSource>
|
||||||
|
</module>
|
||||||
|
</configuration>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
<plugin>
|
<plugin>
|
||||||
<groupId>org.apache.maven.plugins</groupId>
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
<artifactId>maven-jar-plugin</artifactId>
|
<artifactId>maven-jar-plugin</artifactId>
|
||||||
<version>3.2.0</version>
|
<version>3.3.0</version>
|
||||||
<configuration>
|
|
||||||
<archive>
|
|
||||||
<manifestEntries>
|
|
||||||
<Automatic-Module-Name>org.json</Automatic-Module-Name>
|
|
||||||
</manifestEntries>
|
|
||||||
</archive>
|
|
||||||
</configuration>
|
|
||||||
</plugin>
|
</plugin>
|
||||||
</plugins>
|
</plugins>
|
||||||
</build>
|
</build>
|
||||||
|
|||||||
@@ -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 '='.
|
||||||
@@ -46,19 +52,19 @@ public class CookieList {
|
|||||||
* @throws JSONException if a called function fails
|
* @throws JSONException if a called function fails
|
||||||
*/
|
*/
|
||||||
public static String toString(JSONObject jo) throws JSONException {
|
public static String toString(JSONObject jo) throws JSONException {
|
||||||
boolean b = false;
|
boolean isEndOfPair = false;
|
||||||
final StringBuilder sb = new StringBuilder();
|
final StringBuilder sb = new StringBuilder();
|
||||||
// Don't use the new entrySet API to maintain Android support
|
// Don't use the new entrySet API to maintain Android support
|
||||||
for (final String key : jo.keySet()) {
|
for (final String key : jo.keySet()) {
|
||||||
final Object value = jo.opt(key);
|
final Object value = jo.opt(key);
|
||||||
if (!JSONObject.NULL.equals(value)) {
|
if (!JSONObject.NULL.equals(value)) {
|
||||||
if (b) {
|
if (isEndOfPair) {
|
||||||
sb.append(';');
|
sb.append(';');
|
||||||
}
|
}
|
||||||
sb.append(Cookie.escape(key));
|
sb.append(Cookie.escape(key));
|
||||||
sb.append("=");
|
sb.append("=");
|
||||||
sb.append(Cookie.escape(value.toString()));
|
sb.append(Cookie.escape(value.toString()));
|
||||||
b = true;
|
isEndOfPair = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return sb.toString();
|
return sb.toString();
|
||||||
|
|||||||
@@ -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";
|
||||||
|
|
||||||
|
|||||||
@@ -149,11 +149,40 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
* A Collection.
|
* A Collection.
|
||||||
*/
|
*/
|
||||||
public JSONArray(Collection<?> collection) {
|
public JSONArray(Collection<?> collection) {
|
||||||
|
this(collection, 0, new JSONParserConfiguration());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a JSONArray from a Collection.
|
||||||
|
*
|
||||||
|
* @param collection
|
||||||
|
* A Collection.
|
||||||
|
* @param jsonParserConfiguration
|
||||||
|
* Configuration object for the JSON parser
|
||||||
|
*/
|
||||||
|
public JSONArray(Collection<?> collection, JSONParserConfiguration jsonParserConfiguration) {
|
||||||
|
this(collection, 0, jsonParserConfiguration);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a JSONArray from a collection with recursion depth.
|
||||||
|
*
|
||||||
|
* @param collection
|
||||||
|
* A Collection.
|
||||||
|
* @param recursionDepth
|
||||||
|
* Variable for tracking the count of nested object creations.
|
||||||
|
* @param jsonParserConfiguration
|
||||||
|
* Configuration object for the JSON parser
|
||||||
|
*/
|
||||||
|
JSONArray(Collection<?> collection, int recursionDepth, JSONParserConfiguration jsonParserConfiguration) {
|
||||||
|
if (recursionDepth > jsonParserConfiguration.getMaxNestingDepth()) {
|
||||||
|
throw new JSONException("JSONArray has reached recursion depth limit of " + jsonParserConfiguration.getMaxNestingDepth());
|
||||||
|
}
|
||||||
if (collection == null) {
|
if (collection == null) {
|
||||||
this.myArrayList = new ArrayList<Object>();
|
this.myArrayList = new ArrayList<Object>();
|
||||||
} else {
|
} else {
|
||||||
this.myArrayList = new ArrayList<Object>(collection.size());
|
this.myArrayList = new ArrayList<Object>(collection.size());
|
||||||
this.addAll(collection, true);
|
this.addAll(collection, true, recursionDepth, jsonParserConfiguration);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -205,7 +234,7 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
throw new JSONException(
|
throw new JSONException(
|
||||||
"JSONArray initial value should be a string or collection or array.");
|
"JSONArray initial value should be a string or collection or array.");
|
||||||
}
|
}
|
||||||
this.addAll(array, true);
|
this.addAll(array, true, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -599,6 +628,38 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional Boolean object associated with an index. It returns false
|
||||||
|
* if there is no value at that index, or if the value is not Boolean.TRUE
|
||||||
|
* or the String "true".
|
||||||
|
*
|
||||||
|
* @param index
|
||||||
|
* The index must be between 0 and length() - 1.
|
||||||
|
* @return The truth.
|
||||||
|
*/
|
||||||
|
public Boolean optBooleanObject(int index) {
|
||||||
|
return this.optBooleanObject(index, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional Boolean object associated with an index. It returns the
|
||||||
|
* defaultValue if there is no value at that index or if it is not a Boolean
|
||||||
|
* or the String "true" or "false" (case insensitive).
|
||||||
|
*
|
||||||
|
* @param index
|
||||||
|
* The index must be between 0 and length() - 1.
|
||||||
|
* @param defaultValue
|
||||||
|
* A boolean default.
|
||||||
|
* @return The truth.
|
||||||
|
*/
|
||||||
|
public Boolean optBooleanObject(int index, Boolean defaultValue) {
|
||||||
|
try {
|
||||||
|
return this.getBoolean(index);
|
||||||
|
} catch (Exception e) {
|
||||||
|
return defaultValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the optional double value associated with an index. NaN is returned
|
* Get the optional double value associated with an index. NaN is returned
|
||||||
* if there is no value for the index, or if the value is not a number and
|
* if there is no value for the index, or if the value is not a number and
|
||||||
@@ -635,6 +696,42 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
return doubleValue;
|
return doubleValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional Double object associated with an index. NaN is returned
|
||||||
|
* if there is no value for the index, or if the value is not a number and
|
||||||
|
* cannot be converted to a number.
|
||||||
|
*
|
||||||
|
* @param index
|
||||||
|
* The index must be between 0 and length() - 1.
|
||||||
|
* @return The object.
|
||||||
|
*/
|
||||||
|
public Double optDoubleObject(int index) {
|
||||||
|
return this.optDoubleObject(index, Double.NaN);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional double value associated with an index. The defaultValue
|
||||||
|
* is returned if there is no value for the index, or if the value is not a
|
||||||
|
* number and cannot be converted to a number.
|
||||||
|
*
|
||||||
|
* @param index
|
||||||
|
* subscript
|
||||||
|
* @param defaultValue
|
||||||
|
* The default object.
|
||||||
|
* @return The object.
|
||||||
|
*/
|
||||||
|
public Double optDoubleObject(int index, Double defaultValue) {
|
||||||
|
final Number val = this.optNumber(index, null);
|
||||||
|
if (val == null) {
|
||||||
|
return defaultValue;
|
||||||
|
}
|
||||||
|
final Double doubleValue = val.doubleValue();
|
||||||
|
// if (Double.isNaN(doubleValue) || Double.isInfinite(doubleValue)) {
|
||||||
|
// return defaultValue;
|
||||||
|
// }
|
||||||
|
return doubleValue;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the optional float value associated with an index. NaN is returned
|
* Get the optional float value associated with an index. NaN is returned
|
||||||
* if there is no value for the index, or if the value is not a number and
|
* if there is no value for the index, or if the value is not a number and
|
||||||
@@ -671,6 +768,42 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
return floatValue;
|
return floatValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional Float object associated with an index. NaN is returned
|
||||||
|
* if there is no value for the index, or if the value is not a number and
|
||||||
|
* cannot be converted to a number.
|
||||||
|
*
|
||||||
|
* @param index
|
||||||
|
* The index must be between 0 and length() - 1.
|
||||||
|
* @return The object.
|
||||||
|
*/
|
||||||
|
public Float optFloatObject(int index) {
|
||||||
|
return this.optFloatObject(index, Float.NaN);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional Float object associated with an index. The defaultValue
|
||||||
|
* is returned if there is no value for the index, or if the value is not a
|
||||||
|
* number and cannot be converted to a number.
|
||||||
|
*
|
||||||
|
* @param index
|
||||||
|
* subscript
|
||||||
|
* @param defaultValue
|
||||||
|
* The default object.
|
||||||
|
* @return The object.
|
||||||
|
*/
|
||||||
|
public Float optFloatObject(int index, Float defaultValue) {
|
||||||
|
final Number val = this.optNumber(index, null);
|
||||||
|
if (val == null) {
|
||||||
|
return defaultValue;
|
||||||
|
}
|
||||||
|
final Float floatValue = val.floatValue();
|
||||||
|
// if (Float.isNaN(floatValue) || Float.isInfinite(floatValue)) {
|
||||||
|
// return floatValue;
|
||||||
|
// }
|
||||||
|
return floatValue;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the optional int value associated with an index. Zero is returned if
|
* Get the optional int value associated with an index. Zero is returned if
|
||||||
* there is no value for the index, or if the value is not a number and
|
* there is no value for the index, or if the value is not a number and
|
||||||
@@ -703,6 +836,38 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
return val.intValue();
|
return val.intValue();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional Integer object associated with an index. Zero is returned if
|
||||||
|
* there is no value for the index, or if the value is not a number and
|
||||||
|
* cannot be converted to a number.
|
||||||
|
*
|
||||||
|
* @param index
|
||||||
|
* The index must be between 0 and length() - 1.
|
||||||
|
* @return The object.
|
||||||
|
*/
|
||||||
|
public Integer optIntegerObject(int index) {
|
||||||
|
return this.optIntegerObject(index, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional Integer object associated with an index. The defaultValue is
|
||||||
|
* returned if there is no value for the index, or if the value is not a
|
||||||
|
* number and cannot be converted to a number.
|
||||||
|
*
|
||||||
|
* @param index
|
||||||
|
* The index must be between 0 and length() - 1.
|
||||||
|
* @param defaultValue
|
||||||
|
* The default object.
|
||||||
|
* @return The object.
|
||||||
|
*/
|
||||||
|
public Integer optIntegerObject(int index, Integer defaultValue) {
|
||||||
|
final Number val = this.optNumber(index, null);
|
||||||
|
if (val == null) {
|
||||||
|
return defaultValue;
|
||||||
|
}
|
||||||
|
return val.intValue();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the enum value associated with a key.
|
* Get the enum value associated with a key.
|
||||||
*
|
*
|
||||||
@@ -788,30 +953,57 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the optional JSONArray associated with an index.
|
* Get the optional JSONArray associated with an index. Null is returned if
|
||||||
|
* there is no value at that index or if the value is not a JSONArray.
|
||||||
*
|
*
|
||||||
* @param index
|
* @param index
|
||||||
* subscript
|
* The index must be between 0 and length() - 1.
|
||||||
* @return A JSONArray value, or null if the index has no value, or if the
|
* @return A JSONArray value.
|
||||||
* value is not a JSONArray.
|
|
||||||
*/
|
*/
|
||||||
public JSONArray optJSONArray(int index) {
|
public JSONArray optJSONArray(int index) {
|
||||||
Object o = this.opt(index);
|
return this.optJSONArray(index, null);
|
||||||
return o instanceof JSONArray ? (JSONArray) o : null;
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional JSONArray associated with an index. The defaultValue is returned if
|
||||||
|
* there is no value at that index or if the value is not a JSONArray.
|
||||||
|
*
|
||||||
|
* @param index
|
||||||
|
* The index must be between 0 and length() - 1.
|
||||||
|
* @param defaultValue
|
||||||
|
* The default.
|
||||||
|
* @return A JSONArray value.
|
||||||
|
*/
|
||||||
|
public JSONArray optJSONArray(int index, JSONArray defaultValue) {
|
||||||
|
Object object = this.opt(index);
|
||||||
|
return object instanceof JSONArray ? (JSONArray) object : defaultValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the optional JSONObject associated with an index. Null is returned if
|
* Get the optional JSONObject associated with an index. Null is returned if
|
||||||
* the key is not found, or null if the index has no value, or if the value
|
* there is no value at that index or if the value is not a JSONObject.
|
||||||
* is not a JSONObject.
|
|
||||||
*
|
*
|
||||||
* @param index
|
* @param index
|
||||||
* The index must be between 0 and length() - 1.
|
* The index must be between 0 and length() - 1.
|
||||||
* @return A JSONObject value.
|
* @return A JSONObject value.
|
||||||
*/
|
*/
|
||||||
public JSONObject optJSONObject(int index) {
|
public JSONObject optJSONObject(int index) {
|
||||||
Object o = this.opt(index);
|
return this.optJSONObject(index, null);
|
||||||
return o instanceof JSONObject ? (JSONObject) o : null;
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional JSONObject associated with an index. The defaultValue is returned if
|
||||||
|
* there is no value at that index or if the value is not a JSONObject.
|
||||||
|
*
|
||||||
|
* @param index
|
||||||
|
* The index must be between 0 and length() - 1.
|
||||||
|
* @param defaultValue
|
||||||
|
* The default.
|
||||||
|
* @return A JSONObject value.
|
||||||
|
*/
|
||||||
|
public JSONObject optJSONObject(int index, JSONObject defaultValue) {
|
||||||
|
Object object = this.opt(index);
|
||||||
|
return object instanceof JSONObject ? (JSONObject) object : defaultValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -846,6 +1038,38 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
return val.longValue();
|
return val.longValue();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional Long object associated with an index. Zero is returned if
|
||||||
|
* there is no value for the index, or if the value is not a number and
|
||||||
|
* cannot be converted to a number.
|
||||||
|
*
|
||||||
|
* @param index
|
||||||
|
* The index must be between 0 and length() - 1.
|
||||||
|
* @return The object.
|
||||||
|
*/
|
||||||
|
public Long optLongObject(int index) {
|
||||||
|
return this.optLongObject(index, 0L);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional Long object associated with an index. The defaultValue is
|
||||||
|
* returned if there is no value for the index, or if the value is not a
|
||||||
|
* number and cannot be converted to a number.
|
||||||
|
*
|
||||||
|
* @param index
|
||||||
|
* The index must be between 0 and length() - 1.
|
||||||
|
* @param defaultValue
|
||||||
|
* The default object.
|
||||||
|
* @return The object.
|
||||||
|
*/
|
||||||
|
public Long optLongObject(int index, Long defaultValue) {
|
||||||
|
final Number val = this.optNumber(index, null);
|
||||||
|
if (val == null) {
|
||||||
|
return defaultValue;
|
||||||
|
}
|
||||||
|
return val.longValue();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get an optional {@link Number} value associated with a key, or <code>null</code>
|
* Get an optional {@link Number} value associated with a key, or <code>null</code>
|
||||||
* if there is no such key or if the value is not a number. If the value is a string,
|
* if there is no such key or if the value is not a number. If the value is a string,
|
||||||
@@ -1135,7 +1359,8 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
* The subscript.
|
* The subscript.
|
||||||
* @param value
|
* @param value
|
||||||
* The Map value.
|
* The Map value.
|
||||||
* @return this.
|
* @return
|
||||||
|
* reference to self
|
||||||
* @throws JSONException
|
* @throws JSONException
|
||||||
* If the index is negative or if the value is an invalid
|
* If the index is negative or if the value is an invalid
|
||||||
* number.
|
* number.
|
||||||
@@ -1143,7 +1368,27 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
* If a key in the map is <code>null</code>
|
* If a key in the map is <code>null</code>
|
||||||
*/
|
*/
|
||||||
public JSONArray put(int index, Map<?, ?> value) throws JSONException {
|
public JSONArray put(int index, Map<?, ?> value) throws JSONException {
|
||||||
this.put(index, new JSONObject(value));
|
this.put(index, new JSONObject(value, new JSONParserConfiguration()));
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Put a value in the JSONArray, where the value will be a JSONObject that
|
||||||
|
* is produced from a Map.
|
||||||
|
*
|
||||||
|
* @param index
|
||||||
|
* The subscript
|
||||||
|
* @param value
|
||||||
|
* The Map value.
|
||||||
|
* @param jsonParserConfiguration
|
||||||
|
* Configuration object for the JSON parser
|
||||||
|
* @return reference to self
|
||||||
|
* @throws JSONException
|
||||||
|
* If the index is negative or if the value is an invalid
|
||||||
|
* number.
|
||||||
|
*/
|
||||||
|
public JSONArray put(int index, Map<?, ?> value, JSONParserConfiguration jsonParserConfiguration) throws JSONException {
|
||||||
|
this.put(index, new JSONObject(value, jsonParserConfiguration));
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1451,10 +1696,8 @@ 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();
|
StringWriter sw = new StringWriter();
|
||||||
synchronized (sw.getBuffer()) {
|
|
||||||
return this.write(sw, indentFactor, 0).toString();
|
return this.write(sw, indentFactor, 0).toString();
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Write the contents of the JSONArray as JSON text to a writer. For
|
* Write the contents of the JSONArray as JSON text to a writer. For
|
||||||
@@ -1586,13 +1829,14 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
* @param wrap
|
* @param wrap
|
||||||
* {@code true} to call {@link JSONObject#wrap(Object)} for each item,
|
* {@code true} to call {@link JSONObject#wrap(Object)} for each item,
|
||||||
* {@code false} to add the items directly
|
* {@code false} to add the items directly
|
||||||
*
|
* @param recursionDepth
|
||||||
|
* Variable for tracking the count of nested object creations.
|
||||||
*/
|
*/
|
||||||
private void addAll(Collection<?> collection, boolean wrap) {
|
private void addAll(Collection<?> collection, boolean wrap, int recursionDepth, JSONParserConfiguration jsonParserConfiguration) {
|
||||||
this.myArrayList.ensureCapacity(this.myArrayList.size() + collection.size());
|
this.myArrayList.ensureCapacity(this.myArrayList.size() + collection.size());
|
||||||
if (wrap) {
|
if (wrap) {
|
||||||
for (Object o: collection){
|
for (Object o: collection){
|
||||||
this.put(JSONObject.wrap(o));
|
this.put(JSONObject.wrap(o, recursionDepth + 1, jsonParserConfiguration));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
for (Object o: collection){
|
for (Object o: collection){
|
||||||
@@ -1632,19 +1876,55 @@ public class JSONArray implements Iterable<Object> {
|
|||||||
* @param wrap
|
* @param wrap
|
||||||
* {@code true} to call {@link JSONObject#wrap(Object)} for each item,
|
* {@code true} to call {@link JSONObject#wrap(Object)} for each item,
|
||||||
* {@code false} to add the items directly
|
* {@code false} to add the items directly
|
||||||
|
* @throws JSONException
|
||||||
|
* If not an array or if an array value is non-finite number.
|
||||||
|
*/
|
||||||
|
private void addAll(Object array, boolean wrap) throws JSONException {
|
||||||
|
this.addAll(array, wrap, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add an array's elements to the JSONArray.
|
||||||
*
|
*
|
||||||
|
* @param array
|
||||||
|
* Array. If the parameter passed is null, or not an array,
|
||||||
|
* JSONArray, Collection, or Iterable, an exception will be
|
||||||
|
* thrown.
|
||||||
|
* @param wrap
|
||||||
|
* {@code true} to call {@link JSONObject#wrap(Object)} for each item,
|
||||||
|
* {@code false} to add the items directly
|
||||||
|
* @param recursionDepth
|
||||||
|
* Variable for tracking the count of nested object creations.
|
||||||
|
*/
|
||||||
|
private void addAll(Object array, boolean wrap, int recursionDepth) {
|
||||||
|
addAll(array, wrap, recursionDepth, new JSONParserConfiguration());
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Add an array's elements to the JSONArray.
|
||||||
|
*`
|
||||||
|
* @param array
|
||||||
|
* Array. If the parameter passed is null, or not an array,
|
||||||
|
* JSONArray, Collection, or Iterable, an exception will be
|
||||||
|
* thrown.
|
||||||
|
* @param wrap
|
||||||
|
* {@code true} to call {@link JSONObject#wrap(Object)} for each item,
|
||||||
|
* {@code false} to add the items directly
|
||||||
|
* @param recursionDepth
|
||||||
|
* Variable for tracking the count of nested object creations.
|
||||||
|
* @param jsonParserConfiguration
|
||||||
|
* Variable to pass parser custom configuration for json parsing.
|
||||||
* @throws JSONException
|
* @throws JSONException
|
||||||
* If not an array or if an array value is non-finite number.
|
* If not an array or if an array value is non-finite number.
|
||||||
* @throws NullPointerException
|
* @throws NullPointerException
|
||||||
* Thrown if the array parameter is null.
|
* Thrown if the array parameter is null.
|
||||||
*/
|
*/
|
||||||
private void addAll(Object array, boolean wrap) throws JSONException {
|
private void addAll(Object array, boolean wrap, int recursionDepth, JSONParserConfiguration jsonParserConfiguration) throws JSONException {
|
||||||
if (array.getClass().isArray()) {
|
if (array.getClass().isArray()) {
|
||||||
int length = Array.getLength(array);
|
int length = Array.getLength(array);
|
||||||
this.myArrayList.ensureCapacity(this.myArrayList.size() + length);
|
this.myArrayList.ensureCapacity(this.myArrayList.size() + length);
|
||||||
if (wrap) {
|
if (wrap) {
|
||||||
for (int i = 0; i < length; i += 1) {
|
for (int i = 0; i < length; i += 1) {
|
||||||
this.put(JSONObject.wrap(Array.get(array, i)));
|
this.put(JSONObject.wrap(Array.get(array, i), recursionDepth + 1, jsonParserConfiguration));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
for (int i = 0; i < length; i += 1) {
|
for (int i = 0; i < length; i += 1) {
|
||||||
@@ -1657,7 +1937,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);
|
this.addAll((Collection<?>)array, wrap, recursionDepth);
|
||||||
} 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.
|
||||||
|
|||||||
@@ -55,11 +55,13 @@ public class JSONMLParserConfiguration extends ParserConfiguration {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
@Override
|
@Override
|
||||||
public JSONMLParserConfiguration withKeepStrings(final boolean newVal) {
|
public JSONMLParserConfiguration withKeepStrings(final boolean newVal) {
|
||||||
return super.withKeepStrings(newVal);
|
return super.withKeepStrings(newVal);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
@Override
|
@Override
|
||||||
public JSONMLParserConfiguration withMaxNestingDepth(int maxNestingDepth) {
|
public JSONMLParserConfiguration withMaxNestingDepth(int maxNestingDepth) {
|
||||||
return super.withMaxNestingDepth(maxNestingDepth);
|
return super.withMaxNestingDepth(maxNestingDepth);
|
||||||
|
|||||||
@@ -145,6 +145,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();
|
||||||
}
|
}
|
||||||
@@ -208,22 +213,14 @@ public class JSONObject {
|
|||||||
throw x.syntaxError("A JSONObject text must begin with '{'");
|
throw x.syntaxError("A JSONObject text must begin with '{'");
|
||||||
}
|
}
|
||||||
for (;;) {
|
for (;;) {
|
||||||
char prev = x.getPrevious();
|
|
||||||
c = x.nextClean();
|
c = x.nextClean();
|
||||||
switch (c) {
|
switch (c) {
|
||||||
case 0:
|
case 0:
|
||||||
throw x.syntaxError("A JSONObject text must end with '}'");
|
throw x.syntaxError("A JSONObject text must end with '}'");
|
||||||
case '}':
|
case '}':
|
||||||
return;
|
return;
|
||||||
case '{':
|
|
||||||
case '[':
|
|
||||||
if(prev=='{') {
|
|
||||||
throw x.syntaxError("A JSON Object can not directly nest another JSON Object or JSON Array.");
|
|
||||||
}
|
|
||||||
// fall through
|
|
||||||
default:
|
default:
|
||||||
x.back();
|
key = x.nextSimpleValue(c).toString();
|
||||||
key = x.nextValue().toString();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// The key is followed by ':'.
|
// The key is followed by ':'.
|
||||||
@@ -256,6 +253,9 @@ public class JSONObject {
|
|||||||
if (x.nextClean() == '}') {
|
if (x.nextClean() == '}') {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
if (x.end()) {
|
||||||
|
throw x.syntaxError("A JSONObject text must end with '}'");
|
||||||
|
}
|
||||||
x.back();
|
x.back();
|
||||||
break;
|
break;
|
||||||
case '}':
|
case '}':
|
||||||
@@ -278,6 +278,30 @@ public class JSONObject {
|
|||||||
* If a key in the map is <code>null</code>
|
* If a key in the map is <code>null</code>
|
||||||
*/
|
*/
|
||||||
public JSONObject(Map<?, ?> m) {
|
public JSONObject(Map<?, ?> m) {
|
||||||
|
this(m, 0, new JSONParserConfiguration());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a JSONObject from a Map with custom json parse configurations.
|
||||||
|
*
|
||||||
|
* @param m
|
||||||
|
* A map object that can be used to initialize the contents of
|
||||||
|
* the JSONObject.
|
||||||
|
* @param jsonParserConfiguration
|
||||||
|
* Variable to pass parser custom configuration for json parsing.
|
||||||
|
*/
|
||||||
|
public JSONObject(Map<?, ?> m, JSONParserConfiguration jsonParserConfiguration) {
|
||||||
|
this(m, 0, jsonParserConfiguration);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a JSONObject from a map with recursion depth.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
private JSONObject(Map<?, ?> m, int recursionDepth, JSONParserConfiguration jsonParserConfiguration) {
|
||||||
|
if (recursionDepth > jsonParserConfiguration.getMaxNestingDepth()) {
|
||||||
|
throw new JSONException("JSONObject has reached recursion depth limit of " + jsonParserConfiguration.getMaxNestingDepth());
|
||||||
|
}
|
||||||
if (m == null) {
|
if (m == null) {
|
||||||
this.map = new HashMap<String, Object>();
|
this.map = new HashMap<String, Object>();
|
||||||
} else {
|
} else {
|
||||||
@@ -288,7 +312,8 @@ public class JSONObject {
|
|||||||
}
|
}
|
||||||
final Object value = e.getValue();
|
final Object value = e.getValue();
|
||||||
if (value != null) {
|
if (value != null) {
|
||||||
this.map.put(String.valueOf(e.getKey()), wrap(value));
|
testValidity(value);
|
||||||
|
this.map.put(String.valueOf(e.getKey()), wrap(value, recursionDepth + 1, jsonParserConfiguration));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -346,11 +371,12 @@ 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
|
||||||
* a JSONObject.
|
* a JSONObject.
|
||||||
|
* @throws JSONException
|
||||||
|
* If a getter returned a non-finite number.
|
||||||
*/
|
*/
|
||||||
public JSONObject(Object bean) {
|
public JSONObject(Object bean) {
|
||||||
this();
|
this();
|
||||||
@@ -1131,6 +1157,45 @@ public class JSONObject {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get an optional boolean object associated with a key. It returns false if there
|
||||||
|
* is no such key, or if the value is not Boolean.TRUE or the String "true".
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* A key string.
|
||||||
|
* @return The truth.
|
||||||
|
*/
|
||||||
|
public Boolean optBooleanObject(String key) {
|
||||||
|
return this.optBooleanObject(key, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get an optional boolean object associated with a key. It returns the
|
||||||
|
* defaultValue if there is no such key, or if it is not a Boolean or the
|
||||||
|
* String "true" or "false" (case insensitive).
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* A key string.
|
||||||
|
* @param defaultValue
|
||||||
|
* The default.
|
||||||
|
* @return The truth.
|
||||||
|
*/
|
||||||
|
public Boolean optBooleanObject(String key, Boolean defaultValue) {
|
||||||
|
Object val = this.opt(key);
|
||||||
|
if (NULL.equals(val)) {
|
||||||
|
return defaultValue;
|
||||||
|
}
|
||||||
|
if (val instanceof Boolean){
|
||||||
|
return ((Boolean) val).booleanValue();
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
// we'll use the get anyway because it does string conversion.
|
||||||
|
return this.getBoolean(key);
|
||||||
|
} catch (Exception e) {
|
||||||
|
return defaultValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get an optional BigDecimal associated with a key, or the defaultValue if
|
* Get an optional BigDecimal associated with a key, or the defaultValue if
|
||||||
* there is no such key or if its value is not a number. If the value is a
|
* there is no such key or if its value is not a number. If the value is a
|
||||||
@@ -1294,7 +1359,39 @@ public class JSONObject {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the optional double value associated with an index. NaN is returned
|
* Get an optional Double object associated with a key, or NaN if there is no such
|
||||||
|
* key or if its value is not a number. If the value is a string, an attempt
|
||||||
|
* will be made to evaluate it as a number.
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* A string which is the key.
|
||||||
|
* @return An object which is the value.
|
||||||
|
*/
|
||||||
|
public Double optDoubleObject(String key) {
|
||||||
|
return this.optDoubleObject(key, Double.NaN);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get an optional Double object associated with a key, or the defaultValue if
|
||||||
|
* there is no such key or if its value is not a number. If the value is a
|
||||||
|
* string, an attempt will be made to evaluate it as a number.
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* A key string.
|
||||||
|
* @param defaultValue
|
||||||
|
* The default.
|
||||||
|
* @return An object which is the value.
|
||||||
|
*/
|
||||||
|
public Double optDoubleObject(String key, Double defaultValue) {
|
||||||
|
Number val = this.optNumber(key);
|
||||||
|
if (val == null) {
|
||||||
|
return defaultValue;
|
||||||
|
}
|
||||||
|
return val.doubleValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional float value associated with an index. NaN is returned
|
||||||
* if there is no value for the index, or if the value is not a number and
|
* if there is no value for the index, or if the value is not a number and
|
||||||
* cannot be converted to a number.
|
* cannot be converted to a number.
|
||||||
*
|
*
|
||||||
@@ -1307,7 +1404,7 @@ public class JSONObject {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the optional double value associated with an index. The defaultValue
|
* Get the optional float value associated with an index. The defaultValue
|
||||||
* is returned if there is no value for the index, or if the value is not a
|
* is returned if there is no value for the index, or if the value is not a
|
||||||
* number and cannot be converted to a number.
|
* number and cannot be converted to a number.
|
||||||
*
|
*
|
||||||
@@ -1329,6 +1426,42 @@ public class JSONObject {
|
|||||||
return floatValue;
|
return floatValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional Float object associated with an index. NaN is returned
|
||||||
|
* if there is no value for the index, or if the value is not a number and
|
||||||
|
* cannot be converted to a number.
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* A key string.
|
||||||
|
* @return The object.
|
||||||
|
*/
|
||||||
|
public Float optFloatObject(String key) {
|
||||||
|
return this.optFloatObject(key, Float.NaN);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the optional Float object associated with an index. The defaultValue
|
||||||
|
* is returned if there is no value for the index, or if the value is not a
|
||||||
|
* number and cannot be converted to a number.
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* A key string.
|
||||||
|
* @param defaultValue
|
||||||
|
* The default object.
|
||||||
|
* @return The object.
|
||||||
|
*/
|
||||||
|
public Float optFloatObject(String key, Float defaultValue) {
|
||||||
|
Number val = this.optNumber(key);
|
||||||
|
if (val == null) {
|
||||||
|
return defaultValue;
|
||||||
|
}
|
||||||
|
final Float floatValue = val.floatValue();
|
||||||
|
// if (Float.isNaN(floatValue) || Float.isInfinite(floatValue)) {
|
||||||
|
// return defaultValue;
|
||||||
|
// }
|
||||||
|
return floatValue;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get an optional int value associated with a key, or zero if there is no
|
* Get an optional int value associated with a key, or zero if there is no
|
||||||
* such key or if the value is not a number. If the value is a string, an
|
* such key or if the value is not a number. If the value is a string, an
|
||||||
@@ -1361,6 +1494,38 @@ public class JSONObject {
|
|||||||
return val.intValue();
|
return val.intValue();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get an optional Integer object associated with a key, or zero if there is no
|
||||||
|
* such key or if the value is not a number. If the value is a string, an
|
||||||
|
* attempt will be made to evaluate it as a number.
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* A key string.
|
||||||
|
* @return An object which is the value.
|
||||||
|
*/
|
||||||
|
public Integer optIntegerObject(String key) {
|
||||||
|
return this.optIntegerObject(key, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get an optional Integer object associated with a key, or the default if there
|
||||||
|
* is no such key or if the value is not a number. If the value is a string,
|
||||||
|
* an attempt will be made to evaluate it as a number.
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* A key string.
|
||||||
|
* @param defaultValue
|
||||||
|
* The default.
|
||||||
|
* @return An object which is the value.
|
||||||
|
*/
|
||||||
|
public Integer optIntegerObject(String key, Integer defaultValue) {
|
||||||
|
final Number val = this.optNumber(key, null);
|
||||||
|
if (val == null) {
|
||||||
|
return defaultValue;
|
||||||
|
}
|
||||||
|
return val.intValue();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get an optional JSONArray associated with a key. It returns null if there
|
* Get an optional JSONArray associated with a key. It returns null if there
|
||||||
* is no such key, or if its value is not a JSONArray.
|
* is no such key, or if its value is not a JSONArray.
|
||||||
@@ -1370,8 +1535,22 @@ public class JSONObject {
|
|||||||
* @return A JSONArray which is the value.
|
* @return A JSONArray which is the value.
|
||||||
*/
|
*/
|
||||||
public JSONArray optJSONArray(String key) {
|
public JSONArray optJSONArray(String key) {
|
||||||
Object o = this.opt(key);
|
return this.optJSONArray(key, null);
|
||||||
return o instanceof JSONArray ? (JSONArray) o : null;
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get an optional JSONArray associated with a key, or the default if there
|
||||||
|
* is no such key, or if its value is not a JSONArray.
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* A key string.
|
||||||
|
* @param defaultValue
|
||||||
|
* The default.
|
||||||
|
* @return A JSONArray which is the value.
|
||||||
|
*/
|
||||||
|
public JSONArray optJSONArray(String key, JSONArray defaultValue) {
|
||||||
|
Object object = this.opt(key);
|
||||||
|
return object instanceof JSONArray ? (JSONArray) object : defaultValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1432,6 +1611,39 @@ public class JSONObject {
|
|||||||
return val.longValue();
|
return val.longValue();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get an optional Long object associated with a key, or zero if there is no
|
||||||
|
* such key or if the value is not a number. If the value is a string, an
|
||||||
|
* attempt will be made to evaluate it as a number.
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* A key string.
|
||||||
|
* @return An object which is the value.
|
||||||
|
*/
|
||||||
|
public Long optLongObject(String key) {
|
||||||
|
return this.optLongObject(key, 0L);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get an optional Long object associated with a key, or the default if there
|
||||||
|
* is no such key or if the value is not a number. If the value is a string,
|
||||||
|
* an attempt will be made to evaluate it as a number.
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* A key string.
|
||||||
|
* @param defaultValue
|
||||||
|
* The default.
|
||||||
|
* @return An object which is the value.
|
||||||
|
*/
|
||||||
|
public Long optLongObject(String key, Long defaultValue) {
|
||||||
|
final Number val = this.optNumber(key, null);
|
||||||
|
if (val == null) {
|
||||||
|
return defaultValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
return val.longValue();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get an optional {@link Number} value associated with a key, or <code>null</code>
|
* Get an optional {@link Number} value associated with a key, or <code>null</code>
|
||||||
* if there is no such key or if the value is not a number. If the value is a string,
|
* if there is no such key or if the value is not a number. If the value is a string,
|
||||||
@@ -1510,6 +1722,8 @@ public class JSONObject {
|
|||||||
*
|
*
|
||||||
* @param bean
|
* @param bean
|
||||||
* the bean
|
* the bean
|
||||||
|
* @throws JSONException
|
||||||
|
* If a getter returned a non-finite number.
|
||||||
*/
|
*/
|
||||||
private void populateMap(Object bean) {
|
private void populateMap(Object bean) {
|
||||||
populateMap(bean, Collections.newSetFromMap(new IdentityHashMap<Object, Boolean>()));
|
populateMap(bean, Collections.newSetFromMap(new IdentityHashMap<Object, Boolean>()));
|
||||||
@@ -1545,6 +1759,7 @@ public class JSONObject {
|
|||||||
|
|
||||||
objectsRecord.add(result);
|
objectsRecord.add(result);
|
||||||
|
|
||||||
|
testValidity(result);
|
||||||
this.map.put(key, wrap(result, objectsRecord));
|
this.map.put(key, wrap(result, objectsRecord));
|
||||||
|
|
||||||
objectsRecord.remove(result);
|
objectsRecord.remove(result);
|
||||||
@@ -1651,6 +1866,10 @@ public class JSONObject {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//If the superclass is Object, no annotations will be found any more
|
||||||
|
if (c.getSuperclass().equals(Object.class))
|
||||||
|
return null;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
return getAnnotation(
|
return getAnnotation(
|
||||||
c.getSuperclass().getMethod(m.getName(), m.getParameterTypes()),
|
c.getSuperclass().getMethod(m.getName(), m.getParameterTypes()),
|
||||||
@@ -1705,6 +1924,10 @@ public class JSONObject {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//If the superclass is Object, no annotations will be found any more
|
||||||
|
if (c.getSuperclass().equals(Object.class))
|
||||||
|
return -1;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
int d = getAnnotationDepth(
|
int d = getAnnotationDepth(
|
||||||
c.getSuperclass().getMethod(m.getName(), m.getParameterTypes()),
|
c.getSuperclass().getMethod(m.getName(), m.getParameterTypes()),
|
||||||
@@ -2002,7 +2225,6 @@ public class JSONObject {
|
|||||||
@SuppressWarnings("resource")
|
@SuppressWarnings("resource")
|
||||||
public static String quote(String string) {
|
public static String quote(String string) {
|
||||||
StringWriter sw = new StringWriter();
|
StringWriter sw = new StringWriter();
|
||||||
synchronized (sw.getBuffer()) {
|
|
||||||
try {
|
try {
|
||||||
return quote(string, sw).toString();
|
return quote(string, sw).toString();
|
||||||
} catch (IOException ignored) {
|
} catch (IOException ignored) {
|
||||||
@@ -2010,8 +2232,15 @@ public class JSONObject {
|
|||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 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("\"\"");
|
||||||
@@ -2195,6 +2424,49 @@ public class JSONObject {
|
|||||||
|| val.indexOf('E') > -1 || "-0".equals(val);
|
|| val.indexOf('E') > -1 || "-0".equals(val);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Try to convert a string into a number, boolean, or null. If the string
|
||||||
|
* can't be converted, return the string.
|
||||||
|
*
|
||||||
|
* @param string
|
||||||
|
* A String. can not be null.
|
||||||
|
* @return A simple JSON value.
|
||||||
|
* @throws NullPointerException
|
||||||
|
* Thrown if the string is null.
|
||||||
|
*/
|
||||||
|
// Changes to this method must be copied to the corresponding method in
|
||||||
|
// the XML class to keep full support for Android
|
||||||
|
public static Object stringToValue(String string) {
|
||||||
|
if ("".equals(string)) {
|
||||||
|
return string;
|
||||||
|
}
|
||||||
|
|
||||||
|
// check JSON key words true/false/null
|
||||||
|
if ("true".equalsIgnoreCase(string)) {
|
||||||
|
return Boolean.TRUE;
|
||||||
|
}
|
||||||
|
if ("false".equalsIgnoreCase(string)) {
|
||||||
|
return Boolean.FALSE;
|
||||||
|
}
|
||||||
|
if ("null".equalsIgnoreCase(string)) {
|
||||||
|
return JSONObject.NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If it might be a number, try converting it. If a number cannot be
|
||||||
|
* produced, then the value will just be a string.
|
||||||
|
*/
|
||||||
|
|
||||||
|
char initial = string.charAt(0);
|
||||||
|
if ((initial >= '0' && initial <= '9') || initial == '-') {
|
||||||
|
try {
|
||||||
|
return stringToNumber(string);
|
||||||
|
} catch (Exception ignore) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return string;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Converts a string to a number using the narrowest possible type. Possible
|
* Converts a string to a number using the narrowest possible type. Possible
|
||||||
* returns for this function are BigDecimal, Double, BigInteger, Long, and Integer.
|
* returns for this function are BigDecimal, Double, BigInteger, Long, and Integer.
|
||||||
@@ -2265,49 +2537,6 @@ public class JSONObject {
|
|||||||
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Try to convert a string into a number, boolean, or null. If the string
|
|
||||||
* can't be converted, return the string.
|
|
||||||
*
|
|
||||||
* @param string
|
|
||||||
* A String. can not be null.
|
|
||||||
* @return A simple JSON value.
|
|
||||||
* @throws NullPointerException
|
|
||||||
* Thrown if the string is null.
|
|
||||||
*/
|
|
||||||
// Changes to this method must be copied to the corresponding method in
|
|
||||||
// the XML class to keep full support for Android
|
|
||||||
public static Object stringToValue(String string) {
|
|
||||||
if ("".equals(string)) {
|
|
||||||
return string;
|
|
||||||
}
|
|
||||||
|
|
||||||
// check JSON key words true/false/null
|
|
||||||
if ("true".equalsIgnoreCase(string)) {
|
|
||||||
return Boolean.TRUE;
|
|
||||||
}
|
|
||||||
if ("false".equalsIgnoreCase(string)) {
|
|
||||||
return Boolean.FALSE;
|
|
||||||
}
|
|
||||||
if ("null".equalsIgnoreCase(string)) {
|
|
||||||
return JSONObject.NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* If it might be a number, try converting it. If a number cannot be
|
|
||||||
* produced, then the value will just be a string.
|
|
||||||
*/
|
|
||||||
|
|
||||||
char initial = string.charAt(0);
|
|
||||||
if ((initial >= '0' && initial <= '9') || initial == '-') {
|
|
||||||
try {
|
|
||||||
return stringToNumber(string);
|
|
||||||
} catch (Exception ignore) {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return string;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Throw an exception if the object is a NaN or infinite number.
|
* Throw an exception if the object is a NaN or infinite number.
|
||||||
*
|
*
|
||||||
@@ -2395,10 +2624,8 @@ 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();
|
StringWriter w = new StringWriter();
|
||||||
synchronized (w.getBuffer()) {
|
|
||||||
return this.write(w, indentFactor, 0).toString();
|
return this.write(w, indentFactor, 0).toString();
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Make a JSON text of an Object value. If the object has an
|
* Make a JSON text of an Object value. If the object has an
|
||||||
@@ -2448,7 +2675,31 @@ public class JSONObject {
|
|||||||
return wrap(object, null);
|
return wrap(object, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wrap an object, if necessary. If the object is <code>null</code>, return the NULL
|
||||||
|
* object. If it is an array or collection, wrap it in a JSONArray. If it is
|
||||||
|
* a map, wrap it in a JSONObject. If it is a standard property (Double,
|
||||||
|
* String, et al) then it is already wrapped. Otherwise, if it comes from
|
||||||
|
* one of the java packages, turn it into a string. And if it doesn't, try
|
||||||
|
* to wrap it in a JSONObject. If the wrapping fails, then null is returned.
|
||||||
|
*
|
||||||
|
* @param object
|
||||||
|
* The object to wrap
|
||||||
|
* @param recursionDepth
|
||||||
|
* Variable for tracking the count of nested object creations.
|
||||||
|
* @param jsonParserConfiguration
|
||||||
|
* Variable to pass parser custom configuration for json parsing.
|
||||||
|
* @return The wrapped value
|
||||||
|
*/
|
||||||
|
static Object wrap(Object object, int recursionDepth, JSONParserConfiguration jsonParserConfiguration) {
|
||||||
|
return wrap(object, null, recursionDepth, jsonParserConfiguration);
|
||||||
|
}
|
||||||
|
|
||||||
private static Object wrap(Object object, Set<Object> objectsRecord) {
|
private static Object wrap(Object object, Set<Object> objectsRecord) {
|
||||||
|
return wrap(object, objectsRecord, 0, new JSONParserConfiguration());
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Object wrap(Object object, Set<Object> objectsRecord, int recursionDepth, JSONParserConfiguration jsonParserConfiguration) {
|
||||||
try {
|
try {
|
||||||
if (NULL.equals(object)) {
|
if (NULL.equals(object)) {
|
||||||
return NULL;
|
return NULL;
|
||||||
@@ -2466,14 +2717,14 @@ public class JSONObject {
|
|||||||
|
|
||||||
if (object instanceof Collection) {
|
if (object instanceof Collection) {
|
||||||
Collection<?> coll = (Collection<?>) object;
|
Collection<?> coll = (Collection<?>) object;
|
||||||
return new JSONArray(coll);
|
return new JSONArray(coll, recursionDepth, jsonParserConfiguration);
|
||||||
}
|
}
|
||||||
if (object.getClass().isArray()) {
|
if (object.getClass().isArray()) {
|
||||||
return new JSONArray(object);
|
return new JSONArray(object);
|
||||||
}
|
}
|
||||||
if (object instanceof Map) {
|
if (object instanceof Map) {
|
||||||
Map<?, ?> map = (Map<?, ?>) object;
|
Map<?, ?> map = (Map<?, ?>) object;
|
||||||
return new JSONObject(map);
|
return new JSONObject(map, recursionDepth, jsonParserConfiguration);
|
||||||
}
|
}
|
||||||
Package objectPackage = object.getClass().getPackage();
|
Package objectPackage = object.getClass().getPackage();
|
||||||
String objectPackageName = objectPackage != null ? objectPackage
|
String objectPackageName = objectPackage != null ? objectPackage
|
||||||
@@ -2709,4 +2960,24 @@ public class JSONObject {
|
|||||||
"JavaBean object contains recursively defined member variable of key " + quote(key)
|
"JavaBean object contains recursively defined member variable of key " + quote(key)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 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";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
26
src/main/java/org/json/JSONParserConfiguration.java
Normal file
26
src/main/java/org/json/JSONParserConfiguration.java
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
package org.json;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configuration object for the JSON parser. The configuration is immutable.
|
||||||
|
*/
|
||||||
|
public class JSONParserConfiguration extends ParserConfiguration {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configuration with the default values.
|
||||||
|
*/
|
||||||
|
public JSONParserConfiguration() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected JSONParserConfiguration clone() {
|
||||||
|
return new JSONParserConfiguration();
|
||||||
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
@Override
|
||||||
|
public JSONParserConfiguration withMaxNestingDepth(final int maxNestingDepth) {
|
||||||
|
return super.withMaxNestingDepth(maxNestingDepth);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -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();
|
||||||
|
|||||||
@@ -402,12 +402,7 @@ public class JSONTokener {
|
|||||||
*/
|
*/
|
||||||
public Object nextValue() throws JSONException {
|
public Object nextValue() throws JSONException {
|
||||||
char c = this.nextClean();
|
char c = this.nextClean();
|
||||||
String string;
|
|
||||||
|
|
||||||
switch (c) {
|
switch (c) {
|
||||||
case '"':
|
|
||||||
case '\'':
|
|
||||||
return this.nextString(c);
|
|
||||||
case '{':
|
case '{':
|
||||||
this.back();
|
this.back();
|
||||||
try {
|
try {
|
||||||
@@ -423,6 +418,17 @@ public class JSONTokener {
|
|||||||
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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return nextSimpleValue(c);
|
||||||
|
}
|
||||||
|
|
||||||
|
Object nextSimpleValue(char c) {
|
||||||
|
String string;
|
||||||
|
|
||||||
|
switch (c) {
|
||||||
|
case '"':
|
||||||
|
case '\'':
|
||||||
|
return this.nextString(c);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Handle unquoted text. This could be the values true, false, or
|
* Handle unquoted text. This could be the values true, false, or
|
||||||
@@ -519,6 +525,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();
|
||||||
|
|||||||
@@ -29,11 +29,20 @@ public class ParserConfiguration {
|
|||||||
*/
|
*/
|
||||||
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;
|
||||||
@@ -71,9 +80,11 @@ public class ParserConfiguration {
|
|||||||
*
|
*
|
||||||
* @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
|
||||||
*
|
*
|
||||||
* @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")
|
||||||
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;
|
||||||
@@ -96,8 +107,11 @@ public class ParserConfiguration {
|
|||||||
* 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
|
||||||
|
*
|
||||||
* @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")
|
||||||
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();
|
||||||
|
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -10,7 +10,6 @@ import java.math.BigDecimal;
|
|||||||
import java.math.BigInteger;
|
import java.math.BigInteger;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This provides static methods to convert an XML text into a JSONObject, and to
|
* This provides static methods to convert an XML text into a JSONObject, and to
|
||||||
* covert a JSONObject into an XML text.
|
* covert a JSONObject into an XML text.
|
||||||
@@ -21,6 +20,12 @@ import java.util.Iterator;
|
|||||||
@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 = '&';
|
||||||
|
|
||||||
@@ -53,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";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -428,6 +436,9 @@ public class XML {
|
|||||||
&& jsonObject.opt(config.getcDataTagName()) != null) {
|
&& jsonObject.opt(config.getcDataTagName()) != null) {
|
||||||
context.accumulate(tagName, jsonObject.opt(config.getcDataTagName()));
|
context.accumulate(tagName, jsonObject.opt(config.getcDataTagName()));
|
||||||
} else {
|
} else {
|
||||||
|
if (!config.shouldTrimWhiteSpace()) {
|
||||||
|
removeEmpty(jsonObject, config);
|
||||||
|
}
|
||||||
context.accumulate(tagName, jsonObject);
|
context.accumulate(tagName, jsonObject);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -442,58 +453,46 @@ public class XML {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method tries to convert the given string value to the target object
|
* This method removes any JSON entry which has the key set by XMLParserConfiguration.cDataTagName
|
||||||
* @param string String to convert
|
* and contains whitespace as this is caused by whitespace between tags. See test XMLTest.testNestedWithWhitespaceTrimmingDisabled.
|
||||||
* @param typeConverter value converter to convert string to integer, boolean e.t.c
|
* @param jsonObject JSONObject which may require deletion
|
||||||
* @return JSON value of this string or the string
|
* @param config The XMLParserConfiguration which includes the cDataTagName
|
||||||
*/
|
*/
|
||||||
public static Object stringToValue(String string, XMLXsiTypeConverter<?> typeConverter) {
|
private static void removeEmpty(final JSONObject jsonObject, final XMLParserConfiguration config) {
|
||||||
if(typeConverter != null) {
|
if (jsonObject.has(config.getcDataTagName())) {
|
||||||
return typeConverter.convert(string);
|
final Object s = jsonObject.get(config.getcDataTagName());
|
||||||
|
if (s instanceof String) {
|
||||||
|
if (isStringAllWhiteSpace(s.toString())) {
|
||||||
|
jsonObject.remove(config.getcDataTagName());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (s instanceof JSONArray) {
|
||||||
|
final JSONArray sArray = (JSONArray) s;
|
||||||
|
for (int k = sArray.length()-1; k >= 0; k--){
|
||||||
|
final Object eachString = sArray.get(k);
|
||||||
|
if (eachString instanceof String) {
|
||||||
|
String s1 = (String) eachString;
|
||||||
|
if (isStringAllWhiteSpace(s1)) {
|
||||||
|
sArray.remove(k);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (sArray.isEmpty()) {
|
||||||
|
jsonObject.remove(config.getcDataTagName());
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return stringToValue(string);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
private static boolean isStringAllWhiteSpace(final String s) {
|
||||||
* This method is the same as {@link JSONObject#stringToValue(String)}.
|
for (int k = 0; k<s.length(); k++){
|
||||||
*
|
final char eachChar = s.charAt(k);
|
||||||
* @param string String to convert
|
if (!Character.isWhitespace(eachChar)) {
|
||||||
* @return JSON value of this string or the string
|
return false;
|
||||||
*/
|
|
||||||
// To maintain compatibility with the Android API, this method is a direct copy of
|
|
||||||
// the one in JSONObject. Changes made here should be reflected there.
|
|
||||||
// This method should not make calls out of the XML object.
|
|
||||||
public static Object stringToValue(String string) {
|
|
||||||
if ("".equals(string)) {
|
|
||||||
return string;
|
|
||||||
}
|
|
||||||
|
|
||||||
// check JSON key words true/false/null
|
|
||||||
if ("true".equalsIgnoreCase(string)) {
|
|
||||||
return Boolean.TRUE;
|
|
||||||
}
|
|
||||||
if ("false".equalsIgnoreCase(string)) {
|
|
||||||
return Boolean.FALSE;
|
|
||||||
}
|
|
||||||
if ("null".equalsIgnoreCase(string)) {
|
|
||||||
return JSONObject.NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* If it might be a number, try converting it. If a number cannot be
|
|
||||||
* produced, then the value will just be a string.
|
|
||||||
*/
|
|
||||||
|
|
||||||
char initial = string.charAt(0);
|
|
||||||
if ((initial >= '0' && initial <= '9') || initial == '-') {
|
|
||||||
try {
|
|
||||||
return stringToNumber(string);
|
|
||||||
} catch (Exception ignore) {
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return string;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -567,6 +566,58 @@ public class XML {
|
|||||||
|| val.indexOf('E') > -1 || "-0".equals(val);
|
|| val.indexOf('E') > -1 || "-0".equals(val);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method tries to convert the given string value to the target object
|
||||||
|
* @param string String to convert
|
||||||
|
* @param typeConverter value converter to convert string to integer, boolean e.t.c
|
||||||
|
* @return JSON value of this string or the string
|
||||||
|
*/
|
||||||
|
public static Object stringToValue(String string, XMLXsiTypeConverter<?> typeConverter) {
|
||||||
|
if(typeConverter != null) {
|
||||||
|
return typeConverter.convert(string);
|
||||||
|
}
|
||||||
|
return stringToValue(string);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method is the same as {@link JSONObject#stringToValue(String)}.
|
||||||
|
*
|
||||||
|
* @param string String to convert
|
||||||
|
* @return JSON value of this string or the string
|
||||||
|
*/
|
||||||
|
// To maintain compatibility with the Android API, this method is a direct copy of
|
||||||
|
// the one in JSONObject. Changes made here should be reflected there.
|
||||||
|
// This method should not make calls out of the XML object.
|
||||||
|
public static Object stringToValue(String string) {
|
||||||
|
if ("".equals(string)) {
|
||||||
|
return string;
|
||||||
|
}
|
||||||
|
|
||||||
|
// check JSON key words true/false/null
|
||||||
|
if ("true".equalsIgnoreCase(string)) {
|
||||||
|
return Boolean.TRUE;
|
||||||
|
}
|
||||||
|
if ("false".equalsIgnoreCase(string)) {
|
||||||
|
return Boolean.FALSE;
|
||||||
|
}
|
||||||
|
if ("null".equalsIgnoreCase(string)) {
|
||||||
|
return JSONObject.NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If it might be a number, try converting it. If a number cannot be
|
||||||
|
* produced, then the value will just be a string.
|
||||||
|
*/
|
||||||
|
|
||||||
|
char initial = string.charAt(0);
|
||||||
|
if ((initial >= '0' && initial <= '9') || initial == '-') {
|
||||||
|
try {
|
||||||
|
return stringToNumber(string);
|
||||||
|
} catch (Exception ignore) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
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
|
||||||
@@ -659,7 +710,7 @@ public class XML {
|
|||||||
*/
|
*/
|
||||||
public static JSONObject toJSONObject(Reader reader, XMLParserConfiguration config) throws JSONException {
|
public static JSONObject toJSONObject(Reader reader, XMLParserConfiguration config) throws JSONException {
|
||||||
JSONObject jo = new JSONObject();
|
JSONObject jo = new JSONObject();
|
||||||
XMLTokener x = new XMLTokener(reader);
|
XMLTokener x = new XMLTokener(reader, config);
|
||||||
while (x.more()) {
|
while (x.more()) {
|
||||||
x.skipPast("<");
|
x.skipPast("<");
|
||||||
if(x.more()) {
|
if(x.more()) {
|
||||||
@@ -850,13 +901,26 @@ public class XML {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if ("".equals(value)) {
|
} else if ("".equals(value)) {
|
||||||
|
if (config.isCloseEmptyTag()){
|
||||||
|
sb.append(indent(indent));
|
||||||
|
sb.append('<');
|
||||||
|
sb.append(key);
|
||||||
|
sb.append(">");
|
||||||
|
sb.append("</");
|
||||||
|
sb.append(key);
|
||||||
|
sb.append(">");
|
||||||
|
if (indentFactor > 0) {
|
||||||
|
sb.append("\n");
|
||||||
|
}
|
||||||
|
}else {
|
||||||
sb.append(indent(indent));
|
sb.append(indent(indent));
|
||||||
sb.append('<');
|
sb.append('<');
|
||||||
sb.append(key);
|
sb.append(key);
|
||||||
sb.append("/>");
|
sb.append("/>");
|
||||||
if(indentFactor > 0){
|
if (indentFactor > 0) {
|
||||||
sb.append("\n");
|
sb.append("\n");
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Emit a new tag <k>
|
// Emit a new tag <k>
|
||||||
|
|
||||||
@@ -899,14 +963,14 @@ public class XML {
|
|||||||
|
|
||||||
|
|
||||||
string = (object == null) ? "null" : escape(object.toString());
|
string = (object == null) ? "null" : escape(object.toString());
|
||||||
|
String indentationSuffix = (indentFactor > 0) ? "\n" : "";
|
||||||
if(tagName == null){
|
if(tagName == null){
|
||||||
return indent(indent) + "\"" + string + "\"" + ((indentFactor > 0) ? "\n" : "");
|
return indent(indent) + "\"" + string + "\"" + indentationSuffix;
|
||||||
} else if(string.length() == 0){
|
} else if(string.length() == 0){
|
||||||
return indent(indent) + "<" + tagName + "/>" + ((indentFactor > 0) ? "\n" : "");
|
return indent(indent) + "<" + tagName + "/>" + indentationSuffix;
|
||||||
} else {
|
} else {
|
||||||
return indent(indent) + "<" + tagName
|
return indent(indent) + "<" + tagName
|
||||||
+ ">" + string + "</" + tagName + ">" + ((indentFactor > 0) ? "\n" : "");
|
+ ">" + string + "</" + tagName + ">" + indentationSuffix;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -43,6 +43,13 @@ public class XMLParserConfiguration extends ParserConfiguration {
|
|||||||
*/
|
*/
|
||||||
private boolean convertNilAttributeToNull;
|
private boolean convertNilAttributeToNull;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When creating an XML from JSON Object, an empty tag by default will self-close.
|
||||||
|
* If it has to be closed explicitly, with empty content between start and end tag,
|
||||||
|
* this flag is to be turned on.
|
||||||
|
*/
|
||||||
|
private boolean closeEmptyTag;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This will allow type conversion for values in XML if xsi:type attribute is defined
|
* This will allow type conversion for values in XML if xsi:type attribute is defined
|
||||||
*/
|
*/
|
||||||
@@ -54,9 +61,18 @@ public class XMLParserConfiguration extends ParserConfiguration {
|
|||||||
*/
|
*/
|
||||||
private Set<String> forceList;
|
private Set<String> forceList;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Flag to indicate whether white space should be trimmed when parsing XML.
|
||||||
|
* The default behaviour is to trim white space. When this is set to false, inputting XML
|
||||||
|
* with tags that are the same as the value of cDataTagName is unsupported. It is recommended to set cDataTagName
|
||||||
|
* to a distinct value in this case.
|
||||||
|
*/
|
||||||
|
private boolean shouldTrimWhiteSpace;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Default parser configuration. Does not keep strings (tries to implicitly convert
|
* Default parser configuration. Does not keep strings (tries to implicitly convert
|
||||||
* values), and the CDATA Tag Name is "content".
|
* values), and the CDATA Tag Name is "content". Trims whitespace.
|
||||||
*/
|
*/
|
||||||
public XMLParserConfiguration () {
|
public XMLParserConfiguration () {
|
||||||
super();
|
super();
|
||||||
@@ -64,6 +80,7 @@ public class XMLParserConfiguration extends ParserConfiguration {
|
|||||||
this.convertNilAttributeToNull = false;
|
this.convertNilAttributeToNull = false;
|
||||||
this.xsiTypeMap = Collections.emptyMap();
|
this.xsiTypeMap = Collections.emptyMap();
|
||||||
this.forceList = Collections.emptySet();
|
this.forceList = Collections.emptySet();
|
||||||
|
this.shouldTrimWhiteSpace = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -142,15 +159,17 @@ public class XMLParserConfiguration extends ParserConfiguration {
|
|||||||
* xsi:type="integer" as integer, xsi:type="string" as string
|
* xsi:type="integer" as integer, xsi:type="string" as string
|
||||||
* @param forceList <code>new HashSet<String>()</code> to parse the provided tags' values as arrays
|
* @param forceList <code>new HashSet<String>()</code> to parse the provided tags' values as arrays
|
||||||
* @param maxNestingDepth <code>int</code> to limit the nesting depth
|
* @param maxNestingDepth <code>int</code> to limit the nesting depth
|
||||||
|
* @param closeEmptyTag <code>boolean</code> to turn on explicit end tag for tag with empty value
|
||||||
*/
|
*/
|
||||||
private XMLParserConfiguration (final boolean keepStrings, final String cDataTagName,
|
private XMLParserConfiguration (final boolean keepStrings, final String cDataTagName,
|
||||||
final boolean convertNilAttributeToNull, final Map<String, XMLXsiTypeConverter<?>> xsiTypeMap, final Set<String> forceList,
|
final boolean convertNilAttributeToNull, final Map<String, XMLXsiTypeConverter<?>> xsiTypeMap, final Set<String> forceList,
|
||||||
final int maxNestingDepth) {
|
final int maxNestingDepth, final boolean closeEmptyTag) {
|
||||||
super(keepStrings, maxNestingDepth);
|
super(keepStrings, maxNestingDepth);
|
||||||
this.cDataTagName = cDataTagName;
|
this.cDataTagName = cDataTagName;
|
||||||
this.convertNilAttributeToNull = convertNilAttributeToNull;
|
this.convertNilAttributeToNull = convertNilAttributeToNull;
|
||||||
this.xsiTypeMap = Collections.unmodifiableMap(xsiTypeMap);
|
this.xsiTypeMap = Collections.unmodifiableMap(xsiTypeMap);
|
||||||
this.forceList = Collections.unmodifiableSet(forceList);
|
this.forceList = Collections.unmodifiableSet(forceList);
|
||||||
|
this.closeEmptyTag = closeEmptyTag;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -163,14 +182,17 @@ public class XMLParserConfiguration extends ParserConfiguration {
|
|||||||
// item, a new map instance should be created and if possible each value in the
|
// item, a new map instance should be created and if possible each value in the
|
||||||
// map should be cloned as well. If the values of the map are known to also
|
// map should be cloned as well. If the values of the map are known to also
|
||||||
// be immutable, then a shallow clone of the map is acceptable.
|
// be immutable, then a shallow clone of the map is acceptable.
|
||||||
return new XMLParserConfiguration(
|
final XMLParserConfiguration config = new XMLParserConfiguration(
|
||||||
this.keepStrings,
|
this.keepStrings,
|
||||||
this.cDataTagName,
|
this.cDataTagName,
|
||||||
this.convertNilAttributeToNull,
|
this.convertNilAttributeToNull,
|
||||||
this.xsiTypeMap,
|
this.xsiTypeMap,
|
||||||
this.forceList,
|
this.forceList,
|
||||||
this.maxNestingDepth
|
this.maxNestingDepth,
|
||||||
|
this.closeEmptyTag
|
||||||
);
|
);
|
||||||
|
config.shouldTrimWhiteSpace = this.shouldTrimWhiteSpace;
|
||||||
|
return config;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -182,6 +204,7 @@ public class XMLParserConfiguration extends ParserConfiguration {
|
|||||||
*
|
*
|
||||||
* @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")
|
||||||
@Override
|
@Override
|
||||||
public XMLParserConfiguration withKeepStrings(final boolean newVal) {
|
public XMLParserConfiguration withKeepStrings(final boolean newVal) {
|
||||||
return super.withKeepStrings(newVal);
|
return super.withKeepStrings(newVal);
|
||||||
@@ -299,8 +322,51 @@ public class XMLParserConfiguration extends ParserConfiguration {
|
|||||||
* @param maxNestingDepth the maximum nesting depth allowed to the XML parser
|
* @param maxNestingDepth the maximum nesting depth allowed to the XML parser
|
||||||
* @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")
|
||||||
@Override
|
@Override
|
||||||
public XMLParserConfiguration withMaxNestingDepth(int maxNestingDepth) {
|
public XMLParserConfiguration withMaxNestingDepth(int maxNestingDepth) {
|
||||||
return super.withMaxNestingDepth(maxNestingDepth);
|
return super.withMaxNestingDepth(maxNestingDepth);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* To enable explicit end tag with empty value.
|
||||||
|
* @param closeEmptyTag new value for the closeEmptyTag property
|
||||||
|
* @return same instance of configuration with empty tag config updated
|
||||||
|
*/
|
||||||
|
public XMLParserConfiguration withCloseEmptyTag(boolean closeEmptyTag){
|
||||||
|
XMLParserConfiguration clonedConfiguration = this.clone();
|
||||||
|
clonedConfiguration.closeEmptyTag = closeEmptyTag;
|
||||||
|
return clonedConfiguration;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets whether whitespace should be trimmed inside of tags. *NOTE* Do not use this if
|
||||||
|
* you expect your XML tags to have names that are the same as cDataTagName as this is unsupported.
|
||||||
|
* cDataTagName should be set to a distinct value in these cases.
|
||||||
|
* @param shouldTrimWhiteSpace boolean to set trimming on or off. Off is default.
|
||||||
|
* @return same instance of configuration with empty tag config updated
|
||||||
|
*/
|
||||||
|
public XMLParserConfiguration withShouldTrimWhitespace(boolean shouldTrimWhiteSpace){
|
||||||
|
XMLParserConfiguration clonedConfiguration = this.clone();
|
||||||
|
clonedConfiguration.shouldTrimWhiteSpace = shouldTrimWhiteSpace;
|
||||||
|
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() {
|
||||||
|
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() {
|
||||||
|
return this.shouldTrimWhiteSpace;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,6 +20,8 @@ public class XMLTokener extends JSONTokener {
|
|||||||
*/
|
*/
|
||||||
public static final java.util.HashMap<String, Character> entity;
|
public static final java.util.HashMap<String, Character> entity;
|
||||||
|
|
||||||
|
private XMLParserConfiguration configuration = XMLParserConfiguration.ORIGINAL;
|
||||||
|
|
||||||
static {
|
static {
|
||||||
entity = new java.util.HashMap<String, Character>(8);
|
entity = new java.util.HashMap<String, Character>(8);
|
||||||
entity.put("amp", XML.AMP);
|
entity.put("amp", XML.AMP);
|
||||||
@@ -45,6 +47,16 @@ public class XMLTokener extends JSONTokener {
|
|||||||
super(s);
|
super(s);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct an XMLTokener from a Reader and an XMLParserConfiguration.
|
||||||
|
* @param r A source reader.
|
||||||
|
* @param configuration the configuration that can be used to set certain flags
|
||||||
|
*/
|
||||||
|
public XMLTokener(Reader r, XMLParserConfiguration configuration) {
|
||||||
|
super(r);
|
||||||
|
this.configuration = configuration;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the text in the CDATA block.
|
* Get the text in the CDATA block.
|
||||||
* @return The string up to the <code>]]></code>.
|
* @return The string up to the <code>]]></code>.
|
||||||
@@ -83,7 +95,7 @@ public class XMLTokener extends JSONTokener {
|
|||||||
StringBuilder sb;
|
StringBuilder sb;
|
||||||
do {
|
do {
|
||||||
c = next();
|
c = next();
|
||||||
} while (Character.isWhitespace(c));
|
} while (Character.isWhitespace(c) && configuration.shouldTrimWhiteSpace());
|
||||||
if (c == 0) {
|
if (c == 0) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@@ -97,7 +109,9 @@ public class XMLTokener extends JSONTokener {
|
|||||||
}
|
}
|
||||||
if (c == '<') {
|
if (c == '<') {
|
||||||
back();
|
back();
|
||||||
|
if (configuration.shouldTrimWhiteSpace()) {
|
||||||
return sb.toString().trim();
|
return sb.toString().trim();
|
||||||
|
} else return sb.toString();
|
||||||
}
|
}
|
||||||
if (c == '&') {
|
if (c == '&') {
|
||||||
sb.append(nextEntity(c));
|
sb.append(nextEntity(c));
|
||||||
|
|||||||
@@ -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);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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,11 @@ 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: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:\"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:\"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:\"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}]";
|
||||||
"{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 +192,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 +251,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 +265,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);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -293,10 +297,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);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -28,10 +28,13 @@ import java.util.Map;
|
|||||||
import org.json.JSONArray;
|
import org.json.JSONArray;
|
||||||
import org.json.JSONException;
|
import org.json.JSONException;
|
||||||
import org.json.JSONObject;
|
import org.json.JSONObject;
|
||||||
|
import org.json.JSONParserConfiguration;
|
||||||
import org.json.JSONPointerException;
|
import org.json.JSONPointerException;
|
||||||
import org.json.JSONString;
|
import org.json.JSONString;
|
||||||
import org.json.JSONTokener;
|
import org.json.JSONTokener;
|
||||||
|
import org.json.ParserConfiguration;
|
||||||
import org.json.junit.data.MyJsonString;
|
import org.json.junit.data.MyJsonString;
|
||||||
|
import org.junit.Ignore;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
import com.jayway.jsonpath.Configuration;
|
import com.jayway.jsonpath.Configuration;
|
||||||
@@ -118,7 +121,7 @@ public class JSONArrayTest {
|
|||||||
* Expects a JSONException.
|
* Expects a JSONException.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void emptStr() {
|
public void emptyStr() {
|
||||||
String str = "";
|
String str = "";
|
||||||
try {
|
try {
|
||||||
assertNull("Should throw an exception", new JSONArray(str));
|
assertNull("Should throw an exception", new JSONArray(str));
|
||||||
@@ -368,16 +371,16 @@ public class JSONArrayTest {
|
|||||||
"hello".equals(jsonArray.getString(4)));
|
"hello".equals(jsonArray.getString(4)));
|
||||||
// doubles
|
// doubles
|
||||||
assertTrue("Array double",
|
assertTrue("Array double",
|
||||||
new Double(23.45e-4).equals(jsonArray.getDouble(5)));
|
Double.valueOf(23.45e-4).equals(jsonArray.getDouble(5)));
|
||||||
assertTrue("Array string double",
|
assertTrue("Array string double",
|
||||||
new Double(23.45).equals(jsonArray.getDouble(6)));
|
Double.valueOf(23.45).equals(jsonArray.getDouble(6)));
|
||||||
assertTrue("Array double can be float",
|
assertTrue("Array double can be float",
|
||||||
new Float(23.45e-4f).equals(jsonArray.getFloat(5)));
|
Float.valueOf(23.45e-4f).equals(jsonArray.getFloat(5)));
|
||||||
// ints
|
// ints
|
||||||
assertTrue("Array value int",
|
assertTrue("Array value int",
|
||||||
new Integer(42).equals(jsonArray.getInt(7)));
|
Integer.valueOf(42).equals(jsonArray.getInt(7)));
|
||||||
assertTrue("Array value string int",
|
assertTrue("Array value string int",
|
||||||
new Integer(43).equals(jsonArray.getInt(8)));
|
Integer.valueOf(43).equals(jsonArray.getInt(8)));
|
||||||
// nested objects
|
// nested objects
|
||||||
JSONArray nestedJsonArray = jsonArray.getJSONArray(9);
|
JSONArray nestedJsonArray = jsonArray.getJSONArray(9);
|
||||||
assertTrue("Array value JSONArray", nestedJsonArray != null);
|
assertTrue("Array value JSONArray", nestedJsonArray != null);
|
||||||
@@ -385,9 +388,9 @@ public class JSONArrayTest {
|
|||||||
assertTrue("Array value JSONObject", nestedJsonObject != null);
|
assertTrue("Array value JSONObject", nestedJsonObject != null);
|
||||||
// longs
|
// longs
|
||||||
assertTrue("Array value long",
|
assertTrue("Array value long",
|
||||||
new Long(0).equals(jsonArray.getLong(11)));
|
Long.valueOf(0).equals(jsonArray.getLong(11)));
|
||||||
assertTrue("Array value string long",
|
assertTrue("Array value string long",
|
||||||
new Long(-1).equals(jsonArray.getLong(12)));
|
Long.valueOf(-1).equals(jsonArray.getLong(12)));
|
||||||
|
|
||||||
assertTrue("Array value null", jsonArray.isNull(-1));
|
assertTrue("Array value null", jsonArray.isNull(-1));
|
||||||
Util.checkJSONArrayMaps(jsonArray);
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
@@ -460,6 +463,20 @@ public class JSONArrayTest {
|
|||||||
Util.checkJSONArrayMaps(jsonArray);
|
Util.checkJSONArrayMaps(jsonArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The JSON parser is permissive of unambiguous unquoted keys and values.
|
||||||
|
* Such JSON text should be allowed, even if it does not strictly conform
|
||||||
|
* to the spec. However, after being parsed, toString() should emit strictly
|
||||||
|
* conforming JSON text.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void unquotedText() {
|
||||||
|
String str = "[value1, something!, (parens), foo@bar.com, 23, 23+45]";
|
||||||
|
JSONArray jsonArray = new JSONArray(str);
|
||||||
|
List<Object> expected = Arrays.asList("value1", "something!", "(parens)", "foo@bar.com", 23, "23+45");
|
||||||
|
assertEquals(expected, jsonArray.toList());
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Exercise JSONArray.join() by converting a JSONArray into a
|
* Exercise JSONArray.join() by converting a JSONArray into a
|
||||||
* comma-separated string. Since this is very nearly a JSON document,
|
* comma-separated string. Since this is very nearly a JSON document,
|
||||||
@@ -537,43 +554,75 @@ public class JSONArrayTest {
|
|||||||
assertTrue("Array opt boolean implicit default",
|
assertTrue("Array opt boolean implicit default",
|
||||||
Boolean.FALSE == jsonArray.optBoolean(-1));
|
Boolean.FALSE == jsonArray.optBoolean(-1));
|
||||||
|
|
||||||
|
assertTrue("Array opt boolean object",
|
||||||
|
Boolean.TRUE.equals(jsonArray.optBooleanObject(0)));
|
||||||
|
assertTrue("Array opt boolean object default",
|
||||||
|
Boolean.FALSE.equals(jsonArray.optBooleanObject(-1, Boolean.FALSE)));
|
||||||
|
assertTrue("Array opt boolean object implicit default",
|
||||||
|
Boolean.FALSE.equals(jsonArray.optBooleanObject(-1)));
|
||||||
|
|
||||||
assertTrue("Array opt double",
|
assertTrue("Array opt double",
|
||||||
new Double(23.45e-4).equals(jsonArray.optDouble(5)));
|
Double.valueOf(23.45e-4).equals(jsonArray.optDouble(5)));
|
||||||
assertTrue("Array opt double default",
|
assertTrue("Array opt double default",
|
||||||
new Double(1).equals(jsonArray.optDouble(0, 1)));
|
Double.valueOf(1).equals(jsonArray.optDouble(0, 1)));
|
||||||
assertTrue("Array opt double default implicit",
|
assertTrue("Array opt double default implicit",
|
||||||
new Double(jsonArray.optDouble(99)).isNaN());
|
Double.valueOf(jsonArray.optDouble(99)).isNaN());
|
||||||
|
|
||||||
|
assertTrue("Array opt double object",
|
||||||
|
Double.valueOf(23.45e-4).equals(jsonArray.optDoubleObject(5)));
|
||||||
|
assertTrue("Array opt double object default",
|
||||||
|
Double.valueOf(1).equals(jsonArray.optDoubleObject(0, 1D)));
|
||||||
|
assertTrue("Array opt double object default implicit",
|
||||||
|
jsonArray.optDoubleObject(99).isNaN());
|
||||||
|
|
||||||
assertTrue("Array opt float",
|
assertTrue("Array opt float",
|
||||||
new Float(23.45e-4).equals(jsonArray.optFloat(5)));
|
Float.valueOf(Double.valueOf(23.45e-4).floatValue()).equals(jsonArray.optFloat(5)));
|
||||||
assertTrue("Array opt float default",
|
assertTrue("Array opt float default",
|
||||||
new Float(1).equals(jsonArray.optFloat(0, 1)));
|
Float.valueOf(1).equals(jsonArray.optFloat(0, 1)));
|
||||||
assertTrue("Array opt float default implicit",
|
assertTrue("Array opt float default implicit",
|
||||||
new Float(jsonArray.optFloat(99)).isNaN());
|
Float.valueOf(jsonArray.optFloat(99)).isNaN());
|
||||||
|
|
||||||
|
assertTrue("Array opt float object",
|
||||||
|
Float.valueOf(23.45e-4F).equals(jsonArray.optFloatObject(5)));
|
||||||
|
assertTrue("Array opt float object default",
|
||||||
|
Float.valueOf(1).equals(jsonArray.optFloatObject(0, 1F)));
|
||||||
|
assertTrue("Array opt float object default implicit",
|
||||||
|
jsonArray.optFloatObject(99).isNaN());
|
||||||
|
|
||||||
assertTrue("Array opt Number",
|
assertTrue("Array opt Number",
|
||||||
BigDecimal.valueOf(23.45e-4).equals(jsonArray.optNumber(5)));
|
BigDecimal.valueOf(23.45e-4).equals(jsonArray.optNumber(5)));
|
||||||
assertTrue("Array opt Number default",
|
assertTrue("Array opt Number default",
|
||||||
new Double(1).equals(jsonArray.optNumber(0, 1d)));
|
Double.valueOf(1).equals(jsonArray.optNumber(0, 1d)));
|
||||||
assertTrue("Array opt Number default implicit",
|
assertTrue("Array opt Number default implicit",
|
||||||
new Double(jsonArray.optNumber(99,Double.NaN).doubleValue()).isNaN());
|
Double.valueOf(jsonArray.optNumber(99,Double.NaN).doubleValue()).isNaN());
|
||||||
|
|
||||||
assertTrue("Array opt int",
|
assertTrue("Array opt int",
|
||||||
new Integer(42).equals(jsonArray.optInt(7)));
|
Integer.valueOf(42).equals(jsonArray.optInt(7)));
|
||||||
assertTrue("Array opt int default",
|
assertTrue("Array opt int default",
|
||||||
new Integer(-1).equals(jsonArray.optInt(0, -1)));
|
Integer.valueOf(-1).equals(jsonArray.optInt(0, -1)));
|
||||||
assertTrue("Array opt int default implicit",
|
assertTrue("Array opt int default implicit",
|
||||||
0 == jsonArray.optInt(0));
|
0 == jsonArray.optInt(0));
|
||||||
|
|
||||||
|
assertTrue("Array opt int object",
|
||||||
|
Integer.valueOf(42).equals(jsonArray.optIntegerObject(7)));
|
||||||
|
assertTrue("Array opt int object default",
|
||||||
|
Integer.valueOf(-1).equals(jsonArray.optIntegerObject(0, -1)));
|
||||||
|
assertTrue("Array opt int object default implicit",
|
||||||
|
Integer.valueOf(0).equals(jsonArray.optIntegerObject(0)));
|
||||||
|
|
||||||
JSONArray nestedJsonArray = jsonArray.optJSONArray(9);
|
JSONArray nestedJsonArray = jsonArray.optJSONArray(9);
|
||||||
assertTrue("Array opt JSONArray", nestedJsonArray != null);
|
assertTrue("Array opt JSONArray", nestedJsonArray != null);
|
||||||
assertTrue("Array opt JSONArray default",
|
assertTrue("Array opt JSONArray null",
|
||||||
null == jsonArray.optJSONArray(99));
|
null == jsonArray.optJSONArray(99));
|
||||||
|
assertTrue("Array opt JSONArray default",
|
||||||
|
"value".equals(jsonArray.optJSONArray(99, new JSONArray("[\"value\"]")).getString(0)));
|
||||||
|
|
||||||
JSONObject nestedJsonObject = jsonArray.optJSONObject(10);
|
JSONObject nestedJsonObject = jsonArray.optJSONObject(10);
|
||||||
assertTrue("Array opt JSONObject", nestedJsonObject != null);
|
assertTrue("Array opt JSONObject", nestedJsonObject != null);
|
||||||
assertTrue("Array opt JSONObject default",
|
assertTrue("Array opt JSONObject null",
|
||||||
null == jsonArray.optJSONObject(99));
|
null == jsonArray.optJSONObject(99));
|
||||||
|
assertTrue("Array opt JSONObject default",
|
||||||
|
"value".equals(jsonArray.optJSONObject(99, new JSONObject("{\"key\":\"value\"}")).getString("key")));
|
||||||
|
|
||||||
assertTrue("Array opt long",
|
assertTrue("Array opt long",
|
||||||
0 == jsonArray.optLong(11));
|
0 == jsonArray.optLong(11));
|
||||||
@@ -582,6 +631,13 @@ public class JSONArrayTest {
|
|||||||
assertTrue("Array opt long default implicit",
|
assertTrue("Array opt long default implicit",
|
||||||
0 == jsonArray.optLong(-1));
|
0 == jsonArray.optLong(-1));
|
||||||
|
|
||||||
|
assertTrue("Array opt long object",
|
||||||
|
Long.valueOf(0).equals(jsonArray.optLongObject(11)));
|
||||||
|
assertTrue("Array opt long object default",
|
||||||
|
Long.valueOf(-2).equals(jsonArray.optLongObject(-1, -2L)));
|
||||||
|
assertTrue("Array opt long object default implicit",
|
||||||
|
Long.valueOf(0).equals(jsonArray.optLongObject(-1)));
|
||||||
|
|
||||||
assertTrue("Array opt string",
|
assertTrue("Array opt string",
|
||||||
"hello".equals(jsonArray.optString(4)));
|
"hello".equals(jsonArray.optString(4)));
|
||||||
assertTrue("Array opt string default implicit",
|
assertTrue("Array opt string default implicit",
|
||||||
@@ -599,10 +655,15 @@ public class JSONArrayTest {
|
|||||||
public void optStringConversion(){
|
public void optStringConversion(){
|
||||||
JSONArray ja = new JSONArray("[\"123\",\"true\",\"false\"]");
|
JSONArray ja = new JSONArray("[\"123\",\"true\",\"false\"]");
|
||||||
assertTrue("unexpected optBoolean value",ja.optBoolean(1,false)==true);
|
assertTrue("unexpected optBoolean value",ja.optBoolean(1,false)==true);
|
||||||
|
assertTrue("unexpected optBooleanObject value",Boolean.valueOf(true).equals(ja.optBooleanObject(1,false)));
|
||||||
assertTrue("unexpected optBoolean value",ja.optBoolean(2,true)==false);
|
assertTrue("unexpected optBoolean value",ja.optBoolean(2,true)==false);
|
||||||
|
assertTrue("unexpected optBooleanObject value",Boolean.valueOf(false).equals(ja.optBooleanObject(2,true)));
|
||||||
assertTrue("unexpected optInt value",ja.optInt(0,0)==123);
|
assertTrue("unexpected optInt value",ja.optInt(0,0)==123);
|
||||||
|
assertTrue("unexpected optIntegerObject value",Integer.valueOf(123).equals(ja.optIntegerObject(0,0)));
|
||||||
assertTrue("unexpected optLong value",ja.optLong(0,0)==123);
|
assertTrue("unexpected optLong value",ja.optLong(0,0)==123);
|
||||||
|
assertTrue("unexpected optLongObject value",Long.valueOf(123).equals(ja.optLongObject(0,0L)));
|
||||||
assertTrue("unexpected optDouble value",ja.optDouble(0,0.0)==123.0);
|
assertTrue("unexpected optDouble value",ja.optDouble(0,0.0)==123.0);
|
||||||
|
assertTrue("unexpected optDoubleObject value",Double.valueOf(123.0).equals(ja.optDoubleObject(0,0.0)));
|
||||||
assertTrue("unexpected optBigInteger value",ja.optBigInteger(0,BigInteger.ZERO).compareTo(new BigInteger("123"))==0);
|
assertTrue("unexpected optBigInteger value",ja.optBigInteger(0,BigInteger.ZERO).compareTo(new BigInteger("123"))==0);
|
||||||
assertTrue("unexpected optBigDecimal value",ja.optBigDecimal(0,BigDecimal.ZERO).compareTo(new BigDecimal("123"))==0);
|
assertTrue("unexpected optBigDecimal value",ja.optBigDecimal(0,BigDecimal.ZERO).compareTo(new BigDecimal("123"))==0);
|
||||||
Util.checkJSONArrayMaps(ja);
|
Util.checkJSONArrayMaps(ja);
|
||||||
@@ -971,12 +1032,12 @@ public class JSONArrayTest {
|
|||||||
assertTrue("Array double [23.45e-4]",
|
assertTrue("Array double [23.45e-4]",
|
||||||
new BigDecimal("0.002345").equals(it.next()));
|
new BigDecimal("0.002345").equals(it.next()));
|
||||||
assertTrue("Array string double",
|
assertTrue("Array string double",
|
||||||
new Double(23.45).equals(Double.parseDouble((String)it.next())));
|
Double.valueOf(23.45).equals(Double.parseDouble((String)it.next())));
|
||||||
|
|
||||||
assertTrue("Array value int",
|
assertTrue("Array value int",
|
||||||
new Integer(42).equals(it.next()));
|
Integer.valueOf(42).equals(it.next()));
|
||||||
assertTrue("Array value string int",
|
assertTrue("Array value string int",
|
||||||
new Integer(43).equals(Integer.parseInt((String)it.next())));
|
Integer.valueOf(43).equals(Integer.parseInt((String)it.next())));
|
||||||
|
|
||||||
JSONArray nestedJsonArray = (JSONArray)it.next();
|
JSONArray nestedJsonArray = (JSONArray)it.next();
|
||||||
assertTrue("Array value JSONArray", nestedJsonArray != null);
|
assertTrue("Array value JSONArray", nestedJsonArray != null);
|
||||||
@@ -985,9 +1046,9 @@ public class JSONArrayTest {
|
|||||||
assertTrue("Array value JSONObject", nestedJsonObject != null);
|
assertTrue("Array value JSONObject", nestedJsonObject != null);
|
||||||
|
|
||||||
assertTrue("Array value long",
|
assertTrue("Array value long",
|
||||||
new Long(0).equals(((Number) it.next()).longValue()));
|
Long.valueOf(0).equals(((Number) it.next()).longValue()));
|
||||||
assertTrue("Array value string long",
|
assertTrue("Array value string long",
|
||||||
new Long(-1).equals(Long.parseLong((String) it.next())));
|
Long.valueOf(-1).equals(Long.parseLong((String) it.next())));
|
||||||
assertTrue("should be at end of array", !it.hasNext());
|
assertTrue("should be at end of array", !it.hasNext());
|
||||||
Util.checkJSONArraysMaps(new ArrayList<JSONArray>(Arrays.asList(
|
Util.checkJSONArraysMaps(new ArrayList<JSONArray>(Arrays.asList(
|
||||||
jsonArray, nestedJsonArray
|
jsonArray, nestedJsonArray
|
||||||
@@ -1326,14 +1387,15 @@ public class JSONArrayTest {
|
|||||||
/**
|
/**
|
||||||
* Tests for stack overflow. See https://github.com/stleary/JSON-java/issues/654
|
* Tests for stack overflow. See https://github.com/stleary/JSON-java/issues/654
|
||||||
*/
|
*/
|
||||||
|
@Ignore("This test relies on system constraints and may not always pass. See: https://github.com/stleary/JSON-java/issues/821")
|
||||||
@Test(expected = JSONException.class)
|
@Test(expected = JSONException.class)
|
||||||
public void issue654StackOverflowInputWellFormed() {
|
public void issue654StackOverflowInputWellFormed() {
|
||||||
//String input = new String(java.util.Base64.getDecoder().decode(base64Bytes));
|
//String input = new String(java.util.Base64.getDecoder().decode(base64Bytes));
|
||||||
final InputStream resourceAsStream = JSONObjectTest.class.getClassLoader().getResourceAsStream("Issue654WellFormedArray.json");
|
final InputStream resourceAsStream = JSONArrayTest.class.getClassLoader().getResourceAsStream("Issue654WellFormedArray.json");
|
||||||
JSONTokener tokener = new JSONTokener(resourceAsStream);
|
JSONTokener tokener = new JSONTokener(resourceAsStream);
|
||||||
JSONArray json_input = new JSONArray(tokener);
|
JSONArray json_input = new JSONArray(tokener);
|
||||||
assertNotNull(json_input);
|
assertNotNull(json_input);
|
||||||
fail("Excepected Exception.");
|
fail("Excepected Exception due to stack overflow.");
|
||||||
Util.checkJSONArrayMaps(json_input);
|
Util.checkJSONArrayMaps(json_input);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1357,4 +1419,92 @@ public class JSONArrayTest {
|
|||||||
.put(2);
|
.put(2);
|
||||||
assertFalse(ja1.similar(ja3));
|
assertFalse(ja1.similar(ja3));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test(expected = JSONException.class)
|
||||||
|
public void testRecursiveDepth() {
|
||||||
|
HashMap<String, Object> map = new HashMap<>();
|
||||||
|
map.put("t", map);
|
||||||
|
new JSONArray().put(map);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = JSONException.class)
|
||||||
|
public void testRecursiveDepthAtPosition() {
|
||||||
|
HashMap<String, Object> map = new HashMap<>();
|
||||||
|
map.put("t", map);
|
||||||
|
new JSONArray().put(0, map);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = JSONException.class)
|
||||||
|
public void testRecursiveDepthArray() {
|
||||||
|
ArrayList<Object> array = new ArrayList<>();
|
||||||
|
array.add(array);
|
||||||
|
new JSONArray(array);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testRecursiveDepthAtPositionDefaultObject() {
|
||||||
|
HashMap<String, Object> map = JSONObjectTest.buildNestedMap(ParserConfiguration.DEFAULT_MAXIMUM_NESTING_DEPTH);
|
||||||
|
new JSONArray().put(0, map);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testRecursiveDepthAtPosition1000Object() {
|
||||||
|
HashMap<String, Object> map = JSONObjectTest.buildNestedMap(1000);
|
||||||
|
new JSONArray().put(0, map, new JSONParserConfiguration().withMaxNestingDepth(1000));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = JSONException.class)
|
||||||
|
public void testRecursiveDepthAtPosition1001Object() {
|
||||||
|
HashMap<String, Object> map = JSONObjectTest.buildNestedMap(1001);
|
||||||
|
new JSONArray().put(0, map);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = JSONException.class)
|
||||||
|
public void testRecursiveDepthArrayLimitedMaps() {
|
||||||
|
ArrayList<Object> array = new ArrayList<>();
|
||||||
|
array.add(array);
|
||||||
|
new JSONArray(array);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testRecursiveDepthArrayForDefaultLevels() {
|
||||||
|
ArrayList<Object> array = buildNestedArray(ParserConfiguration.DEFAULT_MAXIMUM_NESTING_DEPTH);
|
||||||
|
new JSONArray(array, new JSONParserConfiguration());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testRecursiveDepthArrayFor1000Levels() {
|
||||||
|
try {
|
||||||
|
ArrayList<Object> array = buildNestedArray(1000);
|
||||||
|
JSONParserConfiguration parserConfiguration = new JSONParserConfiguration().withMaxNestingDepth(1000);
|
||||||
|
new JSONArray(array, parserConfiguration);
|
||||||
|
} catch (StackOverflowError e) {
|
||||||
|
String javaVersion = System.getProperty("java.version");
|
||||||
|
if (javaVersion.startsWith("11.")) {
|
||||||
|
System.out.println(
|
||||||
|
"testRecursiveDepthArrayFor1000Levels() allowing intermittent stackoverflow, Java Version: "
|
||||||
|
+ javaVersion);
|
||||||
|
} else {
|
||||||
|
String errorStr = "testRecursiveDepthArrayFor1000Levels() unexpected stackoverflow, Java Version: "
|
||||||
|
+ javaVersion;
|
||||||
|
System.out.println(errorStr);
|
||||||
|
throw new RuntimeException(errorStr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = JSONException.class)
|
||||||
|
public void testRecursiveDepthArrayFor1001Levels() {
|
||||||
|
ArrayList<Object> array = buildNestedArray(1001);
|
||||||
|
new JSONArray(array);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ArrayList<Object> buildNestedArray(int maxDepth) {
|
||||||
|
if (maxDepth <= 0) {
|
||||||
|
return new ArrayList<>();
|
||||||
|
}
|
||||||
|
ArrayList<Object> nestedArray = new ArrayList<>();
|
||||||
|
nestedArray.add(buildNestedArray(maxDepth - 1));
|
||||||
|
return nestedArray;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -762,8 +762,8 @@ public class JSONMLTest {
|
|||||||
final String xml = JSONML.toString(originalObject);
|
final String xml = JSONML.toString(originalObject);
|
||||||
final JSONObject revertedObject = JSONML.toJSONObject(xml, false);
|
final JSONObject revertedObject = JSONML.toJSONObject(xml, false);
|
||||||
final String newJson = revertedObject.toString();
|
final String newJson = revertedObject.toString();
|
||||||
assertTrue("JSON Objects are not similar",originalObject.similar(revertedObject));
|
assertTrue("JSON Objects are not similar", originalObject.similar(revertedObject));
|
||||||
assertEquals("original JSON does not equal the new JSON",originalJson, newJson);
|
assertTrue("JSON Strings are not similar", new JSONObject(originalJson).similar(new JSONObject(newJson)));
|
||||||
}
|
}
|
||||||
|
|
||||||
// these tests do not pass for the following reasons:
|
// these tests do not pass for the following reasons:
|
||||||
|
|||||||
@@ -109,18 +109,38 @@ public class JSONObjectNumberTest {
|
|||||||
assertEquals(value.floatValue(), object.optFloat("value"), 0.0f);
|
assertEquals(value.floatValue(), object.optFloat("value"), 0.0f);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testOptFloatObject() {
|
||||||
|
assertEquals((Float) value.floatValue(), object.optFloatObject("value"), 0.0f);
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testOptDouble() {
|
public void testOptDouble() {
|
||||||
assertEquals(value.doubleValue(), object.optDouble("value"), 0.0d);
|
assertEquals(value.doubleValue(), object.optDouble("value"), 0.0d);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testOptDoubleObject() {
|
||||||
|
assertEquals((Double) value.doubleValue(), object.optDoubleObject("value"), 0.0d);
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testOptInt() {
|
public void testOptInt() {
|
||||||
assertEquals(value.intValue(), object.optInt("value"));
|
assertEquals(value.intValue(), object.optInt("value"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testOptIntegerObject() {
|
||||||
|
assertEquals((Integer) value.intValue(), object.optIntegerObject("value"));
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testOptLong() {
|
public void testOptLong() {
|
||||||
assertEquals(value.longValue(), object.optLong("value"));
|
assertEquals(value.longValue(), object.optLong("value"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testOptLongObject() {
|
||||||
|
assertEquals((Long) value.longValue(), object.optLongObject("value"));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import static org.junit.Assert.assertFalse;
|
|||||||
import static org.junit.Assert.assertNotEquals;
|
import static org.junit.Assert.assertNotEquals;
|
||||||
import static org.junit.Assert.assertNotNull;
|
import static org.junit.Assert.assertNotNull;
|
||||||
import static org.junit.Assert.assertNull;
|
import static org.junit.Assert.assertNull;
|
||||||
|
import static org.junit.Assert.assertThrows;
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.Assert.assertTrue;
|
||||||
import static org.junit.Assert.fail;
|
import static org.junit.Assert.fail;
|
||||||
import static org.mockito.Mockito.mock;
|
import static org.mockito.Mockito.mock;
|
||||||
@@ -30,8 +31,10 @@ import org.json.JSONArray;
|
|||||||
import org.json.JSONException;
|
import org.json.JSONException;
|
||||||
import org.json.JSONObject;
|
import org.json.JSONObject;
|
||||||
import org.json.JSONPointerException;
|
import org.json.JSONPointerException;
|
||||||
|
import org.json.JSONParserConfiguration;
|
||||||
import org.json.JSONString;
|
import org.json.JSONString;
|
||||||
import org.json.JSONTokener;
|
import org.json.JSONTokener;
|
||||||
|
import org.json.ParserConfiguration;
|
||||||
import org.json.XML;
|
import org.json.XML;
|
||||||
import org.json.junit.data.BrokenToString;
|
import org.json.junit.data.BrokenToString;
|
||||||
import org.json.junit.data.ExceptionalBean;
|
import org.json.junit.data.ExceptionalBean;
|
||||||
@@ -53,8 +56,8 @@ 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.Ignore;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.json.junit.Util;
|
|
||||||
|
|
||||||
import com.jayway.jsonpath.Configuration;
|
import com.jayway.jsonpath.Configuration;
|
||||||
import com.jayway.jsonpath.JsonPath;
|
import com.jayway.jsonpath.JsonPath;
|
||||||
@@ -206,13 +209,17 @@ public class JSONObjectTest {
|
|||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void unquotedText() {
|
public void unquotedText() {
|
||||||
String str = "{key1:value1, key2:42}";
|
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();
|
||||||
assertTrue("expected key1", textStr.contains("\"key1\""));
|
assertTrue("expected key1", textStr.contains("\"key1\""));
|
||||||
assertTrue("expected value1", textStr.contains("\"value1\""));
|
assertTrue("expected value1", textStr.contains("\"value1\""));
|
||||||
assertTrue("expected key2", textStr.contains("\"key2\""));
|
assertTrue("expected key2", textStr.contains("\"key2\""));
|
||||||
assertTrue("expected 42", textStr.contains("42"));
|
assertTrue("expected 42", textStr.contains("42"));
|
||||||
|
assertTrue("expected 1.2", textStr.contains("\"1.2\""));
|
||||||
|
assertTrue("expected 3.4", textStr.contains("3.4"));
|
||||||
|
assertTrue("expected -7E+5", textStr.contains("\"-7E+5\""));
|
||||||
|
assertTrue("expected something!", textStr.contains("\"something!\""));
|
||||||
Util.checkJSONObjectMaps(jsonObject);
|
Util.checkJSONObjectMaps(jsonObject);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -231,6 +238,11 @@ public class JSONObjectTest {
|
|||||||
assert 26315000000253009L == actualLong : "Incorrect key value. Got "
|
assert 26315000000253009L == actualLong : "Incorrect key value. Got "
|
||||||
+ actualLong + " expected " + str;
|
+ actualLong + " expected " + str;
|
||||||
|
|
||||||
|
final Long actualLongObject = json.optLongObject("key");
|
||||||
|
assert actualLongObject != 0L : "Unable to extract Long value for string " + str;
|
||||||
|
assert Long.valueOf(26315000000253009L).equals(actualLongObject) : "Incorrect key value. Got "
|
||||||
|
+ actualLongObject + " expected " + str;
|
||||||
|
|
||||||
final String actualString = json.optString("key");
|
final String actualString = json.optString("key");
|
||||||
assert str.equals(actualString) : "Incorrect key value. Got "
|
assert str.equals(actualString) : "Incorrect key value. Got "
|
||||||
+ actualString + " expected " + str;
|
+ actualString + " expected " + str;
|
||||||
@@ -299,12 +311,12 @@ public class JSONObjectTest {
|
|||||||
@Test
|
@Test
|
||||||
public void jsonObjectByMap() {
|
public void jsonObjectByMap() {
|
||||||
Map<String, Object> map = new HashMap<String, Object>();
|
Map<String, Object> map = new HashMap<String, Object>();
|
||||||
map.put("trueKey", new Boolean(true));
|
map.put("trueKey", Boolean.valueOf(true));
|
||||||
map.put("falseKey", new Boolean(false));
|
map.put("falseKey", Boolean.valueOf(false));
|
||||||
map.put("stringKey", "hello world!");
|
map.put("stringKey", "hello world!");
|
||||||
map.put("escapeStringKey", "h\be\tllo w\u1234orld!");
|
map.put("escapeStringKey", "h\be\tllo w\u1234orld!");
|
||||||
map.put("intKey", new Long(42));
|
map.put("intKey", Long.valueOf(42));
|
||||||
map.put("doubleKey", new Double(-23.45e67));
|
map.put("doubleKey", Double.valueOf(-23.45e67));
|
||||||
JSONObject jsonObject = new JSONObject(map);
|
JSONObject jsonObject = new JSONObject(map);
|
||||||
|
|
||||||
// validate JSON
|
// validate JSON
|
||||||
@@ -565,13 +577,13 @@ public class JSONObjectTest {
|
|||||||
@Test
|
@Test
|
||||||
public void jsonObjectByMapWithNullValue() {
|
public void jsonObjectByMapWithNullValue() {
|
||||||
Map<String, Object> map = new HashMap<String, Object>();
|
Map<String, Object> map = new HashMap<String, Object>();
|
||||||
map.put("trueKey", new Boolean(true));
|
map.put("trueKey", Boolean.valueOf(true));
|
||||||
map.put("falseKey", new Boolean(false));
|
map.put("falseKey", Boolean.valueOf(false));
|
||||||
map.put("stringKey", "hello world!");
|
map.put("stringKey", "hello world!");
|
||||||
map.put("nullKey", null);
|
map.put("nullKey", null);
|
||||||
map.put("escapeStringKey", "h\be\tllo w\u1234orld!");
|
map.put("escapeStringKey", "h\be\tllo w\u1234orld!");
|
||||||
map.put("intKey", new Long(42));
|
map.put("intKey", Long.valueOf(42));
|
||||||
map.put("doubleKey", new Double(-23.45e67));
|
map.put("doubleKey", Double.valueOf(-23.45e67));
|
||||||
JSONObject jsonObject = new JSONObject(map);
|
JSONObject jsonObject = new JSONObject(map);
|
||||||
|
|
||||||
// validate JSON
|
// validate JSON
|
||||||
@@ -621,9 +633,9 @@ public class JSONObjectTest {
|
|||||||
assertTrue("expected 42", Integer.valueOf("42").equals(jsonObject.query("/intKey")));
|
assertTrue("expected 42", Integer.valueOf("42").equals(jsonObject.query("/intKey")));
|
||||||
assertTrue("expected -23.45e7", Double.valueOf("-23.45e7").equals(jsonObject.query("/doubleKey")));
|
assertTrue("expected -23.45e7", Double.valueOf("-23.45e7").equals(jsonObject.query("/doubleKey")));
|
||||||
// sorry, mockito artifact
|
// sorry, mockito artifact
|
||||||
assertTrue("expected 2 callbacks items", ((List<?>)(JsonPath.read(doc, "$.callbacks"))).size() == 2);
|
assertTrue("expected 2 mockitoInterceptor items", ((Map<?,?>)(JsonPath.read(doc, "$.mockitoInterceptor"))).size() == 2);
|
||||||
assertTrue("expected 0 handler items", ((Map<?,?>)(JsonPath.read(doc, "$.callbacks[0].handler"))).size() == 0);
|
assertTrue("expected 0 mockitoInterceptor.serializationSupport items",
|
||||||
assertTrue("expected 0 callbacks[1] items", ((Map<?,?>)(JsonPath.read(doc, "$.callbacks[1]"))).size() == 0);
|
((Map<?,?>)(JsonPath.read(doc, "$.mockitoInterceptor.serializationSupport"))).size() == 0);
|
||||||
Util.checkJSONObjectMaps(jsonObject);
|
Util.checkJSONObjectMaps(jsonObject);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -866,9 +878,11 @@ public class JSONObjectTest {
|
|||||||
JSONObject jsonObject = new JSONObject(str);
|
JSONObject jsonObject = new JSONObject(str);
|
||||||
assertTrue("trueKey should be true", jsonObject.getBoolean("trueKey"));
|
assertTrue("trueKey should be true", jsonObject.getBoolean("trueKey"));
|
||||||
assertTrue("opt trueKey should be true", jsonObject.optBoolean("trueKey"));
|
assertTrue("opt trueKey should be true", jsonObject.optBoolean("trueKey"));
|
||||||
|
assertTrue("opt trueKey should be true", jsonObject.optBooleanObject("trueKey"));
|
||||||
assertTrue("falseKey should be false", !jsonObject.getBoolean("falseKey"));
|
assertTrue("falseKey should be false", !jsonObject.getBoolean("falseKey"));
|
||||||
assertTrue("trueStrKey should be true", jsonObject.getBoolean("trueStrKey"));
|
assertTrue("trueStrKey should be true", jsonObject.getBoolean("trueStrKey"));
|
||||||
assertTrue("trueStrKey should be true", jsonObject.optBoolean("trueStrKey"));
|
assertTrue("trueStrKey should be true", jsonObject.optBoolean("trueStrKey"));
|
||||||
|
assertTrue("trueStrKey should be true", jsonObject.optBooleanObject("trueStrKey"));
|
||||||
assertTrue("falseStrKey should be false", !jsonObject.getBoolean("falseStrKey"));
|
assertTrue("falseStrKey should be false", !jsonObject.getBoolean("falseStrKey"));
|
||||||
assertTrue("stringKey should be string",
|
assertTrue("stringKey should be string",
|
||||||
jsonObject.getString("stringKey").equals("hello world!"));
|
jsonObject.getString("stringKey").equals("hello world!"));
|
||||||
@@ -884,6 +898,10 @@ public class JSONObjectTest {
|
|||||||
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", Double.NaN) == 1);
|
jsonObject.optDouble("doubleStrKey", Double.NaN) == 1);
|
||||||
|
assertTrue("opt doubleKey should be Double",
|
||||||
|
Double.valueOf(-23.45e7).equals(jsonObject.optDoubleObject("doubleKey")));
|
||||||
|
assertTrue("opt doubleKey with Default should be Double",
|
||||||
|
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",
|
||||||
@@ -896,6 +914,10 @@ public class JSONObjectTest {
|
|||||||
Double.compare(jsonObject.optDouble("negZeroKey"), -0.0d) == 0);
|
Double.compare(jsonObject.optDouble("negZeroKey"), -0.0d) == 0);
|
||||||
assertTrue("opt negZeroStrKey with Default should be double",
|
assertTrue("opt negZeroStrKey with Default should be double",
|
||||||
Double.compare(jsonObject.optDouble("negZeroStrKey"), -0.0d) == 0);
|
Double.compare(jsonObject.optDouble("negZeroStrKey"), -0.0d) == 0);
|
||||||
|
assertTrue("opt negZeroKey should be Double",
|
||||||
|
Double.valueOf(-0.0d).equals(jsonObject.optDoubleObject("negZeroKey")));
|
||||||
|
assertTrue("opt negZeroStrKey with Default should be Double",
|
||||||
|
Double.valueOf(-0.0d).equals(jsonObject.optDoubleObject("negZeroStrKey")));
|
||||||
assertTrue("optNumber negZeroKey should be -0.0",
|
assertTrue("optNumber negZeroKey should be -0.0",
|
||||||
Double.compare(jsonObject.optNumber("negZeroKey").doubleValue(), -0.0d) == 0);
|
Double.compare(jsonObject.optNumber("negZeroKey").doubleValue(), -0.0d) == 0);
|
||||||
assertTrue("optNumber negZeroStrKey should be -0.0",
|
assertTrue("optNumber negZeroStrKey should be -0.0",
|
||||||
@@ -904,10 +926,18 @@ public class JSONObjectTest {
|
|||||||
jsonObject.optFloat("doubleKey") == -23.45e7f);
|
jsonObject.optFloat("doubleKey") == -23.45e7f);
|
||||||
assertTrue("optFloat doubleKey with Default should be float",
|
assertTrue("optFloat doubleKey with Default should be float",
|
||||||
jsonObject.optFloat("doubleStrKey", Float.NaN) == 1f);
|
jsonObject.optFloat("doubleStrKey", Float.NaN) == 1f);
|
||||||
|
assertTrue("optFloat doubleKey should be Float",
|
||||||
|
Float.valueOf(-23.45e7f).equals(jsonObject.optFloatObject("doubleKey")));
|
||||||
|
assertTrue("optFloat doubleKey with Default should be Float",
|
||||||
|
Float.valueOf(1f).equals(jsonObject.optFloatObject("doubleStrKey", Float.NaN)));
|
||||||
assertTrue("intKey should be int",
|
assertTrue("intKey should be int",
|
||||||
jsonObject.optInt("intKey") == 42);
|
jsonObject.optInt("intKey") == 42);
|
||||||
assertTrue("opt intKey should be int",
|
assertTrue("opt intKey should be int",
|
||||||
jsonObject.optInt("intKey", 0) == 42);
|
jsonObject.optInt("intKey", 0) == 42);
|
||||||
|
assertTrue("intKey should be Integer",
|
||||||
|
Integer.valueOf(42).equals(jsonObject.optIntegerObject("intKey")));
|
||||||
|
assertTrue("opt intKey should be Integer",
|
||||||
|
Integer.valueOf(42).equals(jsonObject.optIntegerObject("intKey", 0)));
|
||||||
assertTrue("opt intKey with default should be int",
|
assertTrue("opt intKey with default should be int",
|
||||||
jsonObject.getInt("intKey") == 42);
|
jsonObject.getInt("intKey") == 42);
|
||||||
assertTrue("intStrKey should be int",
|
assertTrue("intStrKey should be int",
|
||||||
@@ -918,6 +948,10 @@ public class JSONObjectTest {
|
|||||||
jsonObject.optLong("longKey") == 1234567890123456789L);
|
jsonObject.optLong("longKey") == 1234567890123456789L);
|
||||||
assertTrue("opt longKey with default should be long",
|
assertTrue("opt longKey with default should be long",
|
||||||
jsonObject.optLong("longKey", 0) == 1234567890123456789L);
|
jsonObject.optLong("longKey", 0) == 1234567890123456789L);
|
||||||
|
assertTrue("opt longKey should be Long",
|
||||||
|
Long.valueOf(1234567890123456789L).equals(jsonObject.optLongObject("longKey")));
|
||||||
|
assertTrue("opt longKey with default should be Long",
|
||||||
|
Long.valueOf(1234567890123456789L).equals(jsonObject.optLongObject("longKey", 0L)));
|
||||||
assertTrue("longStrKey should be long",
|
assertTrue("longStrKey should be long",
|
||||||
jsonObject.getLong("longStrKey") == 987654321098765432L);
|
jsonObject.getLong("longStrKey") == 987654321098765432L);
|
||||||
assertTrue("optNumber int should return Integer",
|
assertTrue("optNumber int should return Integer",
|
||||||
@@ -969,7 +1003,7 @@ public class JSONObjectTest {
|
|||||||
assertTrue( "0.2 should be a BigDecimal!",
|
assertTrue( "0.2 should be a BigDecimal!",
|
||||||
JSONObject.stringToValue( "0.2" ) instanceof BigDecimal );
|
JSONObject.stringToValue( "0.2" ) instanceof BigDecimal );
|
||||||
assertTrue( "Doubles should be BigDecimal, even when incorrectly converting floats!",
|
assertTrue( "Doubles should be BigDecimal, even when incorrectly converting floats!",
|
||||||
JSONObject.stringToValue( new Double( "0.2f" ).toString() ) instanceof BigDecimal );
|
JSONObject.stringToValue( Double.valueOf( "0.2f" ).toString() ) instanceof BigDecimal );
|
||||||
/**
|
/**
|
||||||
* This test documents a need for BigDecimal conversion.
|
* This test documents a need for BigDecimal conversion.
|
||||||
*/
|
*/
|
||||||
@@ -979,13 +1013,13 @@ public class JSONObjectTest {
|
|||||||
assertTrue( "1 should be an Integer!",
|
assertTrue( "1 should be an Integer!",
|
||||||
JSONObject.stringToValue( "1" ) instanceof Integer );
|
JSONObject.stringToValue( "1" ) instanceof Integer );
|
||||||
assertTrue( "Integer.MAX_VALUE should still be an Integer!",
|
assertTrue( "Integer.MAX_VALUE should still be an Integer!",
|
||||||
JSONObject.stringToValue( new Integer( Integer.MAX_VALUE ).toString() ) instanceof Integer );
|
JSONObject.stringToValue( Integer.valueOf( Integer.MAX_VALUE ).toString() ) instanceof Integer );
|
||||||
assertTrue( "Large integers should be a Long!",
|
assertTrue( "Large integers should be a Long!",
|
||||||
JSONObject.stringToValue( Long.valueOf(((long)Integer.MAX_VALUE) + 1 ) .toString() ) instanceof Long );
|
JSONObject.stringToValue( Long.valueOf(((long)Integer.MAX_VALUE) + 1 ) .toString() ) instanceof Long );
|
||||||
assertTrue( "Long.MAX_VALUE should still be an Integer!",
|
assertTrue( "Long.MAX_VALUE should still be an Integer!",
|
||||||
JSONObject.stringToValue( new Long( Long.MAX_VALUE ).toString() ) instanceof Long );
|
JSONObject.stringToValue( Long.valueOf( Long.MAX_VALUE ).toString() ) instanceof Long );
|
||||||
|
|
||||||
String str = new BigInteger( new Long( Long.MAX_VALUE ).toString() ).add( BigInteger.ONE ).toString();
|
String str = new BigInteger( Long.valueOf( Long.MAX_VALUE ).toString() ).add( BigInteger.ONE ).toString();
|
||||||
assertTrue( "Really large integers currently evaluate to BigInteger",
|
assertTrue( "Really large integers currently evaluate to BigInteger",
|
||||||
JSONObject.stringToValue(str).equals(new BigInteger("9223372036854775808")));
|
JSONObject.stringToValue(str).equals(new BigInteger("9223372036854775808")));
|
||||||
}
|
}
|
||||||
@@ -1058,7 +1092,7 @@ public class JSONObjectTest {
|
|||||||
assertTrue( "negativeFraction currently evaluates to double -0.01",
|
assertTrue( "negativeFraction currently evaluates to double -0.01",
|
||||||
jsonObject.get( "negativeFraction" ).equals(BigDecimal.valueOf(-0.01)));
|
jsonObject.get( "negativeFraction" ).equals(BigDecimal.valueOf(-0.01)));
|
||||||
assertTrue( "tooManyZerosFraction currently evaluates to double 0.001",
|
assertTrue( "tooManyZerosFraction currently evaluates to double 0.001",
|
||||||
jsonObject.get( "tooManyZerosFraction" ).equals(BigDecimal.valueOf(0.001)));
|
jsonObject.optLong( "tooManyZerosFraction" )==0);
|
||||||
assertTrue( "negativeHexFloat currently evaluates to double -3.99951171875",
|
assertTrue( "negativeHexFloat currently evaluates to double -3.99951171875",
|
||||||
jsonObject.get( "negativeHexFloat" ).equals(Double.valueOf(-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",
|
||||||
@@ -1232,8 +1266,8 @@ public class JSONObjectTest {
|
|||||||
String key30 = "key30";
|
String key30 = "key30";
|
||||||
String key31 = "key31";
|
String key31 = "key31";
|
||||||
JSONObject jsonObject = new JSONObject();
|
JSONObject jsonObject = new JSONObject();
|
||||||
jsonObject.put(key30, new Double(3.0));
|
jsonObject.put(key30, Double.valueOf(3.0));
|
||||||
jsonObject.put(key31, new Double(3.1));
|
jsonObject.put(key31, Double.valueOf(3.1));
|
||||||
|
|
||||||
assertTrue("3.0 should remain a double",
|
assertTrue("3.0 should remain a double",
|
||||||
jsonObject.getDouble(key30) == 3);
|
jsonObject.getDouble(key30) == 3);
|
||||||
@@ -1686,19 +1720,19 @@ public class JSONObjectTest {
|
|||||||
*/
|
*/
|
||||||
assertFalse("Document unexpected behaviour with explicit type-casting float as double!", (double)0.2f == 0.2d );
|
assertFalse("Document unexpected behaviour with explicit type-casting float as double!", (double)0.2f == 0.2d );
|
||||||
assertFalse("Document unexpected behaviour with implicit type-cast!", 0.2f == 0.2d );
|
assertFalse("Document unexpected behaviour with implicit type-cast!", 0.2f == 0.2d );
|
||||||
Double d1 = new Double( 1.1f );
|
Double d1 = Double.valueOf( 1.1f );
|
||||||
Double d2 = new Double( "1.1f" );
|
Double d2 = Double.valueOf( "1.1f" );
|
||||||
assertFalse( "Document implicit type cast from float to double before calling Double(double d) constructor", d1.equals( d2 ) );
|
assertFalse( "Document implicit type cast from float to double before calling Double(double d) constructor", d1.equals( d2 ) );
|
||||||
|
|
||||||
assertTrue( "Correctly converting float to double via base10 (string) representation!", new Double( 3.1d ).equals( new Double( new Float( 3.1f ).toString() ) ) );
|
assertTrue( "Correctly converting float to double via base10 (string) representation!", Double.valueOf( 3.1d ).equals( Double.valueOf( Float.valueOf( 3.1f ).toString() ) ) );
|
||||||
|
|
||||||
// Pinpointing the not so obvious "buggy" conversion from float to double in JSONObject
|
// Pinpointing the not so obvious "buggy" conversion from float to double in JSONObject
|
||||||
JSONObject jo = new JSONObject();
|
JSONObject jo = new JSONObject();
|
||||||
jo.put( "bug", 3.1f ); // will call put( String key, double value ) with implicit and "buggy" type-cast from float to double
|
jo.put( "bug", 3.1f ); // will call put( String key, double value ) with implicit and "buggy" type-cast from float to double
|
||||||
assertFalse( "The java-compiler did add some zero bits for you to the mantissa (unexpected, but well documented)", jo.get( "bug" ).equals( new Double( 3.1d ) ) );
|
assertFalse( "The java-compiler did add some zero bits for you to the mantissa (unexpected, but well documented)", jo.get( "bug" ).equals( Double.valueOf( 3.1d ) ) );
|
||||||
|
|
||||||
JSONObject inc = new JSONObject();
|
JSONObject inc = new JSONObject();
|
||||||
inc.put( "bug", new Float( 3.1f ) ); // This will put in instance of Float into JSONObject, i.e. call put( String key, Object value )
|
inc.put( "bug", Float.valueOf( 3.1f ) ); // This will put in instance of Float into JSONObject, i.e. call put( String key, Object value )
|
||||||
assertTrue( "Everything is ok here!", inc.get( "bug" ) instanceof Float );
|
assertTrue( "Everything is ok here!", inc.get( "bug" ) instanceof Float );
|
||||||
inc.increment( "bug" ); // after adding 1, increment will call put( String key, double value ) with implicit and "buggy" type-cast from float to double!
|
inc.increment( "bug" ); // after adding 1, increment will call put( String key, double value ) with implicit and "buggy" type-cast from float to double!
|
||||||
// this.put(key, (Float) value + 1);
|
// this.put(key, (Float) value + 1);
|
||||||
@@ -1942,7 +1976,7 @@ public class JSONObjectTest {
|
|||||||
@Test
|
@Test
|
||||||
public void jsonObjectToStringSuppressWarningOnCastToMap() {
|
public void jsonObjectToStringSuppressWarningOnCastToMap() {
|
||||||
JSONObject jsonObject = new JSONObject();
|
JSONObject jsonObject = new JSONObject();
|
||||||
Map<String, String> map = new HashMap();
|
Map<String, String> map = new HashMap<>();
|
||||||
map.put("abc", "def");
|
map.put("abc", "def");
|
||||||
jsonObject.put("key", map);
|
jsonObject.put("key", map);
|
||||||
|
|
||||||
@@ -2000,7 +2034,9 @@ public class JSONObjectTest {
|
|||||||
"}";
|
"}";
|
||||||
JSONObject jsonObject = new JSONObject(jsonObjectStr);
|
JSONObject jsonObject = new JSONObject(jsonObjectStr);
|
||||||
assertTrue("jsonObject valueToString() incorrect",
|
assertTrue("jsonObject valueToString() incorrect",
|
||||||
JSONObject.valueToString(jsonObject).equals(jsonObject.toString()));
|
new JSONObject(JSONObject.valueToString(jsonObject))
|
||||||
|
.similar(new JSONObject(jsonObject.toString()))
|
||||||
|
);
|
||||||
String jsonArrayStr =
|
String jsonArrayStr =
|
||||||
"[1,2,3]";
|
"[1,2,3]";
|
||||||
JSONArray jsonArray = new JSONArray(jsonArrayStr);
|
JSONArray jsonArray = new JSONArray(jsonArrayStr);
|
||||||
@@ -2011,16 +2047,17 @@ public class JSONObjectTest {
|
|||||||
map.put("key2", "val2");
|
map.put("key2", "val2");
|
||||||
map.put("key3", "val3");
|
map.put("key3", "val3");
|
||||||
assertTrue("map valueToString() incorrect",
|
assertTrue("map valueToString() incorrect",
|
||||||
jsonObject.toString().equals(JSONObject.valueToString(map)));
|
new JSONObject(jsonObject.toString())
|
||||||
|
.similar(new JSONObject(JSONObject.valueToString(map))));
|
||||||
Collection<Integer> collection = new ArrayList<Integer>();
|
Collection<Integer> collection = new ArrayList<Integer>();
|
||||||
collection.add(new Integer(1));
|
collection.add(Integer.valueOf(1));
|
||||||
collection.add(new Integer(2));
|
collection.add(Integer.valueOf(2));
|
||||||
collection.add(new Integer(3));
|
collection.add(Integer.valueOf(3));
|
||||||
assertTrue("collection valueToString() expected: "+
|
assertTrue("collection valueToString() expected: "+
|
||||||
jsonArray.toString()+ " actual: "+
|
jsonArray.toString()+ " actual: "+
|
||||||
JSONObject.valueToString(collection),
|
JSONObject.valueToString(collection),
|
||||||
jsonArray.toString().equals(JSONObject.valueToString(collection)));
|
jsonArray.toString().equals(JSONObject.valueToString(collection)));
|
||||||
Integer[] array = { new Integer(1), new Integer(2), new Integer(3) };
|
Integer[] array = { Integer.valueOf(1), Integer.valueOf(2), Integer.valueOf(3) };
|
||||||
assertTrue("array valueToString() incorrect",
|
assertTrue("array valueToString() incorrect",
|
||||||
jsonArray.toString().equals(JSONObject.valueToString(array)));
|
jsonArray.toString().equals(JSONObject.valueToString(array)));
|
||||||
Util.checkJSONObjectMaps(jsonObject);
|
Util.checkJSONObjectMaps(jsonObject);
|
||||||
@@ -2058,7 +2095,7 @@ public class JSONObjectTest {
|
|||||||
JSONObject.NULL == JSONObject.wrap(null));
|
JSONObject.NULL == JSONObject.wrap(null));
|
||||||
|
|
||||||
// wrap(Integer) returns Integer
|
// wrap(Integer) returns Integer
|
||||||
Integer in = new Integer(1);
|
Integer in = Integer.valueOf(1);
|
||||||
assertTrue("Integer wrap() incorrect",
|
assertTrue("Integer wrap() incorrect",
|
||||||
in == JSONObject.wrap(in));
|
in == JSONObject.wrap(in));
|
||||||
|
|
||||||
@@ -2085,9 +2122,9 @@ public class JSONObjectTest {
|
|||||||
|
|
||||||
// wrap collection returns JSONArray
|
// wrap collection returns JSONArray
|
||||||
Collection<Integer> collection = new ArrayList<Integer>();
|
Collection<Integer> collection = new ArrayList<Integer>();
|
||||||
collection.add(new Integer(1));
|
collection.add(Integer.valueOf(1));
|
||||||
collection.add(new Integer(2));
|
collection.add(Integer.valueOf(2));
|
||||||
collection.add(new Integer(3));
|
collection.add(Integer.valueOf(3));
|
||||||
JSONArray jsonArray = (JSONArray) (JSONObject.wrap(collection));
|
JSONArray jsonArray = (JSONArray) (JSONObject.wrap(collection));
|
||||||
|
|
||||||
// validate JSON
|
// validate JSON
|
||||||
@@ -2098,7 +2135,7 @@ public class JSONObjectTest {
|
|||||||
assertTrue("expected 3", Integer.valueOf(3).equals(jsonArray.query("/2")));
|
assertTrue("expected 3", Integer.valueOf(3).equals(jsonArray.query("/2")));
|
||||||
|
|
||||||
// wrap Array returns JSONArray
|
// wrap Array returns JSONArray
|
||||||
Integer[] array = { new Integer(1), new Integer(2), new Integer(3) };
|
Integer[] array = { Integer.valueOf(1), Integer.valueOf(2), Integer.valueOf(3) };
|
||||||
JSONArray integerArrayJsonArray = (JSONArray)(JSONObject.wrap(array));
|
JSONArray integerArrayJsonArray = (JSONArray)(JSONObject.wrap(array));
|
||||||
|
|
||||||
// validate JSON
|
// validate JSON
|
||||||
@@ -2198,6 +2235,51 @@ public class JSONObjectTest {
|
|||||||
"Expected a ',' or '}' at 15 [character 16 line 1]",
|
"Expected a ',' or '}' at 15 [character 16 line 1]",
|
||||||
e.getMessage());
|
e.getMessage());
|
||||||
}
|
}
|
||||||
|
try {
|
||||||
|
// key is a nested map
|
||||||
|
String str = "{{\"foo\": \"bar\"}: \"baz\"}";
|
||||||
|
assertNull("Expected an exception",new JSONObject(str));
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Missing value at 1 [character 2 line 1]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
// key is a nested array containing a map
|
||||||
|
String str = "{\"a\": 1, [{\"foo\": \"bar\"}]: \"baz\"}";
|
||||||
|
assertNull("Expected an exception",new JSONObject(str));
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Missing value at 9 [character 10 line 1]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
// key contains }
|
||||||
|
String str = "{foo}: 2}";
|
||||||
|
assertNull("Expected an exception",new JSONObject(str));
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Expected a ':' after a key at 5 [character 6 line 1]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
// key contains ]
|
||||||
|
String str = "{foo]: 2}";
|
||||||
|
assertNull("Expected an exception",new JSONObject(str));
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Expected a ':' after a key at 5 [character 6 line 1]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
// \0 after ,
|
||||||
|
String str = "{\"myKey\":true, \0\"myOtherKey\":false}";
|
||||||
|
assertNull("Expected an exception",new JSONObject(str));
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"A JSONObject text must end with '}' at 15 [character 16 line 1]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
try {
|
try {
|
||||||
// append to wrong key
|
// append to wrong key
|
||||||
String str = "{\"myKey\":true, \"myOtherKey\":false}";
|
String str = "{\"myKey\":true, \"myOtherKey\":false}";
|
||||||
@@ -2465,20 +2547,32 @@ public class JSONObjectTest {
|
|||||||
BigInteger.TEN.compareTo(jsonObject.optBigInteger("myKey",BigInteger.TEN ))==0);
|
BigInteger.TEN.compareTo(jsonObject.optBigInteger("myKey",BigInteger.TEN ))==0);
|
||||||
assertTrue("optBoolean() should return default boolean",
|
assertTrue("optBoolean() should return default boolean",
|
||||||
jsonObject.optBoolean("myKey", true));
|
jsonObject.optBoolean("myKey", true));
|
||||||
|
assertTrue("optBooleanObject() should return default Boolean",
|
||||||
|
jsonObject.optBooleanObject("myKey", true));
|
||||||
assertTrue("optInt() should return default int",
|
assertTrue("optInt() should return default int",
|
||||||
42 == jsonObject.optInt("myKey", 42));
|
42 == jsonObject.optInt("myKey", 42));
|
||||||
|
assertTrue("optIntegerObject() should return default Integer",
|
||||||
|
Integer.valueOf(42).equals(jsonObject.optIntegerObject("myKey", 42)));
|
||||||
assertTrue("optEnum() should return default Enum",
|
assertTrue("optEnum() should return default Enum",
|
||||||
MyEnum.VAL1.equals(jsonObject.optEnum(MyEnum.class, "myKey", MyEnum.VAL1)));
|
MyEnum.VAL1.equals(jsonObject.optEnum(MyEnum.class, "myKey", MyEnum.VAL1)));
|
||||||
assertTrue("optJSONArray() should return null ",
|
assertTrue("optJSONArray() should return null ",
|
||||||
null==jsonObject.optJSONArray("myKey"));
|
null==jsonObject.optJSONArray("myKey"));
|
||||||
|
assertTrue("optJSONArray() should return default JSONArray",
|
||||||
|
"value".equals(jsonObject.optJSONArray("myKey", new JSONArray("[\"value\"]")).getString(0)));
|
||||||
assertTrue("optJSONObject() should return default JSONObject ",
|
assertTrue("optJSONObject() should return default JSONObject ",
|
||||||
jsonObject.optJSONObject("myKey", new JSONObject("{\"testKey\":\"testValue\"}")).getString("testKey").equals("testValue"));
|
jsonObject.optJSONObject("myKey", new JSONObject("{\"testKey\":\"testValue\"}")).getString("testKey").equals("testValue"));
|
||||||
assertTrue("optLong() should return default long",
|
assertTrue("optLong() should return default long",
|
||||||
42l == jsonObject.optLong("myKey", 42l));
|
42l == jsonObject.optLong("myKey", 42l));
|
||||||
|
assertTrue("optLongObject() should return default Long",
|
||||||
|
Long.valueOf(42l).equals(jsonObject.optLongObject("myKey", 42l)));
|
||||||
assertTrue("optDouble() should return default double",
|
assertTrue("optDouble() should return default double",
|
||||||
42.3d == jsonObject.optDouble("myKey", 42.3d));
|
42.3d == jsonObject.optDouble("myKey", 42.3d));
|
||||||
|
assertTrue("optDoubleObject() should return default Double",
|
||||||
|
Double.valueOf(42.3d).equals(jsonObject.optDoubleObject("myKey", 42.3d)));
|
||||||
assertTrue("optFloat() should return default float",
|
assertTrue("optFloat() should return default float",
|
||||||
42.3f == jsonObject.optFloat("myKey", 42.3f));
|
42.3f == jsonObject.optFloat("myKey", 42.3f));
|
||||||
|
assertTrue("optFloatObject() should return default Float",
|
||||||
|
Float.valueOf(42.3f).equals(jsonObject.optFloatObject("myKey", 42.3f)));
|
||||||
assertTrue("optNumber() should return default Number",
|
assertTrue("optNumber() should return default Number",
|
||||||
42l == jsonObject.optNumber("myKey", Long.valueOf(42)).longValue());
|
42l == jsonObject.optNumber("myKey", Long.valueOf(42)).longValue());
|
||||||
assertTrue("optString() should return default string",
|
assertTrue("optString() should return default string",
|
||||||
@@ -2502,20 +2596,32 @@ public class JSONObjectTest {
|
|||||||
BigInteger.TEN.compareTo(jsonObject.optBigInteger("myKey",BigInteger.TEN ))==0);
|
BigInteger.TEN.compareTo(jsonObject.optBigInteger("myKey",BigInteger.TEN ))==0);
|
||||||
assertTrue("optBoolean() should return default boolean",
|
assertTrue("optBoolean() should return default boolean",
|
||||||
jsonObject.optBoolean("myKey", true));
|
jsonObject.optBoolean("myKey", true));
|
||||||
|
assertTrue("optBooleanObject() should return default Boolean",
|
||||||
|
jsonObject.optBooleanObject("myKey", true));
|
||||||
assertTrue("optInt() should return default int",
|
assertTrue("optInt() should return default int",
|
||||||
42 == jsonObject.optInt("myKey", 42));
|
42 == jsonObject.optInt("myKey", 42));
|
||||||
|
assertTrue("optIntegerObject() should return default Integer",
|
||||||
|
Integer.valueOf(42).equals(jsonObject.optIntegerObject("myKey", 42)));
|
||||||
assertTrue("optEnum() should return default Enum",
|
assertTrue("optEnum() should return default Enum",
|
||||||
MyEnum.VAL1.equals(jsonObject.optEnum(MyEnum.class, "myKey", MyEnum.VAL1)));
|
MyEnum.VAL1.equals(jsonObject.optEnum(MyEnum.class, "myKey", MyEnum.VAL1)));
|
||||||
|
assertTrue("optJSONArray() should return default JSONArray",
|
||||||
|
"value".equals(jsonObject.optJSONArray("myKey", new JSONArray("[\"value\"]")).getString(0)));
|
||||||
assertTrue("optJSONArray() should return null ",
|
assertTrue("optJSONArray() should return null ",
|
||||||
null==jsonObject.optJSONArray("myKey"));
|
null==jsonObject.optJSONArray("myKey"));
|
||||||
assertTrue("optJSONObject() should return default JSONObject ",
|
assertTrue("optJSONObject() should return default JSONObject ",
|
||||||
jsonObject.optJSONObject("myKey", new JSONObject("{\"testKey\":\"testValue\"}")).getString("testKey").equals("testValue"));
|
jsonObject.optJSONObject("myKey", new JSONObject("{\"testKey\":\"testValue\"}")).getString("testKey").equals("testValue"));
|
||||||
assertTrue("optLong() should return default long",
|
assertTrue("optLong() should return default long",
|
||||||
42l == jsonObject.optLong("myKey", 42l));
|
42l == jsonObject.optLong("myKey", 42l));
|
||||||
|
assertTrue("optLongObject() should return default Long",
|
||||||
|
Long.valueOf(42l).equals(jsonObject.optLongObject("myKey", 42l)));
|
||||||
assertTrue("optDouble() should return default double",
|
assertTrue("optDouble() should return default double",
|
||||||
42.3d == jsonObject.optDouble("myKey", 42.3d));
|
42.3d == jsonObject.optDouble("myKey", 42.3d));
|
||||||
|
assertTrue("optDoubleObject() should return default Double",
|
||||||
|
Double.valueOf(42.3d).equals(jsonObject.optDoubleObject("myKey", 42.3d)));
|
||||||
assertTrue("optFloat() should return default float",
|
assertTrue("optFloat() should return default float",
|
||||||
42.3f == jsonObject.optFloat("myKey", 42.3f));
|
42.3f == jsonObject.optFloat("myKey", 42.3f));
|
||||||
|
assertTrue("optFloatObject() should return default Float",
|
||||||
|
Float.valueOf(42.3f).equals(jsonObject.optFloatObject("myKey", 42.3f)));
|
||||||
assertTrue("optNumber() should return default Number",
|
assertTrue("optNumber() should return default Number",
|
||||||
42l == jsonObject.optNumber("myKey", Long.valueOf(42)).longValue());
|
42l == jsonObject.optNumber("myKey", Long.valueOf(42)).longValue());
|
||||||
assertTrue("optString() should return default string",
|
assertTrue("optString() should return default string",
|
||||||
@@ -2530,11 +2636,17 @@ public class JSONObjectTest {
|
|||||||
public void jsonObjectOptStringConversion() {
|
public void jsonObjectOptStringConversion() {
|
||||||
JSONObject jo = new JSONObject("{\"int\":\"123\",\"true\":\"true\",\"false\":\"false\"}");
|
JSONObject jo = new JSONObject("{\"int\":\"123\",\"true\":\"true\",\"false\":\"false\"}");
|
||||||
assertTrue("unexpected optBoolean value",jo.optBoolean("true",false)==true);
|
assertTrue("unexpected optBoolean value",jo.optBoolean("true",false)==true);
|
||||||
|
assertTrue("unexpected optBooleanObject value",Boolean.valueOf(true).equals(jo.optBooleanObject("true",false)));
|
||||||
assertTrue("unexpected optBoolean value",jo.optBoolean("false",true)==false);
|
assertTrue("unexpected optBoolean value",jo.optBoolean("false",true)==false);
|
||||||
|
assertTrue("unexpected optBooleanObject value",Boolean.valueOf(false).equals(jo.optBooleanObject("false",true)));
|
||||||
assertTrue("unexpected optInt value",jo.optInt("int",0)==123);
|
assertTrue("unexpected optInt value",jo.optInt("int",0)==123);
|
||||||
|
assertTrue("unexpected optIntegerObject value",Integer.valueOf(123).equals(jo.optIntegerObject("int",0)));
|
||||||
assertTrue("unexpected optLong value",jo.optLong("int",0)==123l);
|
assertTrue("unexpected optLong value",jo.optLong("int",0)==123l);
|
||||||
|
assertTrue("unexpected optLongObject value",Long.valueOf(123l).equals(jo.optLongObject("int",0L)));
|
||||||
assertTrue("unexpected optDouble value",jo.optDouble("int",0.0d)==123.0d);
|
assertTrue("unexpected optDouble value",jo.optDouble("int",0.0d)==123.0d);
|
||||||
|
assertTrue("unexpected optDoubleObject value",Double.valueOf(123.0d).equals(jo.optDoubleObject("int",0.0d)));
|
||||||
assertTrue("unexpected optFloat value",jo.optFloat("int",0.0f)==123.0f);
|
assertTrue("unexpected optFloat value",jo.optFloat("int",0.0f)==123.0f);
|
||||||
|
assertTrue("unexpected optFloatObject value",Float.valueOf(123.0f).equals(jo.optFloatObject("int",0.0f)));
|
||||||
assertTrue("unexpected optBigInteger value",jo.optBigInteger("int",BigInteger.ZERO).compareTo(new BigInteger("123"))==0);
|
assertTrue("unexpected optBigInteger value",jo.optBigInteger("int",BigInteger.ZERO).compareTo(new BigInteger("123"))==0);
|
||||||
assertTrue("unexpected optBigDecimal value",jo.optBigDecimal("int",BigDecimal.ZERO).compareTo(new BigDecimal("123"))==0);
|
assertTrue("unexpected optBigDecimal value",jo.optBigDecimal("int",BigDecimal.ZERO).compareTo(new BigDecimal("123"))==0);
|
||||||
assertTrue("unexpected optBigDecimal value",jo.optBigDecimal("int",BigDecimal.ZERO).compareTo(new BigDecimal("123"))==0);
|
assertTrue("unexpected optBigDecimal value",jo.optBigDecimal("int",BigDecimal.ZERO).compareTo(new BigDecimal("123"))==0);
|
||||||
@@ -2555,23 +2667,35 @@ public class JSONObjectTest {
|
|||||||
assertEquals(new BigDecimal("19007199254740993.35481234487103587486413587843213584"), jo.optBigDecimal("largeNumber",null));
|
assertEquals(new BigDecimal("19007199254740993.35481234487103587486413587843213584"), jo.optBigDecimal("largeNumber",null));
|
||||||
assertEquals(new BigInteger("19007199254740993"), jo.optBigInteger("largeNumber",null));
|
assertEquals(new BigInteger("19007199254740993"), jo.optBigInteger("largeNumber",null));
|
||||||
assertEquals(1.9007199254740992E16, jo.optDouble("largeNumber"),0.0);
|
assertEquals(1.9007199254740992E16, jo.optDouble("largeNumber"),0.0);
|
||||||
|
assertEquals(1.9007199254740992E16, jo.optDoubleObject("largeNumber"),0.0);
|
||||||
assertEquals(1.90071995E16f, jo.optFloat("largeNumber"),0.0f);
|
assertEquals(1.90071995E16f, jo.optFloat("largeNumber"),0.0f);
|
||||||
|
assertEquals(1.90071995E16f, jo.optFloatObject("largeNumber"),0.0f);
|
||||||
assertEquals(19007199254740993l, jo.optLong("largeNumber"));
|
assertEquals(19007199254740993l, jo.optLong("largeNumber"));
|
||||||
|
assertEquals(Long.valueOf(19007199254740993l), jo.optLongObject("largeNumber"));
|
||||||
assertEquals(1874919425, jo.optInt("largeNumber"));
|
assertEquals(1874919425, jo.optInt("largeNumber"));
|
||||||
|
assertEquals(Integer.valueOf(1874919425), jo.optIntegerObject("largeNumber"));
|
||||||
|
|
||||||
// conversion from a string
|
// conversion from a string
|
||||||
assertEquals(new BigDecimal("19007199254740993.35481234487103587486413587843213584"), jo.optBigDecimal("largeNumberStr",null));
|
assertEquals(new BigDecimal("19007199254740993.35481234487103587486413587843213584"), jo.optBigDecimal("largeNumberStr",null));
|
||||||
assertEquals(new BigInteger("19007199254740993"), jo.optBigInteger("largeNumberStr",null));
|
assertEquals(new BigInteger("19007199254740993"), jo.optBigInteger("largeNumberStr",null));
|
||||||
assertEquals(1.9007199254740992E16, jo.optDouble("largeNumberStr"),0.0);
|
assertEquals(1.9007199254740992E16, jo.optDouble("largeNumberStr"),0.0);
|
||||||
|
assertEquals(1.9007199254740992E16, jo.optDoubleObject("largeNumberStr"),0.0);
|
||||||
assertEquals(1.90071995E16f, jo.optFloat("largeNumberStr"),0.0f);
|
assertEquals(1.90071995E16f, jo.optFloat("largeNumberStr"),0.0f);
|
||||||
|
assertEquals(1.90071995E16f, jo.optFloatObject("largeNumberStr"),0.0f);
|
||||||
assertEquals(19007199254740993l, jo.optLong("largeNumberStr"));
|
assertEquals(19007199254740993l, jo.optLong("largeNumberStr"));
|
||||||
|
assertEquals(Long.valueOf(19007199254740993l), jo.optLongObject("largeNumberStr"));
|
||||||
assertEquals(1874919425, jo.optInt("largeNumberStr"));
|
assertEquals(1874919425, jo.optInt("largeNumberStr"));
|
||||||
|
assertEquals(Integer.valueOf(1874919425), jo.optIntegerObject("largeNumberStr"));
|
||||||
|
|
||||||
// the integer portion of the actual value is larger than a double can hold.
|
// the integer portion of the actual value is larger than a double can hold.
|
||||||
assertNotEquals((long)Double.parseDouble("19007199254740993.35481234487103587486413587843213584"), jo.optLong("largeNumber"));
|
assertNotEquals((long)Double.parseDouble("19007199254740993.35481234487103587486413587843213584"), jo.optLong("largeNumber"));
|
||||||
|
assertNotEquals(Long.valueOf((long)Double.parseDouble("19007199254740993.35481234487103587486413587843213584")), jo.optLongObject("largeNumber"));
|
||||||
assertNotEquals((int)Double.parseDouble("19007199254740993.35481234487103587486413587843213584"), jo.optInt("largeNumber"));
|
assertNotEquals((int)Double.parseDouble("19007199254740993.35481234487103587486413587843213584"), jo.optInt("largeNumber"));
|
||||||
|
assertNotEquals(Integer.valueOf((int)Double.parseDouble("19007199254740993.35481234487103587486413587843213584")), jo.optIntegerObject("largeNumber"));
|
||||||
assertNotEquals((long)Double.parseDouble("19007199254740993.35481234487103587486413587843213584"), jo.optLong("largeNumberStr"));
|
assertNotEquals((long)Double.parseDouble("19007199254740993.35481234487103587486413587843213584"), jo.optLong("largeNumberStr"));
|
||||||
|
assertNotEquals(Long.valueOf((long)Double.parseDouble("19007199254740993.35481234487103587486413587843213584")), jo.optLongObject("largeNumberStr"));
|
||||||
assertNotEquals((int)Double.parseDouble("19007199254740993.35481234487103587486413587843213584"), jo.optInt("largeNumberStr"));
|
assertNotEquals((int)Double.parseDouble("19007199254740993.35481234487103587486413587843213584"), jo.optInt("largeNumberStr"));
|
||||||
|
assertNotEquals(Integer.valueOf((int)Double.parseDouble("19007199254740993.35481234487103587486413587843213584")), jo.optIntegerObject("largeNumberStr"));
|
||||||
assertEquals(19007199254740992l, (long)Double.parseDouble("19007199254740993.35481234487103587486413587843213584"));
|
assertEquals(19007199254740992l, (long)Double.parseDouble("19007199254740993.35481234487103587486413587843213584"));
|
||||||
assertEquals(2147483647, (int)Double.parseDouble("19007199254740993.35481234487103587486413587843213584"));
|
assertEquals(2147483647, (int)Double.parseDouble("19007199254740993.35481234487103587486413587843213584"));
|
||||||
Util.checkJSONObjectMaps(jo);
|
Util.checkJSONObjectMaps(jo);
|
||||||
@@ -3166,7 +3290,7 @@ public class JSONObjectTest {
|
|||||||
@SuppressWarnings("boxing")
|
@SuppressWarnings("boxing")
|
||||||
@Test
|
@Test
|
||||||
public void testGenericBean() {
|
public void testGenericBean() {
|
||||||
GenericBean<Integer> bean = new GenericBean(42);
|
GenericBean<Integer> bean = new GenericBean<>(42);
|
||||||
final JSONObject jo = new JSONObject(bean);
|
final JSONObject jo = new JSONObject(bean);
|
||||||
assertEquals(jo.keySet().toString(), 8, jo.length());
|
assertEquals(jo.keySet().toString(), 8, jo.length());
|
||||||
assertEquals(42, jo.get("genericValue"));
|
assertEquals(42, jo.get("genericValue"));
|
||||||
@@ -3215,6 +3339,7 @@ public class JSONObjectTest {
|
|||||||
* Sample test case from https://github.com/stleary/JSON-java/issues/531
|
* Sample test case from https://github.com/stleary/JSON-java/issues/531
|
||||||
* which verifies that no regression in double/BigDecimal support is present.
|
* which verifies that no regression in double/BigDecimal support is present.
|
||||||
*/
|
*/
|
||||||
|
@Test
|
||||||
public void testObjectToBigDecimal() {
|
public void testObjectToBigDecimal() {
|
||||||
double value = 1412078745.01074;
|
double value = 1412078745.01074;
|
||||||
Reader reader = new StringReader("[{\"value\": " + value + "}]");
|
Reader reader = new StringReader("[{\"value\": " + value + "}]");
|
||||||
@@ -3479,6 +3604,7 @@ public class JSONObjectTest {
|
|||||||
/**
|
/**
|
||||||
* Tests for stack overflow. See https://github.com/stleary/JSON-java/issues/654
|
* Tests for stack overflow. See https://github.com/stleary/JSON-java/issues/654
|
||||||
*/
|
*/
|
||||||
|
@Ignore("This test relies on system constraints and may not always pass. See: https://github.com/stleary/JSON-java/issues/821")
|
||||||
@Test(expected = JSONException.class)
|
@Test(expected = JSONException.class)
|
||||||
public void issue654StackOverflowInputWellFormed() {
|
public void issue654StackOverflowInputWellFormed() {
|
||||||
//String input = new String(java.util.Base64.getDecoder().decode(base64Bytes));
|
//String input = new String(java.util.Base64.getDecoder().decode(base64Bytes));
|
||||||
@@ -3486,7 +3612,7 @@ public class JSONObjectTest {
|
|||||||
JSONTokener tokener = new JSONTokener(resourceAsStream);
|
JSONTokener tokener = new JSONTokener(resourceAsStream);
|
||||||
JSONObject json_input = new JSONObject(tokener);
|
JSONObject json_input = new JSONObject(tokener);
|
||||||
assertNotNull(json_input);
|
assertNotNull(json_input);
|
||||||
fail("Excepected Exception.");
|
fail("Excepected Exception due to stack overflow.");
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@@ -3509,4 +3635,159 @@ public class JSONObjectTest {
|
|||||||
.put("b", 2);
|
.put("b", 2);
|
||||||
assertFalse(jo1.similar(jo3));
|
assertFalse(jo1.similar(jo3));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static final Number[] NON_FINITE_NUMBERS = { Double.POSITIVE_INFINITY, Double.NEGATIVE_INFINITY, Double.NaN,
|
||||||
|
Float.POSITIVE_INFINITY, Float.NEGATIVE_INFINITY, Float.NaN };
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void issue713MapConstructorWithNonFiniteNumbers() {
|
||||||
|
for (Number nonFinite : NON_FINITE_NUMBERS) {
|
||||||
|
Map<String, Number> map = new HashMap<>();
|
||||||
|
map.put("a", nonFinite);
|
||||||
|
|
||||||
|
assertThrows(JSONException.class, () -> new JSONObject(map));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void issue713BeanConstructorWithNonFiniteNumbers() {
|
||||||
|
for (Number nonFinite : NON_FINITE_NUMBERS) {
|
||||||
|
GenericBean<Number> bean = new GenericBean<>(nonFinite);
|
||||||
|
assertThrows(JSONException.class, () -> new JSONObject(bean));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = JSONException.class)
|
||||||
|
public void issue743SerializationMap() {
|
||||||
|
HashMap<String, Object> map = new HashMap<>();
|
||||||
|
map.put("t", map);
|
||||||
|
JSONObject object = new JSONObject(map);
|
||||||
|
String jsonString = object.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = JSONException.class)
|
||||||
|
public void testCircularReferenceMultipleLevel() {
|
||||||
|
HashMap<String, Object> inside = new HashMap<>();
|
||||||
|
HashMap<String, Object> jsonObject = new HashMap<>();
|
||||||
|
inside.put("inside", jsonObject);
|
||||||
|
jsonObject.put("test", inside);
|
||||||
|
new JSONObject(jsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void issue743SerializationMapWith512Objects() {
|
||||||
|
HashMap<String, Object> map = buildNestedMap(ParserConfiguration.DEFAULT_MAXIMUM_NESTING_DEPTH);
|
||||||
|
JSONObject object = new JSONObject(map);
|
||||||
|
String jsonString = object.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void issue743SerializationMapWith1000Objects() {
|
||||||
|
HashMap<String, Object> map = buildNestedMap(1000);
|
||||||
|
JSONParserConfiguration parserConfiguration = new JSONParserConfiguration().withMaxNestingDepth(1000);
|
||||||
|
JSONObject object = new JSONObject(map, parserConfiguration);
|
||||||
|
String jsonString = object.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = JSONException.class)
|
||||||
|
public void issue743SerializationMapWith1001Objects() {
|
||||||
|
HashMap<String, Object> map = buildNestedMap(1001);
|
||||||
|
JSONObject object = new JSONObject(map);
|
||||||
|
String jsonString = object.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = JSONException.class)
|
||||||
|
public void testCircleReferenceFirstLevel() {
|
||||||
|
Map<Object, Object> jsonObject = new HashMap<>();
|
||||||
|
|
||||||
|
jsonObject.put("test", jsonObject);
|
||||||
|
|
||||||
|
new JSONObject(jsonObject, new JSONParserConfiguration());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = StackOverflowError.class)
|
||||||
|
public void testCircleReferenceMultiplyLevel_notConfigured_expectedStackOverflow() {
|
||||||
|
Map<Object, Object> inside = new HashMap<>();
|
||||||
|
|
||||||
|
Map<Object, Object> jsonObject = new HashMap<>();
|
||||||
|
inside.put("test", jsonObject);
|
||||||
|
jsonObject.put("test", inside);
|
||||||
|
|
||||||
|
new JSONObject(jsonObject, new JSONParserConfiguration().withMaxNestingDepth(99999));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = JSONException.class)
|
||||||
|
public void testCircleReferenceMultiplyLevel_configured_expectedJSONException() {
|
||||||
|
Map<Object, Object> inside = new HashMap<>();
|
||||||
|
|
||||||
|
Map<Object, Object> jsonObject = new HashMap<>();
|
||||||
|
inside.put("test", jsonObject);
|
||||||
|
jsonObject.put("test", inside);
|
||||||
|
|
||||||
|
new JSONObject(jsonObject, new JSONParserConfiguration());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDifferentKeySameInstanceNotACircleReference() {
|
||||||
|
Map<Object, Object> map1 = new HashMap<>();
|
||||||
|
Map<Object, Object> map2 = new HashMap<>();
|
||||||
|
|
||||||
|
map1.put("test1", map2);
|
||||||
|
map1.put("test2", map2);
|
||||||
|
|
||||||
|
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
|
||||||
|
*
|
||||||
|
* @param maxDepth
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public static HashMap<String, Object> buildNestedMap(int maxDepth) {
|
||||||
|
if (maxDepth <= 0) {
|
||||||
|
return new HashMap<>();
|
||||||
|
}
|
||||||
|
HashMap<String, Object> nestedMap = new HashMap<>();
|
||||||
|
nestedMap.put("t", buildNestedMap(maxDepth - 1));
|
||||||
|
return nestedMap;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,11 +4,6 @@ package org.json.junit;
|
|||||||
Public Domain.
|
Public Domain.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
|
||||||
import static org.junit.Assert.assertNotEquals;
|
|
||||||
import static org.junit.Assert.assertTrue;
|
|
||||||
import static org.junit.Assert.fail;
|
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.FileReader;
|
import java.io.FileReader;
|
||||||
import java.io.FileWriter;
|
import java.io.FileWriter;
|
||||||
@@ -27,6 +22,8 @@ import org.junit.Rule;
|
|||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.junit.rules.TemporaryFolder;
|
import org.junit.rules.TemporaryFolder;
|
||||||
|
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests for JSON-Java XML.java with XMLParserConfiguration.java
|
* Tests for JSON-Java XML.java with XMLParserConfiguration.java
|
||||||
@@ -557,6 +554,37 @@ public class XMLConfigurationTest {
|
|||||||
assertEquals(actualXML, resultXML);
|
assertEquals(actualXML, resultXML);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void shouldHandleEmptyNodeValue()
|
||||||
|
{
|
||||||
|
JSONObject inputJSON = new JSONObject();
|
||||||
|
inputJSON.put("Emptyness", "");
|
||||||
|
String expectedXmlWithoutExplicitEndTag = "<Emptyness/>";
|
||||||
|
String expectedXmlWithExplicitEndTag = "<Emptyness></Emptyness>";
|
||||||
|
assertEquals(expectedXmlWithoutExplicitEndTag, XML.toString(inputJSON, null,
|
||||||
|
new XMLParserConfiguration().withCloseEmptyTag(false)));
|
||||||
|
assertEquals(expectedXmlWithExplicitEndTag, XML.toString(inputJSON, null,
|
||||||
|
new XMLParserConfiguration().withCloseEmptyTag(true)));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void shouldKeepConfigurationIntactAndUpdateCloseEmptyTagChoice()
|
||||||
|
{
|
||||||
|
XMLParserConfiguration keepStrings = XMLParserConfiguration.KEEP_STRINGS;
|
||||||
|
XMLParserConfiguration keepStringsAndCloseEmptyTag = keepStrings.withCloseEmptyTag(true);
|
||||||
|
XMLParserConfiguration keepDigits = keepStringsAndCloseEmptyTag.withKeepStrings(false);
|
||||||
|
XMLParserConfiguration keepDigitsAndNoCloseEmptyTag = keepDigits.withCloseEmptyTag(false);
|
||||||
|
assertTrue(keepStrings.isKeepStrings());
|
||||||
|
assertFalse(keepStrings.isCloseEmptyTag());
|
||||||
|
assertTrue(keepStringsAndCloseEmptyTag.isKeepStrings());
|
||||||
|
assertTrue(keepStringsAndCloseEmptyTag.isCloseEmptyTag());
|
||||||
|
assertFalse(keepDigits.isKeepStrings());
|
||||||
|
assertTrue(keepDigits.isCloseEmptyTag());
|
||||||
|
assertFalse(keepDigitsAndNoCloseEmptyTag.isKeepStrings());
|
||||||
|
assertFalse(keepDigitsAndNoCloseEmptyTag.isCloseEmptyTag());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Investigate exactly how the "content" keyword works
|
* Investigate exactly how the "content" keyword works
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ import java.io.InputStream;
|
|||||||
import java.io.InputStreamReader;
|
import java.io.InputStreamReader;
|
||||||
import java.io.Reader;
|
import java.io.Reader;
|
||||||
import java.io.StringReader;
|
import java.io.StringReader;
|
||||||
|
import java.nio.charset.Charset;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
@@ -915,7 +916,7 @@ public class XMLTest {
|
|||||||
InputStream xmlStream = null;
|
InputStream xmlStream = null;
|
||||||
try {
|
try {
|
||||||
xmlStream = XMLTest.class.getClassLoader().getResourceAsStream("Issue537.xml");
|
xmlStream = XMLTest.class.getClassLoader().getResourceAsStream("Issue537.xml");
|
||||||
Reader xmlReader = new InputStreamReader(xmlStream);
|
Reader xmlReader = new InputStreamReader(xmlStream, Charset.forName("UTF-8"));
|
||||||
JSONObject actual = XML.toJSONObject(xmlReader, true);
|
JSONObject actual = XML.toJSONObject(xmlReader, true);
|
||||||
InputStream jsonStream = null;
|
InputStream jsonStream = null;
|
||||||
try {
|
try {
|
||||||
@@ -1176,6 +1177,30 @@ public class XMLTest {
|
|||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void shouldCreateExplicitEndTagWithEmptyValueWhenConfigured(){
|
||||||
|
String jsonString = "{outer:{innerOne:\"\", innerTwo:\"two\"}}";
|
||||||
|
JSONObject jsonObject = new JSONObject(jsonString);
|
||||||
|
String expectedXmlString = "<encloser><outer><innerOne></innerOne><innerTwo>two</innerTwo></outer></encloser>";
|
||||||
|
String xmlForm = XML.toString(jsonObject,"encloser", new XMLParserConfiguration().withCloseEmptyTag(true));
|
||||||
|
JSONObject actualJsonObject = XML.toJSONObject(xmlForm);
|
||||||
|
JSONObject expectedJsonObject = XML.toJSONObject(expectedXmlString);
|
||||||
|
assertTrue(expectedJsonObject.similar(actualJsonObject));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void shouldNotCreateExplicitEndTagWithEmptyValueWhenNotConfigured(){
|
||||||
|
String jsonString = "{outer:{innerOne:\"\", innerTwo:\"two\"}}";
|
||||||
|
JSONObject jsonObject = new JSONObject(jsonString);
|
||||||
|
String expectedXmlString = "<encloser><outer><innerOne/><innerTwo>two</innerTwo></outer></encloser>";
|
||||||
|
String xmlForm = XML.toString(jsonObject,"encloser", new XMLParserConfiguration().withCloseEmptyTag(false));
|
||||||
|
JSONObject actualJsonObject = XML.toJSONObject(xmlForm);
|
||||||
|
JSONObject expectedJsonObject = XML.toJSONObject(expectedXmlString);
|
||||||
|
assertTrue(expectedJsonObject.similar(actualJsonObject));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testIndentSimpleJsonObject(){
|
public void testIndentSimpleJsonObject(){
|
||||||
String str = "{ \"employee\": { \n" +
|
String str = "{ \"employee\": { \n" +
|
||||||
@@ -1222,15 +1247,10 @@ public class XMLTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testIndentComplicatedJsonObjectWithArrayAndWithConfig(){
|
public void testIndentComplicatedJsonObjectWithArrayAndWithConfig(){
|
||||||
try {
|
try (InputStream jsonStream = XMLTest.class.getClassLoader().getResourceAsStream("Issue593.json")) {
|
||||||
InputStream jsonStream = null;
|
|
||||||
try {
|
|
||||||
jsonStream = XMLTest.class.getClassLoader().getResourceAsStream("Issue593.json");
|
|
||||||
final JSONObject object = new JSONObject(new JSONTokener(jsonStream));
|
final JSONObject object = new JSONObject(new JSONTokener(jsonStream));
|
||||||
String actualString = XML.toString(object, null, XMLParserConfiguration.KEEP_STRINGS,2);
|
String actualString = XML.toString(object, null, XMLParserConfiguration.KEEP_STRINGS, 2);
|
||||||
InputStream xmlStream = null;
|
try (InputStream xmlStream = XMLTest.class.getClassLoader().getResourceAsStream("Issue593.xml")) {
|
||||||
try {
|
|
||||||
xmlStream = XMLTest.class.getClassLoader().getResourceAsStream("Issue593.xml");
|
|
||||||
int bufferSize = 1024;
|
int bufferSize = 1024;
|
||||||
char[] buffer = new char[bufferSize];
|
char[] buffer = new char[bufferSize];
|
||||||
StringBuilder expected = new StringBuilder();
|
StringBuilder expected = new StringBuilder();
|
||||||
@@ -1238,16 +1258,7 @@ public class XMLTest {
|
|||||||
for (int numRead; (numRead = in.read(buffer, 0, buffer.length)) > 0; ) {
|
for (int numRead; (numRead = in.read(buffer, 0, buffer.length)) > 0; ) {
|
||||||
expected.append(buffer, 0, numRead);
|
expected.append(buffer, 0, numRead);
|
||||||
}
|
}
|
||||||
assertEquals(expected.toString(), actualString.replaceAll("\\n|\\r\\n", System.getProperty("line.separator")));
|
assertTrue(XML.toJSONObject(expected.toString()).similar(XML.toJSONObject(actualString)));
|
||||||
} finally {
|
|
||||||
if (xmlStream != null) {
|
|
||||||
xmlStream.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
if (jsonStream != null) {
|
|
||||||
jsonStream.close();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
fail("file writer error: " +e.getMessage());
|
fail("file writer error: " +e.getMessage());
|
||||||
@@ -1312,6 +1323,109 @@ public class XMLTest {
|
|||||||
"parameter of the XMLParserConfiguration used");
|
"parameter of the XMLParserConfiguration used");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@Test
|
||||||
|
public void testWithWhitespaceTrimmingDisabled() {
|
||||||
|
String originalXml = "<testXml> Test Whitespace String \t </testXml>";
|
||||||
|
|
||||||
|
JSONObject actualJson = XML.toJSONObject(originalXml, new XMLParserConfiguration().withShouldTrimWhitespace(false));
|
||||||
|
String expectedJsonString = "{\"testXml\":\" Test Whitespace String \t \"}";
|
||||||
|
JSONObject expectedJson = new JSONObject(expectedJsonString);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(actualJson,expectedJson);
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testNestedWithWhitespaceTrimmingDisabled() {
|
||||||
|
String originalXml =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name> Sherlock Holmes </name>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
|
||||||
|
JSONObject actualJson = XML.toJSONObject(originalXml, new XMLParserConfiguration().withShouldTrimWhitespace(false));
|
||||||
|
String expectedJsonString = "{\"addresses\":{\"address\":{\"name\":\" Sherlock Holmes \"}}}";
|
||||||
|
JSONObject expectedJson = new JSONObject(expectedJsonString);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(actualJson,expectedJson);
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void shouldTrimWhitespaceDoesNotSupportTagsEqualingCDataTagName() {
|
||||||
|
// When using withShouldTrimWhitespace = true, input containing tags with same name as cDataTagName is unsupported and should not be used in conjunction
|
||||||
|
String originalXml =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <content> Sherlock Holmes </content>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
|
||||||
|
JSONObject actualJson = XML.toJSONObject(originalXml, new XMLParserConfiguration().withShouldTrimWhitespace(false).withcDataTagName("content"));
|
||||||
|
String expectedJsonString = "{\"addresses\":{\"address\":[[\"\\n \",\" Sherlock Holmes \",\"\\n \"]]}}";
|
||||||
|
JSONObject expectedJson = new JSONObject(expectedJsonString);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(actualJson,expectedJson);
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void shouldTrimWhitespaceEnabledDropsTagsEqualingCDataTagNameButValueRemains() {
|
||||||
|
String originalXml =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <content> Sherlock Holmes </content>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
|
||||||
|
JSONObject actualJson = XML.toJSONObject(originalXml, new XMLParserConfiguration().withShouldTrimWhitespace(true).withcDataTagName("content"));
|
||||||
|
String expectedJsonString = "{\"addresses\":{\"address\":\"Sherlock Holmes\"}}";
|
||||||
|
JSONObject expectedJson = new JSONObject(expectedJsonString);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(actualJson,expectedJson);
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testWithWhitespaceTrimmingEnabled() {
|
||||||
|
String originalXml = "<testXml> Test Whitespace String \t </testXml>";
|
||||||
|
|
||||||
|
JSONObject actualJson = XML.toJSONObject(originalXml, new XMLParserConfiguration().withShouldTrimWhitespace(true));
|
||||||
|
String expectedJsonString = "{\"testXml\":\"Test Whitespace String\"}";
|
||||||
|
JSONObject expectedJson = new JSONObject(expectedJsonString);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(actualJson,expectedJson);
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testWithWhitespaceTrimmingEnabledByDefault() {
|
||||||
|
String originalXml = "<testXml> Test Whitespace String \t </testXml>";
|
||||||
|
|
||||||
|
JSONObject actualJson = XML.toJSONObject(originalXml, new XMLParserConfiguration());
|
||||||
|
String expectedJsonString = "{\"testXml\":\"Test Whitespace String\"}";
|
||||||
|
JSONObject expectedJson = new JSONObject(expectedJsonString);
|
||||||
|
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");
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ import java.io.StringReader;
|
|||||||
* @param <T>
|
* @param <T>
|
||||||
* generic number value
|
* generic number value
|
||||||
*/
|
*/
|
||||||
public class GenericBean<T extends Number & Comparable<T>> implements MyBean {
|
public class GenericBean<T extends Number> implements MyBean {
|
||||||
/**
|
/**
|
||||||
* @param genericValue
|
* @param genericValue
|
||||||
* value to initiate with
|
* value to initiate with
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ import java.util.List;
|
|||||||
*/
|
*/
|
||||||
public class WeirdList {
|
public class WeirdList {
|
||||||
/** */
|
/** */
|
||||||
private final List<Integer> list = new ArrayList();
|
private final List<Integer> list = new ArrayList<>();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param vals
|
* @param vals
|
||||||
@@ -25,14 +25,14 @@ public class WeirdList {
|
|||||||
* @return a copy of the list
|
* @return a copy of the list
|
||||||
*/
|
*/
|
||||||
public List<Integer> get() {
|
public List<Integer> get() {
|
||||||
return new ArrayList(this.list);
|
return new ArrayList<>(this.list);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return a copy of the list
|
* @return a copy of the list
|
||||||
*/
|
*/
|
||||||
public List<Integer> getALL() {
|
public List<Integer> getALL() {
|
||||||
return new ArrayList(this.list);
|
return new ArrayList<>(this.list);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
Reference in New Issue
Block a user