mirror of
https://github.com/ethauvin/JSON-java.git
synced 2025-06-17 07:50:52 -07:00
Compare commits
669 commits
Author | SHA1 | Date | |
---|---|---|---|
0d13e56064 | |||
f1d354ce7b | |||
86e136afc9 | |||
aa0a5a7245 | |||
|
9de97438ac | ||
|
601114ee29 | ||
|
c11c006e88 | ||
|
96bf788515 | ||
|
f4261add91 | ||
|
dced1b4a44 | ||
|
cf00d2f265 | ||
|
95c6373122 | ||
|
b6ed0d4178 | ||
|
75f656fd07 | ||
|
4b84ba2f66 | ||
|
d088cf014e | ||
|
19bb6fd606 | ||
|
fee6ddb922 | ||
|
5a32114792 | ||
|
56d33b8061 | ||
|
6029dece41 | ||
|
d334b58f45 | ||
|
b4a75c7bf8 | ||
|
956bdfa5b7 | ||
|
8546e68e20 | ||
|
0832d1d873 | ||
|
2b0a8838ef | ||
|
1da2b984cd | ||
|
78901383a4 | ||
|
3737c987a2 | ||
|
8e5b516f2b | ||
|
82202dbf65 | ||
|
89d4681e41 | ||
|
1265897f4e | ||
|
74e4932cfc | ||
|
f07ddd93e5 | ||
|
08719d4b3a | ||
|
16da56eb34 | ||
|
6b6e8e85d8 | ||
|
6ecbeaa0d2 | ||
|
3ac647a16f | ||
|
6f06801296 | ||
|
dd7056cb6d | ||
|
e62d763294 | ||
|
4990c3a180 | ||
|
4b49bc94ce | ||
|
9afa90d9a9 | ||
|
065f9a94bc | ||
|
223e328161 | ||
|
18eddf75c3 | ||
|
044b035847 | ||
|
b8fd9d0082 | ||
|
5b62cf13ff | ||
|
a24db2cce2 | ||
|
67e59888a2 | ||
|
e9c27ab376 | ||
|
fb01575394 | ||
|
3e7a0b13d1 | ||
|
328e7d8944 | ||
|
2a6af29235 | ||
|
af6d3c63bd | ||
|
c3ea0249c9 | ||
|
b044b7db4d | ||
|
115f8b660b | ||
|
f63d21fd13 | ||
|
c8ae720caf | ||
|
4d451468fd | ||
|
a03a01531a | ||
|
5b845f28cf | ||
|
6dcd82a72f | ||
|
00e0e6c0a2 | ||
|
fa173fa51a | ||
|
614e8359b9 | ||
|
d5b278539e | ||
|
437ce10ee3 | ||
|
12bbe8cd9a | ||
|
cfec288fe8 | ||
|
09dddb826e | ||
|
19e9bb6c07 | ||
|
e7f7d348cd | ||
|
e699abb1c6 | ||
|
fea0aca2ab | ||
|
d0ea807884 | ||
|
0d053a000d | ||
|
e4186e072a | ||
|
4c64ad3d8a | ||
|
1a811f1ada | ||
|
3e6c0a51bd | ||
|
34cfe6df14 | ||
|
71c6dd1e34 | ||
|
30c1bd16ba | ||
|
bc347d2c19 | ||
|
a63fa03062 | ||
|
3b8b0a681c | ||
|
16225efbdd | ||
|
b8a3342eb1 | ||
|
37f5bf28e9 | ||
|
7a17ae0b3e | ||
|
dedb044f67 | ||
|
d00501eabd | ||
|
7cad4c3b26 | ||
|
05074386d3 | ||
|
a490ebdb78 | ||
|
3c1535d724 | ||
|
a6284df9c7 | ||
|
20d90bfb0b | ||
|
bfb300835f | ||
|
ca9df04539 | ||
|
3fe4a767e6 | ||
|
06e9ad280f | ||
|
2362c930d1 | ||
|
43f3f5e80b | ||
|
f4201cf318 | ||
|
770cb9c4e5 | ||
|
2a6b5bacc5 | ||
|
a509a28ed4 | ||
|
37f1f4c8ca | ||
|
1c1ef5b211 | ||
|
193a3823b5 | ||
|
74b9a60f98 | ||
|
b63b976acb | ||
|
aa5e80bc8d | ||
|
97e180444d | ||
|
d402a99fd8 | ||
|
7073bc8c47 | ||
|
fc881e2631 | ||
|
cc2ed79e57 | ||
|
61cdfefc36 | ||
|
fbad2d0017 | ||
|
15719886f7 | ||
|
ca45b02ffc | ||
|
b6efbabc32 | ||
|
9eb8c27724 | ||
|
195963357c | ||
|
28efdb4860 | ||
|
c88653ca2e | ||
|
b3068d9fe4 | ||
|
dba4afd0cf | ||
|
26160e1619 | ||
|
b7e2eee4d6 | ||
|
77d142d494 | ||
|
28e09dc493 | ||
|
dae88d7c5c | ||
|
4a4b2db8c1 | ||
|
08d93f3eb5 | ||
|
f16682bf44 | ||
|
18952b5ac0 | ||
|
bf26eba0d2 | ||
|
d2a66a4287 | ||
|
722003d479 | ||
|
936db93445 | ||
|
e0801befe5 | ||
|
ee3aa03da1 | ||
|
578a442ef7 | ||
|
dfa37a298f | ||
|
bde6ba1c52 | ||
|
52ecc89702 | ||
|
ed8745cd63 | ||
|
057e0c75ca | ||
|
cdf3cf7f81 | ||
|
af3b7dc443 | ||
|
2565abdaaa | ||
|
2713f2e2a4 | ||
|
de855c50aa | ||
|
cb61bbf720 | ||
|
4cb1ae802a | ||
|
d0e2cf41a3 | ||
|
b90bee0f22 | ||
|
2e0a8137bd | ||
|
f177c97258 | ||
|
68b262914d | ||
|
7d8353401a | ||
|
c365e2a774 | ||
|
df466db7b9 | ||
|
1acb18091a | ||
|
7fed023080 | ||
|
fefd616d73 | ||
|
44c3e321b5 | ||
|
f2f6ad3b1f | ||
|
cf411b3187 | ||
|
d9b8507e6a | ||
|
d345bc528e | ||
|
1aeadd1765 | ||
|
aa562b5ec3 | ||
|
6f238a3698 | ||
|
38d11227ee | ||
|
4dbc5ef803 | ||
|
5c80c9157d | ||
|
e94783f91b | ||
|
7bc8f41023 | ||
|
a129ebe8e4 | ||
|
49117f33dc | ||
|
0e3f23d7a1 | ||
|
641b68dd55 | ||
|
643b25140f | ||
|
3997a90d58 | ||
|
1736a60ffe | ||
|
e8b1b66888 | ||
|
974a5f7d5d | ||
|
5024f2d210 | ||
|
16baa323cf | ||
|
52845366bd | ||
|
899cf528df | ||
|
47ff774f5c | ||
|
af39376d92 | ||
|
e7e6ed9205 | ||
|
0e612ba8a4 | ||
|
971614ac8b | ||
|
3081b4bd96 | ||
|
1add1247fa | ||
|
5b2e5e7579 | ||
|
441fec7498 | ||
|
c9ae1f17d7 | ||
|
246350bbcd | ||
|
2fbe4d96cf | ||
|
3645f91b55 | ||
|
9c092753b0 | ||
|
c5e4b91fa4 | ||
|
d0f5607998 | ||
|
ad6bdd715d | ||
|
ef7a5e40be | ||
|
237bf0adb6 | ||
|
f76fbe7005 | ||
|
4f5bf16676 | ||
|
93ca7b176f | ||
|
fbd2be7431 | ||
|
757b6edb03 | ||
|
f5d3086c55 | ||
|
f2b642a1ca | ||
|
04d76b638b | ||
|
04d6e83fc2 | ||
|
849b392c01 | ||
|
a7f8ff24df | ||
|
cfe6851d8c | ||
|
1ab5260a7a | ||
|
c28a2bdf39 | ||
|
1967bee236 | ||
|
0150639119 | ||
|
382f62e781 | ||
|
bdb1163445 | ||
|
2867aaa8c8 | ||
|
0c7bd725a6 | ||
|
fcdb8671b2 | ||
|
c46774cf13 | ||
|
bd4b180f4e | ||
|
49d47e3ff2 | ||
|
a8d4e4734f | ||
|
1d040ec407 | ||
|
4865f51dd5 | ||
|
c870094f69 | ||
|
cbd8b18c4a | ||
|
c945b53308 | ||
|
c233ae709e | ||
|
95da4246a2 | ||
|
9df5d34bbe | ||
|
f12fa9ba5f | ||
|
ae1e9e2b6a | ||
|
f6ab6d7b27 | ||
|
80e2ea2a80 | ||
|
d1a5f15f0c | ||
|
82ff14ed8d | ||
|
e41972a574 | ||
|
2917104b53 | ||
|
9e0fc5e680 | ||
|
a66abf22a8 | ||
|
724fb888f7 | ||
|
f41e1d012a | ||
|
928179a1f3 | ||
|
eb806f4c14 | ||
|
5ff8b4cb08 | ||
|
5ef4f58ef1 | ||
|
413bb53b48 | ||
|
df9c27c53f | ||
|
237376eca6 | ||
|
97e3d6c7ce | ||
|
e0616a129e | ||
|
93ffca36c3 | ||
|
e477d7002b | ||
|
fb1db9341e | ||
|
adb0478f66 | ||
|
f6a00e94c7 | ||
|
f58a0f4684 | ||
|
c11e09959c | ||
|
68f92eb395 | ||
|
2b87f334d0 | ||
|
34652a8706 | ||
|
5027a283c1 | ||
|
c8563ff93d | ||
|
a2d3b59394 | ||
|
474711c4ea | ||
|
58aebaa14f | ||
|
c24be0e4ea | ||
|
88f65c5bea | ||
|
3570890be7 | ||
|
ebe69df8e4 | ||
|
59761f6f64 | ||
|
2f2cd4dfc5 | ||
|
0b1dbe9369 | ||
|
a66b97f60b | ||
|
cbd0418704 | ||
|
bbd3fd5571 | ||
|
349a209df3 | ||
|
7851e9b2e8 | ||
|
7232a95c0b | ||
|
f96f505188 | ||
|
91107e3e82 | ||
|
5779400f26 | ||
|
c400de3cfe | ||
|
4e8e24d49d | ||
|
f881b61c81 | ||
|
37582a44ad | ||
|
a418d07460 | ||
|
154cfda9aa | ||
|
8bae09f81b | ||
|
80e36eb63c | ||
|
a2c311527b | ||
|
45a7decba4 | ||
|
0c157cae75 | ||
|
8e079599c4 | ||
|
62524b531d | ||
|
3080b8beeb | ||
|
2c228ecf1a | ||
|
e57881f8fa | ||
|
efe33a1e37 | ||
|
1246e12827 | ||
|
ffcfa66d77 | ||
|
5d8ea6fa4e | ||
|
cdfdaba95b | ||
|
2307f6f85e | ||
|
6b4edbd40c | ||
|
ae77b5cd83 | ||
|
72c2b911bf | ||
|
c3ba4bdbe5 | ||
|
3890bfae52 | ||
|
abf2963bbe | ||
|
215321cd28 | ||
|
09d37e59b8 | ||
|
93704371bb | ||
|
42e0944708 | ||
|
71c1eba1e7 | ||
|
c2de224711 | ||
|
7627d40d10 | ||
|
04181fb6e2 | ||
|
239e0b7070 | ||
|
a8a71898a3 | ||
|
86e8f7b313 | ||
|
01af31718e | ||
|
1204ea9dcf | ||
|
c5deff636b | ||
|
46a1c9acf9 | ||
|
42791ab12d | ||
|
80f9e48e64 | ||
|
0e0f3f2167 | ||
|
b3abaa5b4c | ||
|
40f170b508 | ||
|
eb569b58fc | ||
|
c7fdada0fd | ||
|
16a86d73df | ||
|
dfa651e777 | ||
|
51bcbebaa8 | ||
|
612dafc750 | ||
|
808320801a | ||
|
b2bde1f468 | ||
|
a1893ebc02 | ||
|
56be31e7a8 | ||
|
8ed0362683 | ||
|
15f48a0500 | ||
|
9a81b40334 | ||
|
45cbc66f5b | ||
|
4a458a9f1c | ||
|
06ae87c456 | ||
|
7038ea884e | ||
|
0112d82755 | ||
|
46044bfc51 | ||
|
8a72509e6e | ||
|
a9ff159c78 | ||
|
b843d67a92 | ||
|
691734f342 | ||
|
ccc7a7af29 | ||
|
adb3118d31 | ||
|
c044eb14dd | ||
|
2eed4be5fc | ||
|
5ae6a66e38 | ||
|
6edc093803 | ||
|
d833c2d8de | ||
|
f21ffbe189 | ||
|
cad23423ab | ||
|
ebf08f5651 | ||
|
77d0873abd | ||
|
f239dc75ad | ||
|
052ce94a34 | ||
|
4a3565afb3 | ||
|
1ca8933a8f | ||
|
cbb1546c53 | ||
|
c1a789a70c | ||
|
bff352791c | ||
|
5223aadd67 | ||
|
45bd72c15d | ||
|
792c6f6a9c | ||
|
e748c60eb1 | ||
|
f857dda5d8 | ||
|
66f740eb45 | ||
|
6211384f87 | ||
|
5bee7e3b45 | ||
|
e00191798e | ||
|
9c47ba299d | ||
|
612e41950c | ||
|
86cbfbc864 | ||
|
25a87975be | ||
|
2f3af56535 | ||
|
26477f4e76 | ||
|
f024b52108 | ||
|
2657915293 | ||
|
07a0f60b6e | ||
|
60349ece54 | ||
|
8f16e065c5 | ||
|
ba2585fe6c | ||
|
93c79ca566 | ||
|
62486fdea4 | ||
|
97f1b2744f | ||
|
c2b3f2bdb1 | ||
|
3007fc8ebe | ||
|
974c09b22a | ||
|
bd958e0830 | ||
|
67d888e9be | ||
|
39b1c0cb66 | ||
|
07b2d65e30 | ||
|
5b67330b71 | ||
|
d3b197b1b9 | ||
|
706d898648 | ||
|
ace08f1944 | ||
|
9950350f12 | ||
|
2b2fac3eb1 | ||
|
e2a0bb16a2 | ||
|
280ce71285 | ||
|
a971736f5b | ||
|
147343c51b | ||
|
757fd566ab | ||
|
e6f5047742 | ||
|
dc8c9e382d | ||
|
871a3e46d7 | ||
|
4df6984233 | ||
|
c88d06eede | ||
|
0dc886d11b | ||
|
fc318a765c | ||
|
0990f340db | ||
|
4ddd6a19a7 | ||
|
7f83a51718 | ||
|
95cf86688d | ||
|
91fcd6092f | ||
|
54cd97ded1 | ||
|
633ab108e7 | ||
|
472439546a | ||
|
cfec741090 | ||
|
abe421e6bb | ||
|
48c872f66f | ||
|
7187006bae | ||
|
d329b6514c | ||
|
a5390a0685 | ||
|
c6204a9f01 | ||
|
8688494876 | ||
|
48302592cf | ||
|
03dd662e72 | ||
|
23cf659730 | ||
|
ec8f649467 | ||
|
39e3ccc671 | ||
|
2ec538f420 | ||
|
cadba9400c | ||
|
c578216844 | ||
|
38cbc31624 | ||
|
6406c7a379 | ||
|
3850b5fd25 | ||
|
105426b53f | ||
|
7886c96204 | ||
|
91c6f09be8 | ||
|
0dbd9be0f1 | ||
|
4a2f9b8cd3 | ||
|
cb63a968fa | ||
|
4b0db65877 | ||
|
1a5718dc39 | ||
|
3f78a85908 | ||
|
9dd0ca7b81 | ||
|
045fc74688 | ||
|
1f6e07c914 | ||
|
8f71e01ae3 | ||
|
ccbec8127c | ||
|
1081ae092b | ||
|
58d72fe20f | ||
|
8e48caeb3d | ||
|
4a3bbedc32 | ||
|
41bfdad91f | ||
|
6f5bcb32e5 | ||
|
9ce62b9540 | ||
|
f66cc8d5c4 | ||
|
a0108f3e8e | ||
|
ee0a53d494 | ||
|
f69466f4c2 | ||
|
1f4e836863 | ||
|
8ac8c34e9f | ||
|
ec7eb25565 | ||
|
4e38ed01e5 | ||
|
752f66746b | ||
|
0361cc58d6 | ||
|
b39c3df766 | ||
|
b9c6f335ee | ||
|
f48b6439f6 | ||
|
2c026eb5f8 | ||
|
0d2d0e3f6c | ||
|
5eadebb797 | ||
|
16fa69c0f6 | ||
|
b06182cb73 | ||
|
0056b1af94 | ||
|
6cca292020 | ||
|
ab143af146 | ||
|
27b22b4724 | ||
|
7ed1f78f5f | ||
|
6dd85ad5b6 | ||
|
99927c5516 | ||
|
3de0a0a70e | ||
|
e056fc0881 | ||
|
355e832337 | ||
|
6c48db010f | ||
|
8c1a0c47b7 | ||
|
d27bf852e6 | ||
|
c5173e7cc3 | ||
|
9a9973c9ca | ||
|
0640856462 | ||
|
cb7b602f35 | ||
|
44f98e6a13 | ||
|
994a19b831 | ||
|
56aa2f8607 | ||
|
9cf532828d | ||
|
a5b00a5244 | ||
|
969e2d4fd5 | ||
|
f6bdc908d8 | ||
|
67a0c734b6 | ||
|
86f4bda2d4 | ||
|
6b03f1bbe7 | ||
|
32ea7e0ba3 | ||
|
e9ea5ca98f | ||
|
fa79826f0c | ||
|
88756c0490 | ||
|
dc7c59b23b | ||
|
60e84bff92 | ||
|
c72ac516a0 | ||
|
327c0e177e | ||
|
a9dd8e7b1d | ||
|
481ecd7964 | ||
|
f2ef541c2d | ||
|
0a995318e7 | ||
|
fb36918d85 | ||
|
fcb8048038 | ||
|
49d4985828 | ||
|
ad440b4f11 | ||
|
95b8cd5b03 | ||
|
fbd07da05a | ||
|
6195bd248b | ||
|
43396cf603 | ||
|
31614fe826 | ||
|
31cadbd810 | ||
|
912350ec75 | ||
|
65ae3e663f | ||
|
f115d6a3b8 | ||
|
ee06176ab2 | ||
|
da2e548f56 | ||
|
d2e19cc78c | ||
|
571b1a79bb | ||
|
3112e32089 | ||
|
102d273be3 | ||
|
15b5306f48 | ||
|
9f500c242c | ||
|
9bca907208 | ||
|
890fd4a397 | ||
|
30c86811c0 | ||
|
5acbee2719 | ||
|
35a4fefd2e | ||
|
f0d175c5b2 | ||
|
9df929963f | ||
|
d9e04ec5b6 | ||
|
9e78cfc48d | ||
|
0409c9bfb2 | ||
|
d75a96ae59 | ||
|
e5c01e4ff8 | ||
|
98cdaf6a39 | ||
|
67fbfa12ea | ||
|
9bffd5b3ca | ||
|
51e8a2d1e5 | ||
|
76c30539cc | ||
|
46a43b7b0d | ||
|
6a2c974581 | ||
|
a857869522 | ||
|
9835285f0c | ||
|
c4d9a9c5f9 | ||
|
dcaf5fa23a | ||
|
2784c614d4 | ||
|
bef37079dc | ||
|
a9bce1d6b2 | ||
|
22d5fd3aed | ||
|
2219b5919b | ||
|
8168e6f52a | ||
|
b2e0a77ae0 | ||
|
2db11cd4db | ||
|
03d1f0af72 | ||
|
702a918271 | ||
|
76cb83643d | ||
|
400bbd7fbf | ||
|
837b7c3037 | ||
|
f7b51414b8 | ||
|
83ac581f3d | ||
|
970e7a45a9 | ||
|
519c21c8b0 | ||
|
3406acd0aa | ||
|
ac8b1b098a | ||
|
5ee4a3fc12 | ||
|
5fc222229f | ||
|
e80ded6ebe | ||
|
45d7503e24 | ||
|
de13c7de86 | ||
|
9a6215c3be | ||
|
ef7e0c7d08 | ||
|
bc07b5196b | ||
|
1ff945de69 | ||
|
210bb41ba1 | ||
|
89f359e4f8 | ||
|
a18e9f7a25 | ||
|
2df27fc6e7 | ||
|
4c6da0e6f9 | ||
|
4fbe651e57 | ||
|
8b9c3cbf47 | ||
|
964cb540fb | ||
|
2876b27ec5 | ||
|
407e7fb8d6 | ||
|
30c31db095 | ||
|
231c3de79e | ||
|
d0223c2d08 | ||
|
25596c9578 | ||
|
e4ef254d8f | ||
|
e4ebd8c443 | ||
|
f8c37b1e05 | ||
|
e150039261 | ||
|
8374382947 | ||
|
2b0a9cc7c8 | ||
|
42ad66b032 | ||
|
09b5562ad9 | ||
|
b0ce7f3bd9 | ||
|
1bde00ed4c | ||
|
dca3726bf1 | ||
|
01a2fad7fb | ||
|
68e01afd7e | ||
|
441a00afc9 | ||
|
b5b4961a9d | ||
|
fd56452f6b | ||
|
8d9783d574 | ||
|
e899a2970d | ||
|
fac377fc13 | ||
|
ab08db4ad6 | ||
|
b557180bca | ||
|
d75ad1f0c0 | ||
|
df68a7b593 | ||
|
3d3325aaca | ||
|
d613203eec | ||
|
4a5809910b | ||
|
a859e4f547 | ||
|
5fc0c4e1df | ||
|
6c5e25dcb9 | ||
|
5b56b57074 | ||
|
03192b0162 | ||
|
7559b574dd | ||
|
03bb6d08be |
75 changed files with 16439 additions and 1884 deletions
74
.github/workflows/pipeline.yml
vendored
Normal file
74
.github/workflows/pipeline.yml
vendored
Normal file
|
@ -0,0 +1,74 @@
|
||||||
|
# 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
|
||||||
|
|
||||||
|
name: Java CI with Maven
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
# branches: [ master ]
|
||||||
|
pull_request:
|
||||||
|
branches: [ master ]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
# old-school build and jar method. No tests run or compiled.
|
||||||
|
build-1_6:
|
||||||
|
runs-on: ubuntu-16.04
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
# build for java 1.6, however don't run any tests
|
||||||
|
java: [ 1.6 ]
|
||||||
|
name: Java ${{ matrix.java }}
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- name: Setup java
|
||||||
|
uses: actions/setup-java@v1
|
||||||
|
with:
|
||||||
|
java-version: ${{ matrix.java }}
|
||||||
|
- name: Compile Java ${{ matrix.java }}
|
||||||
|
run: |
|
||||||
|
mkdir -p target/classes
|
||||||
|
javac -d target/classes/ src/main/java/org/json/*.java
|
||||||
|
- name: Create java ${{ matrix.java }} JAR
|
||||||
|
run: |
|
||||||
|
jar cvf target/org.json.jar -C target/classes .
|
||||||
|
- name: Upload Java ${{ matrix.java }} JAR
|
||||||
|
uses: actions/upload-artifact@v1
|
||||||
|
with:
|
||||||
|
name: Java ${{ matrix.java }} JAR
|
||||||
|
path: target/org.json.jar
|
||||||
|
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-16.04
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
# build against supported Java LTS versions:
|
||||||
|
java: [ 1.7, 8, 11 ]
|
||||||
|
name: Java ${{ matrix.java }}
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- name: Setup java
|
||||||
|
uses: actions/setup-java@v1
|
||||||
|
with:
|
||||||
|
java-version: ${{ 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
|
||||||
|
- name: Run Tests ${{ matrix.java }}
|
||||||
|
run: |
|
||||||
|
mvn test -Dmaven.compiler.source=${{ matrix.java }} -Dmaven.compiler.target=${{ matrix.java }}
|
||||||
|
- name: Build Test Report ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
run: |
|
||||||
|
mvn surefire-report:report-only -Dmaven.compiler.source=${{ matrix.java }} -Dmaven.compiler.target=${{ matrix.java }}
|
||||||
|
mvn site -DgenerateReports=false -Dmaven.compiler.source=${{ matrix.java }} -Dmaven.compiler.target=${{ matrix.java }}
|
||||||
|
- name: Upload Test Results ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
uses: actions/upload-artifact@v1
|
||||||
|
with:
|
||||||
|
name: Test Results ${{ matrix.java }}
|
||||||
|
path: target/surefire-reports/
|
||||||
|
- name: Upload Test Report ${{ matrix.java }}
|
||||||
|
if: ${{ always() }}
|
||||||
|
uses: actions/upload-artifact@v1
|
||||||
|
with:
|
||||||
|
name: Test Report ${{ matrix.java }}
|
||||||
|
path: target/site/
|
16
.gitignore
vendored
Normal file
16
.gitignore
vendored
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
# ignore eclipse project files
|
||||||
|
.project
|
||||||
|
.classpath
|
||||||
|
# ignore Intellij Idea project files
|
||||||
|
.idea
|
||||||
|
*.iml
|
||||||
|
/target/
|
||||||
|
|
||||||
|
/bin/
|
||||||
|
build
|
||||||
|
.settings/
|
||||||
|
/.gradle/
|
||||||
|
/gradle/
|
||||||
|
/gradlew
|
||||||
|
/gradlew.bat
|
||||||
|
.gitmodules
|
|
@ -1,45 +0,0 @@
|
||||||
package org.json;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The JSONException is thrown by the JSON.org classes when things are amiss.
|
|
||||||
*
|
|
||||||
* @author JSON.org
|
|
||||||
* @version 2015-10-18
|
|
||||||
*/
|
|
||||||
public class JSONException extends RuntimeException {
|
|
||||||
/** Serialization ID */
|
|
||||||
private static final long serialVersionUID = 0;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Constructs a JSONException with an explanatory message.
|
|
||||||
*
|
|
||||||
* @param message
|
|
||||||
* Detail about the reason for the exception.
|
|
||||||
*/
|
|
||||||
public JSONException(final String message) {
|
|
||||||
super(message);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Constructs a JSONException with an explanatory message and cause.
|
|
||||||
*
|
|
||||||
* @param message
|
|
||||||
* Detail about the reason for the exception.
|
|
||||||
* @param cause
|
|
||||||
* The cause.
|
|
||||||
*/
|
|
||||||
public JSONException(final String message, final Throwable cause) {
|
|
||||||
super(message, cause);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Constructs a new JSONException with the specified cause.
|
|
||||||
*
|
|
||||||
* @param cause
|
|
||||||
* The cause.
|
|
||||||
*/
|
|
||||||
public JSONException(final Throwable cause) {
|
|
||||||
super(cause.getMessage(), cause);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,18 +0,0 @@
|
||||||
package org.json;
|
|
||||||
/**
|
|
||||||
* The <code>JSONString</code> interface allows a <code>toJSONString()</code>
|
|
||||||
* method so that a class can change the behavior of
|
|
||||||
* <code>JSONObject.toString()</code>, <code>JSONArray.toString()</code>,
|
|
||||||
* and <code>JSONWriter.value(</code>Object<code>)</code>. The
|
|
||||||
* <code>toJSONString</code> method will be used instead of the default behavior
|
|
||||||
* of using the Object's <code>toString()</code> method and quoting the result.
|
|
||||||
*/
|
|
||||||
public interface JSONString {
|
|
||||||
/**
|
|
||||||
* The <code>toJSONString</code> method allows a class to produce its own JSON
|
|
||||||
* serialization.
|
|
||||||
*
|
|
||||||
* @return A strictly syntactically correct JSON text.
|
|
||||||
*/
|
|
||||||
public String toJSONString();
|
|
||||||
}
|
|
23
LICENSE
Normal file
23
LICENSE
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
|
||||||
|
Copyright (c) 2002 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
|
74
README
74
README
|
@ -1,74 +0,0 @@
|
||||||
JSON in Java [package org.json]
|
|
||||||
|
|
||||||
JSON is a light-weight, language independent, data interchange format.
|
|
||||||
See http://www.JSON.org/
|
|
||||||
|
|
||||||
The files in this package implement JSON encoders/decoders in Java.
|
|
||||||
It also includes the capability to convert between JSON and XML, HTTP
|
|
||||||
headers, Cookies, and CDL.
|
|
||||||
|
|
||||||
This is a reference implementation. There is a large number of JSON packages
|
|
||||||
in Java. Perhaps someday the Java community will standardize on one. Until
|
|
||||||
then, choose carefully.
|
|
||||||
|
|
||||||
The license includes this restriction: "The software shall be used for good,
|
|
||||||
not evil." If your conscience cannot live with that, then choose a different
|
|
||||||
package.
|
|
||||||
|
|
||||||
The package compiles on Java 1.8.
|
|
||||||
|
|
||||||
|
|
||||||
JSONObject.java: The JSONObject can parse text from a String or a JSONTokener
|
|
||||||
to produce a map-like object. The object provides methods for manipulating its
|
|
||||||
contents, and for producing a JSON compliant object serialization.
|
|
||||||
|
|
||||||
JSONArray.java: The JSONObject can parse text from a String or a JSONTokener
|
|
||||||
to produce a vector-like object. The object provides methods for manipulating
|
|
||||||
its contents, and for producing a JSON compliant array serialization.
|
|
||||||
|
|
||||||
JSONTokener.java: The JSONTokener breaks a text into a sequence of individual
|
|
||||||
tokens. It can be constructed from a String, Reader, or InputStream.
|
|
||||||
|
|
||||||
JSONException.java: The JSONException is the standard exception type thrown
|
|
||||||
by this package.
|
|
||||||
|
|
||||||
|
|
||||||
JSONString.java: The JSONString interface requires a toJSONString method,
|
|
||||||
allowing an object to provide its own serialization.
|
|
||||||
|
|
||||||
JSONStringer.java: The JSONStringer provides a convenient facility for
|
|
||||||
building JSON strings.
|
|
||||||
|
|
||||||
JSONWriter.java: The JSONWriter provides a convenient facility for building
|
|
||||||
JSON text through a writer.
|
|
||||||
|
|
||||||
|
|
||||||
CDL.java: CDL provides support for converting between JSON and comma
|
|
||||||
delimited lists.
|
|
||||||
|
|
||||||
Cookie.java: Cookie provides support for converting between JSON and cookies.
|
|
||||||
|
|
||||||
CookieList.java: CookieList provides support for converting between JSON and
|
|
||||||
cookie lists.
|
|
||||||
|
|
||||||
HTTP.java: HTTP provides support for converting between JSON and HTTP headers.
|
|
||||||
|
|
||||||
HTTPTokener.java: HTTPTokener extends JSONTokener for parsing HTTP headers.
|
|
||||||
|
|
||||||
XML.java: XML provides support for converting between JSON and XML.
|
|
||||||
|
|
||||||
JSONML.java: JSONML provides support for converting between JSONML and XML.
|
|
||||||
|
|
||||||
XMLTokener.java: XMLTokener extends JSONTokener for parsing XML text.
|
|
||||||
|
|
||||||
Unit tests are maintained in a separate project. Contributing developers can test JSON-java pull requests with the code in this project: https://github.com/stleary/JSON-Java-unit-test
|
|
||||||
|
|
||||||
Release history:
|
|
||||||
|
|
||||||
20151123 JSONObject and JSONArray initialization with generics. Contains the
|
|
||||||
latest code as of 23 Nov, 2015.
|
|
||||||
|
|
||||||
20150729 Checkpoint for Maven central repository release. Contains the latest code as of 29 July, 2015.
|
|
||||||
|
|
||||||
JSON-java releases can be found by searching the Maven repository for groupId "org.json" and artifactId "json". For example:
|
|
||||||
https://search.maven.org/#search%7Cgav%7C1%7Cg%3A%22org.json%22%20AND%20a%3A%22json%22
|
|
207
README.md
Normal file
207
README.md
Normal file
|
@ -0,0 +1,207 @@
|
||||||
|
JSON in Java [package org.json]
|
||||||
|
===============================
|
||||||
|
|
||||||
|
[](https://mvnrepository.com/artifact/org.json/json)
|
||||||
|
|
||||||
|
**[Click here if you just want the latest release jar file.](https://repo1.maven.org/maven2/org/json/json/20200518/json-20200518.jar)**
|
||||||
|
|
||||||
|
JSON is a light-weight, language independent, data interchange format.
|
||||||
|
See http://www.JSON.org/
|
||||||
|
|
||||||
|
The files in this package implement JSON encoders/decoders in Java.
|
||||||
|
It also includes the capability to convert between JSON and XML, HTTP
|
||||||
|
headers, Cookies, and CDL.
|
||||||
|
|
||||||
|
This is a reference implementation. There is a large number of JSON packages
|
||||||
|
in Java. Perhaps someday the Java community will standardize on one. Until
|
||||||
|
then, choose carefully.
|
||||||
|
|
||||||
|
The license includes this restriction: "The software shall be used for good,
|
||||||
|
not evil." If your conscience cannot live with that, then choose a different
|
||||||
|
package.
|
||||||
|
|
||||||
|
The package compiles on Java 1.6-1.8.
|
||||||
|
|
||||||
|
# With commit [#515 Merge tests and pom and code](https://github.com/stleary/JSON-java/pull/515), the structure of the project has changed from a flat directory containing all of the Java files to a directory structure that includes unit tests. If you have difficulty using the new structure, please open an issue so we can work through it.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
**JSONObject.java**: The `JSONObject` can parse text from a `String` or a `JSONTokener`
|
||||||
|
to produce a map-like object. The object provides methods for manipulating its
|
||||||
|
contents, and for producing a JSON compliant object serialization.
|
||||||
|
|
||||||
|
**JSONArray.java**: The `JSONArray` can parse text from a String or a `JSONTokener`
|
||||||
|
to produce a vector-like object. The object provides methods for manipulating
|
||||||
|
its contents, and for producing a JSON compliant array serialization.
|
||||||
|
|
||||||
|
**JSONTokener.java**: The `JSONTokener` breaks a text into a sequence of individual
|
||||||
|
tokens. It can be constructed from a `String`, `Reader`, or `InputStream`. It also can
|
||||||
|
parse text from a `String`, `Number`, `Boolean` or `null` like `"hello"`, `42`, `true`,
|
||||||
|
`null` to produce a simple json object.
|
||||||
|
|
||||||
|
**JSONException.java**: The `JSONException` is the standard exception type thrown
|
||||||
|
by this package.
|
||||||
|
|
||||||
|
**JSONPointer.java**: Implementation of
|
||||||
|
[JSON Pointer (RFC 6901)](https://tools.ietf.org/html/rfc6901). Supports
|
||||||
|
JSON Pointers both in the form of string representation and URI fragment
|
||||||
|
representation.
|
||||||
|
|
||||||
|
**JSONPropertyIgnore.java**: Annotation class that can be used on Java Bean getter methods.
|
||||||
|
When used on a bean method that would normally be serialized into a `JSONObject`, it
|
||||||
|
overrides the getter-to-key-name logic and forces the property to be excluded from the
|
||||||
|
resulting `JSONObject`.
|
||||||
|
|
||||||
|
**JSONPropertyName.java**: Annotation class that can be used on Java Bean getter methods.
|
||||||
|
When used on a bean method that would normally be serialized into a `JSONObject`, it
|
||||||
|
overrides the getter-to-key-name logic and uses the value of the annotation. The Bean
|
||||||
|
processor will look through the class hierarchy. This means you can use the annotation on
|
||||||
|
a base class or interface and the value of the annotation will be used even if the getter
|
||||||
|
is overridden in a child class.
|
||||||
|
|
||||||
|
**JSONString.java**: The `JSONString` interface requires a `toJSONString` method,
|
||||||
|
allowing an object to provide its own serialization.
|
||||||
|
|
||||||
|
**JSONStringer.java**: The `JSONStringer` provides a convenient facility for
|
||||||
|
building JSON strings.
|
||||||
|
|
||||||
|
**JSONWriter.java**: The `JSONWriter` provides a convenient facility for building
|
||||||
|
JSON text through a writer.
|
||||||
|
|
||||||
|
|
||||||
|
**CDL.java**: `CDL` provides support for converting between JSON and comma
|
||||||
|
delimited lists.
|
||||||
|
|
||||||
|
**Cookie.java**: `Cookie` provides support for converting between JSON and cookies.
|
||||||
|
|
||||||
|
**CookieList.java**: `CookieList` provides support for converting between JSON and
|
||||||
|
cookie lists.
|
||||||
|
|
||||||
|
**HTTP.java**: `HTTP` provides support for converting between JSON and HTTP headers.
|
||||||
|
|
||||||
|
**HTTPTokener.java**: `HTTPTokener` extends `JSONTokener` for parsing HTTP headers.
|
||||||
|
|
||||||
|
**XML.java**: `XML` provides support for converting between JSON and XML.
|
||||||
|
|
||||||
|
**JSONML.java**: `JSONML` provides support for converting between JSONML and XML.
|
||||||
|
|
||||||
|
**XMLTokener.java**: `XMLTokener` extends `JSONTokener` for parsing XML text.
|
||||||
|
|
||||||
|
Unit tests are now included in the project, but require Java 1.8 at the present time. This will be fixed in a forthcoming commit.
|
||||||
|
|
||||||
|
Numeric types in this package comply with
|
||||||
|
[ECMA-404: The JSON Data Interchange Format](http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf) and
|
||||||
|
[RFC 8259: The JavaScript Object Notation (JSON) Data Interchange Format](https://tools.ietf.org/html/rfc8259#section-6).
|
||||||
|
This package fully supports `Integer`, `Long`, and `Double` Java types. Partial support
|
||||||
|
for `BigInteger` and `BigDecimal` values in `JSONObject` and `JSONArray` objects is provided
|
||||||
|
in the form of `get()`, `opt()`, and `put()` API methods.
|
||||||
|
|
||||||
|
Although 1.6 compatibility is currently supported, it is not a project goal and may be
|
||||||
|
removed in some future release.
|
||||||
|
|
||||||
|
In compliance with RFC8259 page 10 section 9, the parser is more lax with what is valid
|
||||||
|
JSON than the Generator. For Example, the tab character (U+0009) is allowed when reading
|
||||||
|
JSON Text strings, but when output by the Generator, tab is properly converted to \t in
|
||||||
|
the string. Other instances may occur where reading invalid JSON text does not cause an
|
||||||
|
error to be generated. Malformed JSON Texts such as missing end " (quote) on strings or
|
||||||
|
invalid number formats (1.2e6.3) will cause errors as such documents can not be read
|
||||||
|
reliably.
|
||||||
|
|
||||||
|
Some notible exceptions that the JSON Parser in this library accepts are:
|
||||||
|
* Unquoted keys `{ key: "value" }`
|
||||||
|
* Unquoted values `{ "key": value }`
|
||||||
|
* Unescaped literals like "tab" in string values `{ "key": "value with an unescaped tab" }`
|
||||||
|
* Numbers out of range for `Double` or `Long` are parsed as strings
|
||||||
|
|
||||||
|
Release history:
|
||||||
|
|
||||||
|
~~~
|
||||||
|
20200518 Recent commits and snapshot before project structure change
|
||||||
|
|
||||||
|
20190722 Recent commits
|
||||||
|
|
||||||
|
20180813 POM change to include Automatic-Module-Name (#431)
|
||||||
|
|
||||||
|
20180130 Recent commits
|
||||||
|
|
||||||
|
20171018 Checkpoint for recent commits.
|
||||||
|
|
||||||
|
20170516 Roll up recent commits.
|
||||||
|
|
||||||
|
20160810 Revert code that was breaking opt*() methods.
|
||||||
|
|
||||||
|
20160807 This release contains a bug in the JSONObject.opt*() and JSONArray.opt*() methods,
|
||||||
|
it is not recommended for use.
|
||||||
|
Java 1.6 compatability fixed, JSONArray.toList() and JSONObject.toMap(),
|
||||||
|
RFC4180 compatibility, JSONPointer, some exception fixes, optional XML type conversion.
|
||||||
|
Contains the latest code as of 7 Aug, 2016
|
||||||
|
|
||||||
|
20160212 Java 1.6 compatibility, OSGi bundle. Contains the latest code as of 12 Feb, 2016.
|
||||||
|
|
||||||
|
20151123 JSONObject and JSONArray initialization with generics. Contains the
|
||||||
|
latest code as of 23 Nov, 2015.
|
||||||
|
|
||||||
|
20150729 Checkpoint for Maven central repository release. Contains the latest code
|
||||||
|
as of 29 July, 2015.
|
||||||
|
~~~
|
||||||
|
|
||||||
|
|
||||||
|
JSON-java releases can be found by searching the Maven repository for groupId "org.json"
|
||||||
|
and artifactId "json". For example:
|
||||||
|
https://search.maven.org/search?q=g:org.json%20AND%20a:json&core=gav
|
||||||
|
|
||||||
|
# Unit tests
|
||||||
|
The test suite can be executed with Maven by running:
|
||||||
|
```
|
||||||
|
mvn test
|
||||||
|
```
|
||||||
|
The test suite can be executed with Gradle (6.4 or greater) by running:
|
||||||
|
```
|
||||||
|
gradle clean build test
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## Conventions
|
||||||
|
Test filenames should consist of the name of the module being tested, with the suffix "Test".
|
||||||
|
For example, <b>Cookie.java</b> is tested by <b>CookieTest.java</b>.
|
||||||
|
|
||||||
|
<b>The fundamental issues with JSON-Java testing are:</b><br>
|
||||||
|
* <b>JSONObjects</b> are unordered, making simple string comparison ineffective.
|
||||||
|
* Comparisons via **equals()** is not currently supported. Neither <b>JSONArray</b> nor <b>JSONObject</b> override <b>hashCode()</b> or <b>equals()</b>, so comparison defaults to the <b>Object</b> equals(), which is not useful.
|
||||||
|
* Access to the <b>JSONArray</b> and <b>JSONObject</b> internal containers for comparison is not currently available.
|
||||||
|
|
||||||
|
<b>General issues with unit testing are:</b><br>
|
||||||
|
* Just writing tests to make coverage goals tends to result in poor tests.
|
||||||
|
* Unit tests are a form of documentation - how a given method actually works is demonstrated by the test. So for a code reviewer or future developer looking at code a good test helps explain how a function is supposed to work according to the original author. This can be difficult if you are not the original developer.
|
||||||
|
* It is difficult to evaluate unit tests in a vacuum. You also need to see the code being tested to understand if a test is good.
|
||||||
|
* Without unit tests it is hard to feel confident about the quality of the code, especially when fixing bugs or refactoring. Good tests prevents regressions and keeps the intent of the code correct.
|
||||||
|
* If you have unit test results along with pull requests, the reviewer has an easier time understanding your code and determining if the it works as intended.
|
||||||
|
|
||||||
|
|
||||||
|
**Caveats:**
|
||||||
|
JSON-Java is Java 1.6-compatible, but JSON-Java-unit-tests requires Java 1.8. If you see this error when building JSON-Java-unit-test, make sure you have 1.8 installed, on your path, and set in JAVA_HOME:
|
||||||
|
```
|
||||||
|
Execution failed for task ':compileJava'.
|
||||||
|
> invalid flag: -parameters
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
| Resource files used in test |
|
||||||
|
| ------------- |
|
||||||
|
| EnumTest.java |
|
||||||
|
| MyBean.java |
|
||||||
|
| MyBigNumberBean.java |
|
||||||
|
| MyEnum.java |
|
||||||
|
| MyEnumClass.java |
|
||||||
|
| MyEnumField.java |
|
||||||
|
| MyJsonString.java |
|
||||||
|
| MyPublicClass.java |
|
||||||
|
| PropertyTest.java |
|
||||||
|
| JunitTestSuite.java |
|
||||||
|
| StringsResourceBundle.java |
|
||||||
|
| TestRunner.java |
|
||||||
|
| Util.java |
|
||||||
|
|
||||||
|
|
||||||
|
|
492
XML.java
492
XML.java
|
@ -1,492 +0,0 @@
|
||||||
package org.json;
|
|
||||||
|
|
||||||
/*
|
|
||||||
Copyright (c) 2015 JSON.org
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
The Software shall be used for Good, not Evil.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import java.util.Iterator;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This provides static methods to convert an XML text into a JSONObject,
|
|
||||||
* and to covert a JSONObject into an XML text.
|
|
||||||
* @author JSON.org
|
|
||||||
* @version 2015-10-18
|
|
||||||
*/
|
|
||||||
public class XML {
|
|
||||||
|
|
||||||
/** The Character '&'. */
|
|
||||||
public static final Character AMP = '&';
|
|
||||||
|
|
||||||
/** The Character '''. */
|
|
||||||
public static final Character APOS = '\'';
|
|
||||||
|
|
||||||
/** The Character '!'. */
|
|
||||||
public static final Character BANG = '!';
|
|
||||||
|
|
||||||
/** The Character '='. */
|
|
||||||
public static final Character EQ = '=';
|
|
||||||
|
|
||||||
/** The Character '>'. */
|
|
||||||
public static final Character GT = '>';
|
|
||||||
|
|
||||||
/** The Character '<'. */
|
|
||||||
public static final Character LT = '<';
|
|
||||||
|
|
||||||
/** The Character '?'. */
|
|
||||||
public static final Character QUEST = '?';
|
|
||||||
|
|
||||||
/** The Character '"'. */
|
|
||||||
public static final Character QUOT = '"';
|
|
||||||
|
|
||||||
/** The Character '/'. */
|
|
||||||
public static final Character SLASH = '/';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Replace special characters with XML escapes:
|
|
||||||
* <pre>
|
|
||||||
* & <small>(ampersand)</small> is replaced by &amp;
|
|
||||||
* < <small>(less than)</small> is replaced by &lt;
|
|
||||||
* > <small>(greater than)</small> is replaced by &gt;
|
|
||||||
* " <small>(double quote)</small> is replaced by &quot;
|
|
||||||
* </pre>
|
|
||||||
* @param string The string to be escaped.
|
|
||||||
* @return The escaped string.
|
|
||||||
*/
|
|
||||||
public static String escape(String string) {
|
|
||||||
StringBuilder sb = new StringBuilder(string.length());
|
|
||||||
for (int i = 0, length = string.length(); i < length; i++) {
|
|
||||||
char c = string.charAt(i);
|
|
||||||
switch (c) {
|
|
||||||
case '&':
|
|
||||||
sb.append("&");
|
|
||||||
break;
|
|
||||||
case '<':
|
|
||||||
sb.append("<");
|
|
||||||
break;
|
|
||||||
case '>':
|
|
||||||
sb.append(">");
|
|
||||||
break;
|
|
||||||
case '"':
|
|
||||||
sb.append(""");
|
|
||||||
break;
|
|
||||||
case '\'':
|
|
||||||
sb.append("'");
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
sb.append(c);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return sb.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Throw an exception if the string contains whitespace.
|
|
||||||
* Whitespace is not allowed in tagNames and attributes.
|
|
||||||
* @param string A string.
|
|
||||||
* @throws JSONException
|
|
||||||
*/
|
|
||||||
public static void noSpace(String string) throws JSONException {
|
|
||||||
int i, length = string.length();
|
|
||||||
if (length == 0) {
|
|
||||||
throw new JSONException("Empty string.");
|
|
||||||
}
|
|
||||||
for (i = 0; i < length; i += 1) {
|
|
||||||
if (Character.isWhitespace(string.charAt(i))) {
|
|
||||||
throw new JSONException("'" + string +
|
|
||||||
"' contains a space character.");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Scan the content following the named tag, attaching it to the context.
|
|
||||||
* @param x The XMLTokener containing the source string.
|
|
||||||
* @param context The JSONObject that will include the new material.
|
|
||||||
* @param name The tag name.
|
|
||||||
* @return true if the close tag is processed.
|
|
||||||
* @throws JSONException
|
|
||||||
*/
|
|
||||||
private static boolean parse(XMLTokener x, JSONObject context,
|
|
||||||
String name) throws JSONException {
|
|
||||||
char c;
|
|
||||||
int i;
|
|
||||||
JSONObject jsonobject = null;
|
|
||||||
String string;
|
|
||||||
String tagName;
|
|
||||||
Object token;
|
|
||||||
|
|
||||||
// Test for and skip past these forms:
|
|
||||||
// <!-- ... -->
|
|
||||||
// <! ... >
|
|
||||||
// <![ ... ]]>
|
|
||||||
// <? ... ?>
|
|
||||||
// Report errors for these forms:
|
|
||||||
// <>
|
|
||||||
// <=
|
|
||||||
// <<
|
|
||||||
|
|
||||||
token = x.nextToken();
|
|
||||||
|
|
||||||
// <!
|
|
||||||
|
|
||||||
if (token == BANG) {
|
|
||||||
c = x.next();
|
|
||||||
if (c == '-') {
|
|
||||||
if (x.next() == '-') {
|
|
||||||
x.skipPast("-->");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
x.back();
|
|
||||||
} else if (c == '[') {
|
|
||||||
token = x.nextToken();
|
|
||||||
if ("CDATA".equals(token)) {
|
|
||||||
if (x.next() == '[') {
|
|
||||||
string = x.nextCDATA();
|
|
||||||
if (string.length() > 0) {
|
|
||||||
context.accumulate("content", string);
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
throw x.syntaxError("Expected 'CDATA['");
|
|
||||||
}
|
|
||||||
i = 1;
|
|
||||||
do {
|
|
||||||
token = x.nextMeta();
|
|
||||||
if (token == null) {
|
|
||||||
throw x.syntaxError("Missing '>' after '<!'.");
|
|
||||||
} else if (token == LT) {
|
|
||||||
i += 1;
|
|
||||||
} else if (token == GT) {
|
|
||||||
i -= 1;
|
|
||||||
}
|
|
||||||
} while (i > 0);
|
|
||||||
return false;
|
|
||||||
} else if (token == QUEST) {
|
|
||||||
|
|
||||||
// <?
|
|
||||||
|
|
||||||
x.skipPast("?>");
|
|
||||||
return false;
|
|
||||||
} else if (token == SLASH) {
|
|
||||||
|
|
||||||
// Close tag </
|
|
||||||
|
|
||||||
token = x.nextToken();
|
|
||||||
if (name == null) {
|
|
||||||
throw x.syntaxError("Mismatched close tag " + token);
|
|
||||||
}
|
|
||||||
if (!token.equals(name)) {
|
|
||||||
throw x.syntaxError("Mismatched " + name + " and " + token);
|
|
||||||
}
|
|
||||||
if (x.nextToken() != GT) {
|
|
||||||
throw x.syntaxError("Misshaped close tag");
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
|
|
||||||
} else if (token instanceof Character) {
|
|
||||||
throw x.syntaxError("Misshaped tag");
|
|
||||||
|
|
||||||
// Open tag <
|
|
||||||
|
|
||||||
} else {
|
|
||||||
tagName = (String)token;
|
|
||||||
token = null;
|
|
||||||
jsonobject = new JSONObject();
|
|
||||||
for (;;) {
|
|
||||||
if (token == null) {
|
|
||||||
token = x.nextToken();
|
|
||||||
}
|
|
||||||
|
|
||||||
// attribute = value
|
|
||||||
|
|
||||||
if (token instanceof String) {
|
|
||||||
string = (String)token;
|
|
||||||
token = x.nextToken();
|
|
||||||
if (token == EQ) {
|
|
||||||
token = x.nextToken();
|
|
||||||
if (!(token instanceof String)) {
|
|
||||||
throw x.syntaxError("Missing value");
|
|
||||||
}
|
|
||||||
jsonobject.accumulate(string,
|
|
||||||
XML.stringToValue((String)token));
|
|
||||||
token = null;
|
|
||||||
} else {
|
|
||||||
jsonobject.accumulate(string, "");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Empty tag <.../>
|
|
||||||
|
|
||||||
} else if (token == SLASH) {
|
|
||||||
if (x.nextToken() != GT) {
|
|
||||||
throw x.syntaxError("Misshaped tag");
|
|
||||||
}
|
|
||||||
if (jsonobject.length() > 0) {
|
|
||||||
context.accumulate(tagName, jsonobject);
|
|
||||||
} else {
|
|
||||||
context.accumulate(tagName, "");
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
|
|
||||||
// Content, between <...> and </...>
|
|
||||||
|
|
||||||
} else if (token == GT) {
|
|
||||||
for (;;) {
|
|
||||||
token = x.nextContent();
|
|
||||||
if (token == null) {
|
|
||||||
if (tagName != null) {
|
|
||||||
throw x.syntaxError("Unclosed tag " + tagName);
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
} else if (token instanceof String) {
|
|
||||||
string = (String)token;
|
|
||||||
if (string.length() > 0) {
|
|
||||||
jsonobject.accumulate("content",
|
|
||||||
XML.stringToValue(string));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Nested element
|
|
||||||
|
|
||||||
} else if (token == LT) {
|
|
||||||
if (parse(x, jsonobject, tagName)) {
|
|
||||||
if (jsonobject.length() == 0) {
|
|
||||||
context.accumulate(tagName, "");
|
|
||||||
} else if (jsonobject.length() == 1 &&
|
|
||||||
jsonobject.opt("content") != null) {
|
|
||||||
context.accumulate(tagName,
|
|
||||||
jsonobject.opt("content"));
|
|
||||||
} else {
|
|
||||||
context.accumulate(tagName, jsonobject);
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
throw x.syntaxError("Misshaped tag");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Try to convert a string into a number, boolean, or null. If the string
|
|
||||||
* can't be converted, return the string. This is much less ambitious than
|
|
||||||
* JSONObject.stringToValue, especially because it does not attempt to
|
|
||||||
* convert plus forms, octal forms, hex forms, or E forms lacking decimal
|
|
||||||
* points.
|
|
||||||
* @param string A String.
|
|
||||||
* @return A simple JSON value.
|
|
||||||
*/
|
|
||||||
public static Object stringToValue(String string) {
|
|
||||||
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, first as a Long, and then as a
|
|
||||||
// Double. If that doesn't work, return the string.
|
|
||||||
|
|
||||||
try {
|
|
||||||
char initial = string.charAt(0);
|
|
||||||
if (initial == '-' || (initial >= '0' && initial <= '9')) {
|
|
||||||
Long value = new Long(string);
|
|
||||||
if (value.toString().equals(string)) {
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (Exception ignore) {
|
|
||||||
try {
|
|
||||||
Double value = new Double(string);
|
|
||||||
if (value.toString().equals(string)) {
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
} catch (Exception ignoreAlso) {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return string;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Convert a well-formed (but not necessarily valid) XML string into a
|
|
||||||
* JSONObject. Some information may be lost in this transformation
|
|
||||||
* because JSON is a data format and XML is a document format. XML uses
|
|
||||||
* elements, attributes, and content text, while JSON uses unordered
|
|
||||||
* collections of name/value pairs and arrays of values. JSON does not
|
|
||||||
* does not like to distinguish between elements and attributes.
|
|
||||||
* Sequences of similar elements are represented as JSONArrays. Content
|
|
||||||
* text may be placed in a "content" member. Comments, prologs, DTDs, and
|
|
||||||
* <code><[ [ ]]></code> are ignored.
|
|
||||||
* @param string The source string.
|
|
||||||
* @return A JSONObject containing the structured data from the XML string.
|
|
||||||
* @throws JSONException
|
|
||||||
*/
|
|
||||||
public static JSONObject toJSONObject(String string) throws JSONException {
|
|
||||||
JSONObject jo = new JSONObject();
|
|
||||||
XMLTokener x = new XMLTokener(string);
|
|
||||||
while (x.more() && x.skipPast("<")) {
|
|
||||||
parse(x, jo, null);
|
|
||||||
}
|
|
||||||
return jo;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Convert a JSONObject into a well-formed, element-normal XML string.
|
|
||||||
* @param object A JSONObject.
|
|
||||||
* @return A string.
|
|
||||||
* @throws JSONException
|
|
||||||
*/
|
|
||||||
public static String toString(Object object) throws JSONException {
|
|
||||||
return toString(object, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Convert a JSONObject into a well-formed, element-normal XML string.
|
|
||||||
* @param object A JSONObject.
|
|
||||||
* @param tagName The optional name of the enclosing tag.
|
|
||||||
* @return A string.
|
|
||||||
* @throws JSONException
|
|
||||||
*/
|
|
||||||
public static String toString(Object object, String tagName)
|
|
||||||
throws JSONException {
|
|
||||||
StringBuilder sb = new StringBuilder();
|
|
||||||
int i;
|
|
||||||
JSONArray ja;
|
|
||||||
JSONObject jo;
|
|
||||||
String key;
|
|
||||||
Iterator<String> keys;
|
|
||||||
int length;
|
|
||||||
String string;
|
|
||||||
Object value;
|
|
||||||
if (object instanceof JSONObject) {
|
|
||||||
|
|
||||||
// Emit <tagName>
|
|
||||||
|
|
||||||
if (tagName != null) {
|
|
||||||
sb.append('<');
|
|
||||||
sb.append(tagName);
|
|
||||||
sb.append('>');
|
|
||||||
}
|
|
||||||
|
|
||||||
// Loop thru the keys.
|
|
||||||
|
|
||||||
jo = (JSONObject)object;
|
|
||||||
keys = jo.keys();
|
|
||||||
while (keys.hasNext()) {
|
|
||||||
key = keys.next();
|
|
||||||
value = jo.opt(key);
|
|
||||||
if (value == null) {
|
|
||||||
value = "";
|
|
||||||
}
|
|
||||||
string = value instanceof String ? (String)value : null;
|
|
||||||
|
|
||||||
// Emit content in body
|
|
||||||
|
|
||||||
if ("content".equals(key)) {
|
|
||||||
if (value instanceof JSONArray) {
|
|
||||||
ja = (JSONArray)value;
|
|
||||||
length = ja.length();
|
|
||||||
for (i = 0; i < length; i += 1) {
|
|
||||||
if (i > 0) {
|
|
||||||
sb.append('\n');
|
|
||||||
}
|
|
||||||
sb.append(escape(ja.get(i).toString()));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
sb.append(escape(value.toString()));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Emit an array of similar keys
|
|
||||||
|
|
||||||
} else if (value instanceof JSONArray) {
|
|
||||||
ja = (JSONArray)value;
|
|
||||||
length = ja.length();
|
|
||||||
for (i = 0; i < length; i += 1) {
|
|
||||||
value = ja.get(i);
|
|
||||||
if (value instanceof JSONArray) {
|
|
||||||
sb.append('<');
|
|
||||||
sb.append(key);
|
|
||||||
sb.append('>');
|
|
||||||
sb.append(toString(value));
|
|
||||||
sb.append("</");
|
|
||||||
sb.append(key);
|
|
||||||
sb.append('>');
|
|
||||||
} else {
|
|
||||||
sb.append(toString(value, key));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else if ("".equals(value)) {
|
|
||||||
sb.append('<');
|
|
||||||
sb.append(key);
|
|
||||||
sb.append("/>");
|
|
||||||
|
|
||||||
// Emit a new tag <k>
|
|
||||||
|
|
||||||
} else {
|
|
||||||
sb.append(toString(value, key));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (tagName != null) {
|
|
||||||
|
|
||||||
// Emit the </tagname> close tag
|
|
||||||
|
|
||||||
sb.append("</");
|
|
||||||
sb.append(tagName);
|
|
||||||
sb.append('>');
|
|
||||||
}
|
|
||||||
return sb.toString();
|
|
||||||
|
|
||||||
// XML does not have good support for arrays. If an array appears in a place
|
|
||||||
// where XML is lacking, synthesize an <array> element.
|
|
||||||
|
|
||||||
}
|
|
||||||
if(object!=null){
|
|
||||||
if (object.getClass().isArray()) {
|
|
||||||
object = new JSONArray(object);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (object instanceof JSONArray) {
|
|
||||||
ja = (JSONArray)object;
|
|
||||||
length = ja.length();
|
|
||||||
for (i = 0; i < length; i += 1) {
|
|
||||||
sb.append(toString(ja.opt(i), tagName == null ? "array" : tagName));
|
|
||||||
}
|
|
||||||
return sb.toString();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
string = (object == null) ? "null" : escape(object.toString());
|
|
||||||
return (tagName == null) ? "\"" + string + "\"" :
|
|
||||||
(string.length() == 0) ? "<" + tagName + "/>" :
|
|
||||||
"<" + tagName + ">" + string + "</" + tagName + ">";
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
58
build.gradle
Normal file
58
build.gradle
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
/*
|
||||||
|
* This file was generated by the Gradle 'init' task.
|
||||||
|
*/
|
||||||
|
apply plugin: 'java'
|
||||||
|
apply plugin: 'eclipse'
|
||||||
|
// apply plugin: 'jacoco'
|
||||||
|
apply plugin: 'maven-publish'
|
||||||
|
|
||||||
|
//plugins {
|
||||||
|
// id 'java'
|
||||||
|
//id 'maven-publish'
|
||||||
|
// }
|
||||||
|
|
||||||
|
repositories {
|
||||||
|
mavenLocal()
|
||||||
|
maven {
|
||||||
|
url = uri('https://oss.sonatype.org/content/repositories/snapshots')
|
||||||
|
}
|
||||||
|
|
||||||
|
maven {
|
||||||
|
url = uri('http://repo.maven.apache.org/maven2')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
testImplementation 'junit:junit:4.12'
|
||||||
|
testImplementation 'com.jayway.jsonpath:json-path:2.1.0'
|
||||||
|
testImplementation 'org.mockito:mockito-core:1.9.5'
|
||||||
|
}
|
||||||
|
|
||||||
|
subprojects {
|
||||||
|
tasks.withType(Javadoc).all { enabled = false }
|
||||||
|
}
|
||||||
|
|
||||||
|
group = 'org.json'
|
||||||
|
version = 'v20200429-SNAPSHOT'
|
||||||
|
description = 'JSON in Java'
|
||||||
|
sourceCompatibility = '1.7'
|
||||||
|
|
||||||
|
configurations.all {
|
||||||
|
}
|
||||||
|
|
||||||
|
java {
|
||||||
|
withSourcesJar()
|
||||||
|
withJavadocJar()
|
||||||
|
}
|
||||||
|
|
||||||
|
publishing {
|
||||||
|
publications {
|
||||||
|
maven(MavenPublication) {
|
||||||
|
from(components.java)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
tasks.withType(JavaCompile) {
|
||||||
|
options.encoding = 'UTF-8'
|
||||||
|
}
|
BIN
gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
BIN
gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
Binary file not shown.
5
gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
5
gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
distributionBase=GRADLE_USER_HOME
|
||||||
|
distributionPath=wrapper/dists
|
||||||
|
distributionUrl=https\://services.gradle.org/distributions/gradle-6.3-bin.zip
|
||||||
|
zipStoreBase=GRADLE_USER_HOME
|
||||||
|
zipStorePath=wrapper/dists
|
183
gradlew
vendored
Normal file
183
gradlew
vendored
Normal file
|
@ -0,0 +1,183 @@
|
||||||
|
#!/usr/bin/env sh
|
||||||
|
|
||||||
|
#
|
||||||
|
# Copyright 2015 the original author or authors.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
#
|
||||||
|
|
||||||
|
##############################################################################
|
||||||
|
##
|
||||||
|
## Gradle start up script for UN*X
|
||||||
|
##
|
||||||
|
##############################################################################
|
||||||
|
|
||||||
|
# Attempt to set APP_HOME
|
||||||
|
# Resolve links: $0 may be a link
|
||||||
|
PRG="$0"
|
||||||
|
# Need this for relative symlinks.
|
||||||
|
while [ -h "$PRG" ] ; do
|
||||||
|
ls=`ls -ld "$PRG"`
|
||||||
|
link=`expr "$ls" : '.*-> \(.*\)$'`
|
||||||
|
if expr "$link" : '/.*' > /dev/null; then
|
||||||
|
PRG="$link"
|
||||||
|
else
|
||||||
|
PRG=`dirname "$PRG"`"/$link"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
SAVED="`pwd`"
|
||||||
|
cd "`dirname \"$PRG\"`/" >/dev/null
|
||||||
|
APP_HOME="`pwd -P`"
|
||||||
|
cd "$SAVED" >/dev/null
|
||||||
|
|
||||||
|
APP_NAME="Gradle"
|
||||||
|
APP_BASE_NAME=`basename "$0"`
|
||||||
|
|
||||||
|
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||||
|
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
|
||||||
|
|
||||||
|
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
||||||
|
MAX_FD="maximum"
|
||||||
|
|
||||||
|
warn () {
|
||||||
|
echo "$*"
|
||||||
|
}
|
||||||
|
|
||||||
|
die () {
|
||||||
|
echo
|
||||||
|
echo "$*"
|
||||||
|
echo
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
# OS specific support (must be 'true' or 'false').
|
||||||
|
cygwin=false
|
||||||
|
msys=false
|
||||||
|
darwin=false
|
||||||
|
nonstop=false
|
||||||
|
case "`uname`" in
|
||||||
|
CYGWIN* )
|
||||||
|
cygwin=true
|
||||||
|
;;
|
||||||
|
Darwin* )
|
||||||
|
darwin=true
|
||||||
|
;;
|
||||||
|
MINGW* )
|
||||||
|
msys=true
|
||||||
|
;;
|
||||||
|
NONSTOP* )
|
||||||
|
nonstop=true
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
|
||||||
|
|
||||||
|
# Determine the Java command to use to start the JVM.
|
||||||
|
if [ -n "$JAVA_HOME" ] ; then
|
||||||
|
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
||||||
|
# IBM's JDK on AIX uses strange locations for the executables
|
||||||
|
JAVACMD="$JAVA_HOME/jre/sh/java"
|
||||||
|
else
|
||||||
|
JAVACMD="$JAVA_HOME/bin/java"
|
||||||
|
fi
|
||||||
|
if [ ! -x "$JAVACMD" ] ; then
|
||||||
|
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
|
||||||
|
|
||||||
|
Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
location of your Java installation."
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
JAVACMD="java"
|
||||||
|
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||||
|
|
||||||
|
Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
location of your Java installation."
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Increase the maximum file descriptors if we can.
|
||||||
|
if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then
|
||||||
|
MAX_FD_LIMIT=`ulimit -H -n`
|
||||||
|
if [ $? -eq 0 ] ; then
|
||||||
|
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
|
||||||
|
MAX_FD="$MAX_FD_LIMIT"
|
||||||
|
fi
|
||||||
|
ulimit -n $MAX_FD
|
||||||
|
if [ $? -ne 0 ] ; then
|
||||||
|
warn "Could not set maximum file descriptor limit: $MAX_FD"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# For Darwin, add options to specify how the application appears in the dock
|
||||||
|
if $darwin; then
|
||||||
|
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
|
||||||
|
fi
|
||||||
|
|
||||||
|
# For Cygwin or MSYS, switch paths to Windows format before running java
|
||||||
|
if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then
|
||||||
|
APP_HOME=`cygpath --path --mixed "$APP_HOME"`
|
||||||
|
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
|
||||||
|
JAVACMD=`cygpath --unix "$JAVACMD"`
|
||||||
|
|
||||||
|
# We build the pattern for arguments to be converted via cygpath
|
||||||
|
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
|
||||||
|
SEP=""
|
||||||
|
for dir in $ROOTDIRSRAW ; do
|
||||||
|
ROOTDIRS="$ROOTDIRS$SEP$dir"
|
||||||
|
SEP="|"
|
||||||
|
done
|
||||||
|
OURCYGPATTERN="(^($ROOTDIRS))"
|
||||||
|
# Add a user-defined pattern to the cygpath arguments
|
||||||
|
if [ "$GRADLE_CYGPATTERN" != "" ] ; then
|
||||||
|
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
|
||||||
|
fi
|
||||||
|
# Now convert the arguments - kludge to limit ourselves to /bin/sh
|
||||||
|
i=0
|
||||||
|
for arg in "$@" ; do
|
||||||
|
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
|
||||||
|
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
|
||||||
|
|
||||||
|
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
|
||||||
|
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
|
||||||
|
else
|
||||||
|
eval `echo args$i`="\"$arg\""
|
||||||
|
fi
|
||||||
|
i=`expr $i + 1`
|
||||||
|
done
|
||||||
|
case $i in
|
||||||
|
0) set -- ;;
|
||||||
|
1) set -- "$args0" ;;
|
||||||
|
2) set -- "$args0" "$args1" ;;
|
||||||
|
3) set -- "$args0" "$args1" "$args2" ;;
|
||||||
|
4) set -- "$args0" "$args1" "$args2" "$args3" ;;
|
||||||
|
5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
|
||||||
|
6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
|
||||||
|
7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
|
||||||
|
8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
|
||||||
|
9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
|
||||||
|
esac
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Escape application args
|
||||||
|
save () {
|
||||||
|
for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
|
||||||
|
echo " "
|
||||||
|
}
|
||||||
|
APP_ARGS=`save "$@"`
|
||||||
|
|
||||||
|
# Collect all arguments for the java command, following the shell quoting and substitution rules
|
||||||
|
eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS"
|
||||||
|
|
||||||
|
exec "$JAVACMD" "$@"
|
103
gradlew.bat
vendored
Normal file
103
gradlew.bat
vendored
Normal file
|
@ -0,0 +1,103 @@
|
||||||
|
@rem
|
||||||
|
@rem Copyright 2015 the original author or authors.
|
||||||
|
@rem
|
||||||
|
@rem Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
@rem you may not use this file except in compliance with the License.
|
||||||
|
@rem You may obtain a copy of the License at
|
||||||
|
@rem
|
||||||
|
@rem https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
@rem
|
||||||
|
@rem Unless required by applicable law or agreed to in writing, software
|
||||||
|
@rem distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
@rem See the License for the specific language governing permissions and
|
||||||
|
@rem limitations under the License.
|
||||||
|
@rem
|
||||||
|
|
||||||
|
@if "%DEBUG%" == "" @echo off
|
||||||
|
@rem ##########################################################################
|
||||||
|
@rem
|
||||||
|
@rem Gradle startup script for Windows
|
||||||
|
@rem
|
||||||
|
@rem ##########################################################################
|
||||||
|
|
||||||
|
@rem Set local scope for the variables with windows NT shell
|
||||||
|
if "%OS%"=="Windows_NT" setlocal
|
||||||
|
|
||||||
|
set DIRNAME=%~dp0
|
||||||
|
if "%DIRNAME%" == "" set DIRNAME=.
|
||||||
|
set APP_BASE_NAME=%~n0
|
||||||
|
set APP_HOME=%DIRNAME%
|
||||||
|
|
||||||
|
@rem Resolve any "." and ".." in APP_HOME to make it shorter.
|
||||||
|
for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
|
||||||
|
|
||||||
|
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||||
|
set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
|
||||||
|
|
||||||
|
@rem Find java.exe
|
||||||
|
if defined JAVA_HOME goto findJavaFromJavaHome
|
||||||
|
|
||||||
|
set JAVA_EXE=java.exe
|
||||||
|
%JAVA_EXE% -version >NUL 2>&1
|
||||||
|
if "%ERRORLEVEL%" == "0" goto init
|
||||||
|
|
||||||
|
echo.
|
||||||
|
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||||
|
echo.
|
||||||
|
echo Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
echo location of your Java installation.
|
||||||
|
|
||||||
|
goto fail
|
||||||
|
|
||||||
|
:findJavaFromJavaHome
|
||||||
|
set JAVA_HOME=%JAVA_HOME:"=%
|
||||||
|
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
|
||||||
|
|
||||||
|
if exist "%JAVA_EXE%" goto init
|
||||||
|
|
||||||
|
echo.
|
||||||
|
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
|
||||||
|
echo.
|
||||||
|
echo Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
echo location of your Java installation.
|
||||||
|
|
||||||
|
goto fail
|
||||||
|
|
||||||
|
:init
|
||||||
|
@rem Get command-line arguments, handling Windows variants
|
||||||
|
|
||||||
|
if not "%OS%" == "Windows_NT" goto win9xME_args
|
||||||
|
|
||||||
|
:win9xME_args
|
||||||
|
@rem Slurp the command line arguments.
|
||||||
|
set CMD_LINE_ARGS=
|
||||||
|
set _SKIP=2
|
||||||
|
|
||||||
|
:win9xME_args_slurp
|
||||||
|
if "x%~1" == "x" goto execute
|
||||||
|
|
||||||
|
set CMD_LINE_ARGS=%*
|
||||||
|
|
||||||
|
:execute
|
||||||
|
@rem Setup the command line
|
||||||
|
|
||||||
|
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
|
||||||
|
|
||||||
|
@rem Execute Gradle
|
||||||
|
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
|
||||||
|
|
||||||
|
:end
|
||||||
|
@rem End local scope for the variables with windows NT shell
|
||||||
|
if "%ERRORLEVEL%"=="0" goto mainEnd
|
||||||
|
|
||||||
|
:fail
|
||||||
|
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
|
||||||
|
rem the _cmd.exe /c_ return code!
|
||||||
|
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
|
||||||
|
exit /b 1
|
||||||
|
|
||||||
|
:mainEnd
|
||||||
|
if "%OS%"=="Windows_NT" endlocal
|
||||||
|
|
||||||
|
:omega
|
193
pom.xml
Normal file
193
pom.xml
Normal file
|
@ -0,0 +1,193 @@
|
||||||
|
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||||
|
<modelVersion>4.0.0</modelVersion>
|
||||||
|
|
||||||
|
<groupId>org.json</groupId>
|
||||||
|
<artifactId>json</artifactId>
|
||||||
|
<version>v20200429-SNAPSHOT</version>
|
||||||
|
<packaging>bundle</packaging>
|
||||||
|
|
||||||
|
<name>JSON in Java</name>
|
||||||
|
<description>
|
||||||
|
JSON is a light-weight, language independent, data interchange format.
|
||||||
|
See http://www.JSON.org/
|
||||||
|
|
||||||
|
The files in this package implement JSON encoders/decoders in Java.
|
||||||
|
It also includes the capability to convert between JSON and XML, HTTP
|
||||||
|
headers, Cookies, and CDL.
|
||||||
|
|
||||||
|
This is a reference implementation. There is a large number of JSON packages
|
||||||
|
in Java. Perhaps someday the Java community will standardize on one. Until
|
||||||
|
then, choose carefully.
|
||||||
|
|
||||||
|
The license includes this restriction: "The software shall be used for good,
|
||||||
|
not evil." If your conscience cannot live with that, then choose a different
|
||||||
|
package.
|
||||||
|
</description>
|
||||||
|
<url>https://github.com/douglascrockford/JSON-java</url>
|
||||||
|
|
||||||
|
<parent>
|
||||||
|
<groupId>org.sonatype.oss</groupId>
|
||||||
|
<artifactId>oss-parent</artifactId>
|
||||||
|
<version>9</version>
|
||||||
|
</parent>
|
||||||
|
|
||||||
|
<scm>
|
||||||
|
<url>https://github.com/douglascrockford/JSON-java.git</url>
|
||||||
|
<connection>scm:git:git://github.com/douglascrockford/JSON-java.git</connection>
|
||||||
|
<developerConnection>scm:git:git@github.com:douglascrockford/JSON-java.git</developerConnection>
|
||||||
|
</scm>
|
||||||
|
|
||||||
|
<licenses>
|
||||||
|
<license>
|
||||||
|
<name>The JSON License</name>
|
||||||
|
<url>http://json.org/license.html</url>
|
||||||
|
<distribution>repo</distribution>
|
||||||
|
<comments>Copyright (c) 2002 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and
|
||||||
|
associated documentation files (the "Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the
|
||||||
|
following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all copies or substantial
|
||||||
|
portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT
|
||||||
|
LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
||||||
|
NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
||||||
|
WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
||||||
|
SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
</comments>
|
||||||
|
</license>
|
||||||
|
</licenses>
|
||||||
|
|
||||||
|
<developers>
|
||||||
|
<developer>
|
||||||
|
<name>Douglas Crockford</name>
|
||||||
|
<email>douglas@crockford.com</email>
|
||||||
|
</developer>
|
||||||
|
</developers>
|
||||||
|
|
||||||
|
<properties>
|
||||||
|
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
||||||
|
</properties>
|
||||||
|
|
||||||
|
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>junit</groupId>
|
||||||
|
<artifactId>junit</artifactId>
|
||||||
|
<version>4.12</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.jayway.jsonpath</groupId>
|
||||||
|
<artifactId>json-path</artifactId>
|
||||||
|
<version>2.1.0</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.mockito</groupId>
|
||||||
|
<artifactId>mockito-core</artifactId>
|
||||||
|
<version>1.9.5</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
</dependencies>
|
||||||
|
|
||||||
|
<build>
|
||||||
|
<plugins>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.felix</groupId>
|
||||||
|
<artifactId>maven-bundle-plugin</artifactId>
|
||||||
|
<version>3.0.1</version>
|
||||||
|
<extensions>true</extensions>
|
||||||
|
<configuration>
|
||||||
|
<instructions>
|
||||||
|
<Export-Package>
|
||||||
|
org.json
|
||||||
|
</Export-Package>
|
||||||
|
<Bundle-SymbolicName>${project.artifactId}</Bundle-SymbolicName>
|
||||||
|
</instructions>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-compiler-plugin</artifactId>
|
||||||
|
<version>2.3.2</version>
|
||||||
|
<configuration>
|
||||||
|
<source>1.7</source>
|
||||||
|
<target>1.7</target>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-source-plugin</artifactId>
|
||||||
|
<version>2.1.2</version>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<id>attach-sources</id>
|
||||||
|
<goals>
|
||||||
|
<goal>jar-no-fork</goal>
|
||||||
|
</goals>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-javadoc-plugin</artifactId>
|
||||||
|
<version>2.7</version>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<id>attach-javadocs</id>
|
||||||
|
<goals>
|
||||||
|
<goal>jar</goal>
|
||||||
|
</goals>
|
||||||
|
<configuration>
|
||||||
|
<additionalparam>-Xdoclint:none</additionalparam>
|
||||||
|
</configuration>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-gpg-plugin</artifactId>
|
||||||
|
<version>1.5</version>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<id>sign-artifacts</id>
|
||||||
|
<phase>verify</phase>
|
||||||
|
<goals>
|
||||||
|
<goal>sign</goal>
|
||||||
|
</goals>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.sonatype.plugins</groupId>
|
||||||
|
<artifactId>nexus-staging-maven-plugin</artifactId>
|
||||||
|
<version>1.6.3</version>
|
||||||
|
<extensions>true</extensions>
|
||||||
|
<configuration>
|
||||||
|
<serverId>ossrh</serverId>
|
||||||
|
<nexusUrl>https://oss.sonatype.org/</nexusUrl>
|
||||||
|
<autoReleaseAfterClose>false</autoReleaseAfterClose>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-jar-plugin</artifactId>
|
||||||
|
<version>3.2.0</version>
|
||||||
|
<configuration>
|
||||||
|
<archive>
|
||||||
|
<manifestEntries>
|
||||||
|
<Automatic-Module-Name>org.json</Automatic-Module-Name>
|
||||||
|
</manifestEntries>
|
||||||
|
</archive>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
</plugins>
|
||||||
|
</build>
|
||||||
|
</project>
|
32
CDL.java → src/main/java/org/json/CDL.java
Executable file → Normal file
32
CDL.java → src/main/java/org/json/CDL.java
Executable file → Normal file
|
@ -41,7 +41,7 @@ SOFTWARE.
|
||||||
* The names for the elements in the JSONObjects can be taken from the names
|
* The names for the elements in the JSONObjects can be taken from the names
|
||||||
* in the first row.
|
* in the first row.
|
||||||
* @author JSON.org
|
* @author JSON.org
|
||||||
* @version 2015-05-01
|
* @version 2016-05-01
|
||||||
*/
|
*/
|
||||||
public class CDL {
|
public class CDL {
|
||||||
|
|
||||||
|
@ -55,7 +55,7 @@ public class CDL {
|
||||||
private static String getValue(JSONTokener x) throws JSONException {
|
private static String getValue(JSONTokener x) throws JSONException {
|
||||||
char c;
|
char c;
|
||||||
char q;
|
char q;
|
||||||
StringBuffer sb;
|
StringBuilder sb;
|
||||||
do {
|
do {
|
||||||
c = x.next();
|
c = x.next();
|
||||||
} while (c == ' ' || c == '\t');
|
} while (c == ' ' || c == '\t');
|
||||||
|
@ -65,12 +65,20 @@ public class CDL {
|
||||||
case '"':
|
case '"':
|
||||||
case '\'':
|
case '\'':
|
||||||
q = c;
|
q = c;
|
||||||
sb = new StringBuffer();
|
sb = new StringBuilder();
|
||||||
for (;;) {
|
for (;;) {
|
||||||
c = x.next();
|
c = x.next();
|
||||||
if (c == q) {
|
if (c == q) {
|
||||||
|
//Handle escaped double-quote
|
||||||
|
char nextC = x.next();
|
||||||
|
if(nextC != '\"') {
|
||||||
|
// if our quote was the end of the file, don't step
|
||||||
|
if(nextC > 0) {
|
||||||
|
x.back();
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
if (c == 0 || c == '\n' || c == '\r') {
|
if (c == 0 || c == '\n' || c == '\r') {
|
||||||
throw x.syntaxError("Missing close quote '" + q + "'.");
|
throw x.syntaxError("Missing close quote '" + q + "'.");
|
||||||
}
|
}
|
||||||
|
@ -90,7 +98,7 @@ public class CDL {
|
||||||
* Produce a JSONArray of strings from a row of comma delimited values.
|
* Produce a JSONArray of strings from a row of comma delimited values.
|
||||||
* @param x A JSONTokener of the source text.
|
* @param x A JSONTokener of the source text.
|
||||||
* @return A JSONArray of strings.
|
* @return A JSONArray of strings.
|
||||||
* @throws JSONException
|
* @throws JSONException if a called function fails
|
||||||
*/
|
*/
|
||||||
public static JSONArray rowToJSONArray(JSONTokener x) throws JSONException {
|
public static JSONArray rowToJSONArray(JSONTokener x) throws JSONException {
|
||||||
JSONArray ja = new JSONArray();
|
JSONArray ja = new JSONArray();
|
||||||
|
@ -126,7 +134,7 @@ public class CDL {
|
||||||
* method.
|
* method.
|
||||||
* @param x A JSONTokener of the source text.
|
* @param x A JSONTokener of the source text.
|
||||||
* @return A JSONObject combining the names and values.
|
* @return A JSONObject combining the names and values.
|
||||||
* @throws JSONException
|
* @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 {
|
||||||
|
@ -176,7 +184,7 @@ public class CDL {
|
||||||
* using the first row as a source of names.
|
* using the first row as a source of names.
|
||||||
* @param string The comma delimited text.
|
* @param string The comma delimited text.
|
||||||
* @return A JSONArray of JSONObjects.
|
* @return A JSONArray of JSONObjects.
|
||||||
* @throws JSONException
|
* @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(new JSONTokener(string));
|
||||||
|
@ -187,7 +195,7 @@ public class CDL {
|
||||||
* using the first row as a source of names.
|
* using the first row as a source of names.
|
||||||
* @param x The JSONTokener containing the comma delimited text.
|
* @param x The JSONTokener containing the comma delimited text.
|
||||||
* @return A JSONArray of JSONObjects.
|
* @return A JSONArray of JSONObjects.
|
||||||
* @throws JSONException
|
* @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(rowToJSONArray(x), x);
|
||||||
|
@ -199,7 +207,7 @@ public class CDL {
|
||||||
* @param names A JSONArray of strings.
|
* @param names A JSONArray of strings.
|
||||||
* @param string The comma delimited text.
|
* @param string The comma delimited text.
|
||||||
* @return A JSONArray of JSONObjects.
|
* @return A JSONArray of JSONObjects.
|
||||||
* @throws JSONException
|
* @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 {
|
||||||
|
@ -212,7 +220,7 @@ public class CDL {
|
||||||
* @param names A JSONArray of strings.
|
* @param names A JSONArray of strings.
|
||||||
* @param x A JSONTokener of the source text.
|
* @param x A JSONTokener of the source text.
|
||||||
* @return A JSONArray of JSONObjects.
|
* @return A JSONArray of JSONObjects.
|
||||||
* @throws JSONException
|
* @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 {
|
||||||
|
@ -240,7 +248,7 @@ public class CDL {
|
||||||
* JSONObject.
|
* JSONObject.
|
||||||
* @param ja A JSONArray of JSONObjects.
|
* @param ja A JSONArray of JSONObjects.
|
||||||
* @return A comma delimited text.
|
* @return A comma delimited text.
|
||||||
* @throws JSONException
|
* @throws JSONException if a called function fails
|
||||||
*/
|
*/
|
||||||
public static String toString(JSONArray ja) throws JSONException {
|
public static String toString(JSONArray ja) throws JSONException {
|
||||||
JSONObject jo = ja.optJSONObject(0);
|
JSONObject jo = ja.optJSONObject(0);
|
||||||
|
@ -260,14 +268,14 @@ public class CDL {
|
||||||
* @param names A JSONArray of strings.
|
* @param names A JSONArray of strings.
|
||||||
* @param ja A JSONArray of JSONObjects.
|
* @param ja A JSONArray of JSONObjects.
|
||||||
* @return A comma delimited text.
|
* @return A comma delimited text.
|
||||||
* @throws JSONException
|
* @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 {
|
||||||
if (names == null || names.length() == 0) {
|
if (names == null || names.length() == 0) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
StringBuffer sb = new StringBuffer();
|
StringBuilder sb = new StringBuilder();
|
||||||
for (int i = 0; i < ja.length(); i += 1) {
|
for (int i = 0; i < ja.length(); i += 1) {
|
||||||
JSONObject jo = ja.optJSONObject(i);
|
JSONObject jo = ja.optJSONObject(i);
|
||||||
if (jo != null) {
|
if (jo != null) {
|
121
Cookie.java → src/main/java/org/json/Cookie.java
Executable file → Normal file
121
Cookie.java → src/main/java/org/json/Cookie.java
Executable file → Normal file
|
@ -1,5 +1,7 @@
|
||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
|
import java.util.Locale;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2002 JSON.org
|
Copyright (c) 2002 JSON.org
|
||||||
|
|
||||||
|
@ -27,8 +29,9 @@ SOFTWARE.
|
||||||
/**
|
/**
|
||||||
* Convert a web browser cookie specification to a JSONObject and back.
|
* Convert a web browser cookie specification to a JSONObject and back.
|
||||||
* JSON and Cookies are both notations for name/value pairs.
|
* JSON and Cookies are both notations for name/value pairs.
|
||||||
|
* See also: <a href="https://tools.ietf.org/html/rfc6265">https://tools.ietf.org/html/rfc6265</a>
|
||||||
* @author JSON.org
|
* @author JSON.org
|
||||||
* @version 2014-05-03
|
* @version 2015-12-09
|
||||||
*/
|
*/
|
||||||
public class Cookie {
|
public class Cookie {
|
||||||
|
|
||||||
|
@ -65,77 +68,129 @@ public class Cookie {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert a cookie specification string into a JSONObject. The string
|
* Convert a cookie specification string into a JSONObject. The string
|
||||||
* will contain a name value pair separated by '='. The name and the value
|
* must contain a name value pair separated by '='. The name and the value
|
||||||
* will be unescaped, possibly converting '+' and '%' sequences. The
|
* will be unescaped, possibly converting '+' and '%' sequences. The
|
||||||
* cookie properties may follow, separated by ';', also represented as
|
* cookie properties may follow, separated by ';', also represented as
|
||||||
* name=value (except the secure property, which does not have a value).
|
* name=value (except the Attribute properties like "Secure" or "HttpOnly",
|
||||||
|
* which do not have a value. The value {@link Boolean#TRUE} will be used for these).
|
||||||
* The name will be stored under the key "name", and the value will be
|
* The name will be stored under the key "name", and the value will be
|
||||||
* stored under the key "value". This method does not do checking or
|
* stored under the key "value". This method does not do checking or
|
||||||
* validation of the parameters. It only converts the cookie string into
|
* validation of the parameters. It only converts the cookie string into
|
||||||
* a JSONObject.
|
* a JSONObject. All attribute names are converted to lower case keys in the
|
||||||
|
* JSONObject (HttpOnly => httponly). If an attribute is specified more than
|
||||||
|
* once, only the value found closer to the end of the cookie-string is kept.
|
||||||
* @param string The cookie specification string.
|
* @param string The cookie specification string.
|
||||||
* @return A JSONObject containing "name", "value", and possibly other
|
* @return A JSONObject containing "name", "value", and possibly other
|
||||||
* members.
|
* members.
|
||||||
* @throws JSONException
|
* @throws JSONException If there is an error parsing the Cookie String.
|
||||||
|
* Cookie strings must have at least one '=' character and the 'name'
|
||||||
|
* portion of the cookie must not be blank.
|
||||||
*/
|
*/
|
||||||
public static JSONObject toJSONObject(String string) throws JSONException {
|
public static JSONObject toJSONObject(String string) {
|
||||||
|
final JSONObject jo = new JSONObject();
|
||||||
String name;
|
String name;
|
||||||
JSONObject jo = new JSONObject();
|
|
||||||
Object value;
|
Object value;
|
||||||
|
|
||||||
|
|
||||||
JSONTokener x = new JSONTokener(string);
|
JSONTokener x = new JSONTokener(string);
|
||||||
jo.put("name", x.nextTo('='));
|
|
||||||
|
name = unescape(x.nextTo('=').trim());
|
||||||
|
//per RFC6265, if the name is blank, the cookie should be ignored.
|
||||||
|
if("".equals(name)) {
|
||||||
|
throw new JSONException("Cookies must have a 'name'");
|
||||||
|
}
|
||||||
|
jo.put("name", name);
|
||||||
|
// per RFC6265, if there is no '=', the cookie should be ignored.
|
||||||
|
// the 'next' call here throws an exception if the '=' is not found.
|
||||||
x.next('=');
|
x.next('=');
|
||||||
jo.put("value", x.nextTo(';'));
|
jo.put("value", unescape(x.nextTo(';')).trim());
|
||||||
|
// discard the ';'
|
||||||
x.next();
|
x.next();
|
||||||
|
// parse the remaining cookie attributes
|
||||||
while (x.more()) {
|
while (x.more()) {
|
||||||
name = unescape(x.nextTo("=;"));
|
name = unescape(x.nextTo("=;")).trim().toLowerCase(Locale.ROOT);
|
||||||
|
// don't allow a cookies attributes to overwrite it's name or value.
|
||||||
|
if("name".equalsIgnoreCase(name)) {
|
||||||
|
throw new JSONException("Illegal attribute name: 'name'");
|
||||||
|
}
|
||||||
|
if("value".equalsIgnoreCase(name)) {
|
||||||
|
throw new JSONException("Illegal attribute name: 'value'");
|
||||||
|
}
|
||||||
|
// check to see if it's a flag property
|
||||||
if (x.next() != '=') {
|
if (x.next() != '=') {
|
||||||
if (name.equals("secure")) {
|
|
||||||
value = Boolean.TRUE;
|
value = Boolean.TRUE;
|
||||||
} else {
|
} else {
|
||||||
throw x.syntaxError("Missing '=' in cookie parameter.");
|
value = unescape(x.nextTo(';')).trim();
|
||||||
}
|
|
||||||
} else {
|
|
||||||
value = unescape(x.nextTo(';'));
|
|
||||||
x.next();
|
x.next();
|
||||||
}
|
}
|
||||||
|
// only store non-blank attributes
|
||||||
|
if(!"".equals(name) && !"".equals(value)) {
|
||||||
jo.put(name, value);
|
jo.put(name, value);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return jo;
|
return jo;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert a JSONObject into a cookie specification string. The JSONObject
|
* Convert a JSONObject into a cookie specification string. The JSONObject
|
||||||
* must contain "name" and "value" members.
|
* must contain "name" and "value" members (case insensitive).
|
||||||
* If the JSONObject contains "expires", "domain", "path", or "secure"
|
* If the JSONObject contains other members, they will be appended to the cookie
|
||||||
* members, they will be appended to the cookie specification string.
|
* specification string. User-Agents are instructed to ignore unknown attributes,
|
||||||
* All other members are ignored.
|
* so ensure your JSONObject is using only known attributes.
|
||||||
|
* See also: <a href="https://tools.ietf.org/html/rfc6265">https://tools.ietf.org/html/rfc6265</a>
|
||||||
* @param jo A JSONObject
|
* @param jo A JSONObject
|
||||||
* @return A cookie specification string
|
* @return A cookie specification string
|
||||||
* @throws JSONException
|
* @throws JSONException thrown if the cookie has no name.
|
||||||
*/
|
*/
|
||||||
public static String toString(JSONObject jo) throws JSONException {
|
public static String toString(JSONObject jo) throws JSONException {
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
|
|
||||||
sb.append(escape(jo.getString("name")));
|
String name = null;
|
||||||
|
Object value = null;
|
||||||
|
for(String key : jo.keySet()){
|
||||||
|
if("name".equalsIgnoreCase(key)) {
|
||||||
|
name = jo.getString(key).trim();
|
||||||
|
}
|
||||||
|
if("value".equalsIgnoreCase(key)) {
|
||||||
|
value=jo.getString(key).trim();
|
||||||
|
}
|
||||||
|
if(name != null && value != null) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(name == null || "".equals(name.trim())) {
|
||||||
|
throw new JSONException("Cookie does not have a name");
|
||||||
|
}
|
||||||
|
if(value == null) {
|
||||||
|
value = "";
|
||||||
|
}
|
||||||
|
|
||||||
|
sb.append(escape(name));
|
||||||
sb.append("=");
|
sb.append("=");
|
||||||
sb.append(escape(jo.getString("value")));
|
sb.append(escape((String)value));
|
||||||
if (jo.has("expires")) {
|
|
||||||
sb.append(";expires=");
|
for(String key : jo.keySet()){
|
||||||
sb.append(jo.getString("expires"));
|
if("name".equalsIgnoreCase(key)
|
||||||
|
|| "value".equalsIgnoreCase(key)) {
|
||||||
|
// already processed above
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
if (jo.has("domain")) {
|
value = jo.opt(key);
|
||||||
sb.append(";domain=");
|
if(value instanceof Boolean) {
|
||||||
sb.append(escape(jo.getString("domain")));
|
if(Boolean.TRUE.equals(value)) {
|
||||||
|
sb.append(';').append(escape(key));
|
||||||
}
|
}
|
||||||
if (jo.has("path")) {
|
// don't emit false values
|
||||||
sb.append(";path=");
|
} else {
|
||||||
sb.append(escape(jo.getString("path")));
|
sb.append(';')
|
||||||
|
.append(escape(key))
|
||||||
|
.append('=')
|
||||||
|
.append(escape(value.toString()));
|
||||||
}
|
}
|
||||||
if (jo.optBoolean("secure")) {
|
|
||||||
sb.append(";secure");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return sb.toString();
|
return sb.toString();
|
||||||
}
|
}
|
||||||
|
|
25
CookieList.java → src/main/java/org/json/CookieList.java
Executable file → Normal file
25
CookieList.java → src/main/java/org/json/CookieList.java
Executable file → Normal file
|
@ -24,12 +24,10 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
SOFTWARE.
|
SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.util.Iterator;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert a web browser cookie list string to a JSONObject and back.
|
* Convert a web browser cookie list string to a JSONObject and back.
|
||||||
* @author JSON.org
|
* @author JSON.org
|
||||||
* @version 2014-05-03
|
* @version 2015-12-09
|
||||||
*/
|
*/
|
||||||
public class CookieList {
|
public class CookieList {
|
||||||
|
|
||||||
|
@ -39,12 +37,12 @@ public class CookieList {
|
||||||
* The pairs are separated by ';'. The names and the values
|
* The pairs are separated by ';'. The names and the values
|
||||||
* will be unescaped, possibly converting '+' and '%' sequences.
|
* will be unescaped, possibly converting '+' and '%' sequences.
|
||||||
*
|
*
|
||||||
* To add a cookie to a cooklist,
|
* To add a cookie to a cookie list,
|
||||||
* cookielistJSONObject.put(cookieJSONObject.getString("name"),
|
* cookielistJSONObject.put(cookieJSONObject.getString("name"),
|
||||||
* cookieJSONObject.getString("value"));
|
* cookieJSONObject.getString("value"));
|
||||||
* @param string A cookie list string
|
* @param string A cookie list string
|
||||||
* @return A JSONObject
|
* @return A JSONObject
|
||||||
* @throws JSONException
|
* @throws JSONException if a called function fails
|
||||||
*/
|
*/
|
||||||
public static JSONObject toJSONObject(String string) throws JSONException {
|
public static JSONObject toJSONObject(String string) throws JSONException {
|
||||||
JSONObject jo = new JSONObject();
|
JSONObject jo = new JSONObject();
|
||||||
|
@ -65,22 +63,21 @@ public class CookieList {
|
||||||
* in the names and values are replaced by "%hh".
|
* in the names and values are replaced by "%hh".
|
||||||
* @param jo A JSONObject
|
* @param jo A JSONObject
|
||||||
* @return A cookie list string
|
* @return A cookie list string
|
||||||
* @throws JSONException
|
* @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 b = false;
|
||||||
Iterator<String> keys = jo.keys();
|
final StringBuilder sb = new StringBuilder();
|
||||||
String string;
|
// Don't use the new entrySet API to maintain Android support
|
||||||
StringBuilder sb = new StringBuilder();
|
for (final String key : jo.keySet()) {
|
||||||
while (keys.hasNext()) {
|
final Object value = jo.opt(key);
|
||||||
string = keys.next();
|
if (!JSONObject.NULL.equals(value)) {
|
||||||
if (!jo.isNull(string)) {
|
|
||||||
if (b) {
|
if (b) {
|
||||||
sb.append(';');
|
sb.append(';');
|
||||||
}
|
}
|
||||||
sb.append(Cookie.escape(string));
|
sb.append(Cookie.escape(key));
|
||||||
sb.append("=");
|
sb.append("=");
|
||||||
sb.append(Cookie.escape(jo.getString(string)));
|
sb.append(Cookie.escape(value.toString()));
|
||||||
b = true;
|
b = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
31
HTTP.java → src/main/java/org/json/HTTP.java
Executable file → Normal file
31
HTTP.java → src/main/java/org/json/HTTP.java
Executable file → Normal file
|
@ -24,12 +24,12 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
SOFTWARE.
|
SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.util.Iterator;
|
import java.util.Locale;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert an HTTP header to a JSONObject and back.
|
* Convert an HTTP header to a JSONObject and back.
|
||||||
* @author JSON.org
|
* @author JSON.org
|
||||||
* @version 2014-05-03
|
* @version 2015-12-09
|
||||||
*/
|
*/
|
||||||
public class HTTP {
|
public class HTTP {
|
||||||
|
|
||||||
|
@ -51,12 +51,12 @@ public class HTTP {
|
||||||
* "Reason-Phrase": "OK" (for example)
|
* "Reason-Phrase": "OK" (for example)
|
||||||
* }</pre>
|
* }</pre>
|
||||||
* In addition, the other parameters in the header will be captured, using
|
* In addition, the other parameters in the header will be captured, using
|
||||||
* the HTTP field names as JSON names, so that <pre>
|
* the HTTP field names as JSON names, so that <pre>{@code
|
||||||
* Date: Sun, 26 May 2002 18:06:04 GMT
|
* Date: Sun, 26 May 2002 18:06:04 GMT
|
||||||
* Cookie: Q=q2=PPEAsg--; B=677gi6ouf29bn&b=2&f=s
|
* Cookie: Q=q2=PPEAsg--; B=677gi6ouf29bn&b=2&f=s
|
||||||
* Cache-Control: no-cache</pre>
|
* Cache-Control: no-cache}</pre>
|
||||||
* become
|
* become
|
||||||
* <pre>{...
|
* <pre>{@code
|
||||||
* Date: "Sun, 26 May 2002 18:06:04 GMT",
|
* Date: "Sun, 26 May 2002 18:06:04 GMT",
|
||||||
* Cookie: "Q=q2=PPEAsg--; B=677gi6ouf29bn&b=2&f=s",
|
* Cookie: "Q=q2=PPEAsg--; B=677gi6ouf29bn&b=2&f=s",
|
||||||
* "Cache-Control": "no-cache",
|
* "Cache-Control": "no-cache",
|
||||||
|
@ -66,7 +66,7 @@ public class HTTP {
|
||||||
* @param string An HTTP header string.
|
* @param string An HTTP header string.
|
||||||
* @return A JSONObject containing the elements and attributes
|
* @return A JSONObject containing the elements and attributes
|
||||||
* of the XML string.
|
* of the XML string.
|
||||||
* @throws JSONException
|
* @throws JSONException if a called function fails
|
||||||
*/
|
*/
|
||||||
public static JSONObject toJSONObject(String string) throws JSONException {
|
public static JSONObject toJSONObject(String string) throws JSONException {
|
||||||
JSONObject jo = new JSONObject();
|
JSONObject jo = new JSONObject();
|
||||||
|
@ -74,7 +74,7 @@ public class HTTP {
|
||||||
String token;
|
String token;
|
||||||
|
|
||||||
token = x.nextToken();
|
token = x.nextToken();
|
||||||
if (token.toUpperCase().startsWith("HTTP")) {
|
if (token.toUpperCase(Locale.ROOT).startsWith("HTTP")) {
|
||||||
|
|
||||||
// Response
|
// Response
|
||||||
|
|
||||||
|
@ -125,8 +125,6 @@ public class HTTP {
|
||||||
* information.
|
* information.
|
||||||
*/
|
*/
|
||||||
public static String toString(JSONObject jo) throws JSONException {
|
public static String toString(JSONObject jo) throws JSONException {
|
||||||
Iterator<String> keys = jo.keys();
|
|
||||||
String string;
|
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
if (jo.has("Status-Code") && jo.has("Reason-Phrase")) {
|
if (jo.has("Status-Code") && jo.has("Reason-Phrase")) {
|
||||||
sb.append(jo.getString("HTTP-Version"));
|
sb.append(jo.getString("HTTP-Version"));
|
||||||
|
@ -146,14 +144,15 @@ public class HTTP {
|
||||||
throw new JSONException("Not enough material for an HTTP header.");
|
throw new JSONException("Not enough material for an HTTP header.");
|
||||||
}
|
}
|
||||||
sb.append(CRLF);
|
sb.append(CRLF);
|
||||||
while (keys.hasNext()) {
|
// Don't use the new entrySet API to maintain Android support
|
||||||
string = keys.next();
|
for (final String key : jo.keySet()) {
|
||||||
if (!"HTTP-Version".equals(string) && !"Status-Code".equals(string) &&
|
String value = jo.optString(key);
|
||||||
!"Reason-Phrase".equals(string) && !"Method".equals(string) &&
|
if (!"HTTP-Version".equals(key) && !"Status-Code".equals(key) &&
|
||||||
!"Request-URI".equals(string) && !jo.isNull(string)) {
|
!"Reason-Phrase".equals(key) && !"Method".equals(key) &&
|
||||||
sb.append(string);
|
!"Request-URI".equals(key) && !JSONObject.NULL.equals(value)) {
|
||||||
|
sb.append(key);
|
||||||
sb.append(": ");
|
sb.append(": ");
|
||||||
sb.append(jo.getString(string));
|
sb.append(jo.optString(key));
|
||||||
sb.append(CRLF);
|
sb.append(CRLF);
|
||||||
}
|
}
|
||||||
}
|
}
|
4
HTTPTokener.java → src/main/java/org/json/HTTPTokener.java
Executable file → Normal file
4
HTTPTokener.java → src/main/java/org/json/HTTPTokener.java
Executable file → Normal file
|
@ -28,7 +28,7 @@ SOFTWARE.
|
||||||
* The HTTPTokener extends the JSONTokener to provide additional methods
|
* The HTTPTokener extends the JSONTokener to provide additional methods
|
||||||
* for the parsing of HTTP headers.
|
* for the parsing of HTTP headers.
|
||||||
* @author JSON.org
|
* @author JSON.org
|
||||||
* @version 2014-05-03
|
* @version 2015-12-09
|
||||||
*/
|
*/
|
||||||
public class HTTPTokener extends JSONTokener {
|
public class HTTPTokener extends JSONTokener {
|
||||||
|
|
||||||
|
@ -43,8 +43,8 @@ public class HTTPTokener extends JSONTokener {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the next token or string. This is used in parsing HTTP headers.
|
* Get the next token or string. This is used in parsing HTTP headers.
|
||||||
* @throws JSONException
|
|
||||||
* @return A String.
|
* @return A String.
|
||||||
|
* @throws JSONException if a syntax error occurs
|
||||||
*/
|
*/
|
||||||
public String nextToken() throws JSONException {
|
public String nextToken() throws JSONException {
|
||||||
char c;
|
char c;
|
File diff suppressed because it is too large
Load diff
69
src/main/java/org/json/JSONException.java
Normal file
69
src/main/java/org/json/JSONException.java
Normal file
|
@ -0,0 +1,69 @@
|
||||||
|
package org.json;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2002 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The JSONException is thrown by the JSON.org classes when things are amiss.
|
||||||
|
*
|
||||||
|
* @author JSON.org
|
||||||
|
* @version 2015-12-09
|
||||||
|
*/
|
||||||
|
public class JSONException extends RuntimeException {
|
||||||
|
/** Serialization ID */
|
||||||
|
private static final long serialVersionUID = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a JSONException with an explanatory message.
|
||||||
|
*
|
||||||
|
* @param message
|
||||||
|
* Detail about the reason for the exception.
|
||||||
|
*/
|
||||||
|
public JSONException(final String message) {
|
||||||
|
super(message);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a JSONException with an explanatory message and cause.
|
||||||
|
*
|
||||||
|
* @param message
|
||||||
|
* Detail about the reason for the exception.
|
||||||
|
* @param cause
|
||||||
|
* The cause.
|
||||||
|
*/
|
||||||
|
public JSONException(final String message, final Throwable cause) {
|
||||||
|
super(message, cause);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new JSONException with the specified cause.
|
||||||
|
*
|
||||||
|
* @param cause
|
||||||
|
* The cause.
|
||||||
|
*/
|
||||||
|
public JSONException(final Throwable cause) {
|
||||||
|
super(cause.getMessage(), cause);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
191
JSONML.java → src/main/java/org/json/JSONML.java
Executable file → Normal file
191
JSONML.java → src/main/java/org/json/JSONML.java
Executable file → Normal file
|
@ -24,32 +24,30 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
SOFTWARE.
|
SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.util.Iterator;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This provides static methods to convert an XML text into a JSONArray or
|
* This provides static methods to convert an XML text into a JSONArray or
|
||||||
* JSONObject, and to covert a JSONArray or JSONObject into an XML text using
|
* JSONObject, and to covert a JSONArray or JSONObject into an XML text using
|
||||||
* the JsonML transform.
|
* the JsonML transform.
|
||||||
*
|
*
|
||||||
* @author JSON.org
|
* @author JSON.org
|
||||||
* @version 2014-05-03
|
* @version 2016-01-30
|
||||||
*/
|
*/
|
||||||
public class JSONML {
|
public class 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.
|
||||||
* @param arrayForm true if array form, false if object form.
|
* @param arrayForm true if array form, false if object form.
|
||||||
* @param ja The JSONArray that is containing the current tag or null
|
* @param ja The JSONArray that is containing the current tag or null
|
||||||
* if we are at the outermost level.
|
* if we are at the outermost level.
|
||||||
|
* @param keepStrings Don't type-convert text nodes and attribute values
|
||||||
* @return A JSONArray if the value is the outermost tag, otherwise null.
|
* @return A JSONArray if the value is the outermost tag, otherwise null.
|
||||||
* @throws JSONException
|
* @throws JSONException if a parsing error occurs
|
||||||
*/
|
*/
|
||||||
private static Object parse(
|
private static Object parse(
|
||||||
XMLTokener x,
|
XMLTokener x,
|
||||||
boolean arrayForm,
|
boolean arrayForm,
|
||||||
JSONArray ja
|
JSONArray ja,
|
||||||
|
boolean keepStrings
|
||||||
) throws JSONException {
|
) throws JSONException {
|
||||||
String attribute;
|
String attribute;
|
||||||
char c;
|
char c;
|
||||||
|
@ -174,7 +172,7 @@ public class JSONML {
|
||||||
if (!(token instanceof String)) {
|
if (!(token instanceof String)) {
|
||||||
throw x.syntaxError("Missing value");
|
throw x.syntaxError("Missing value");
|
||||||
}
|
}
|
||||||
newjo.accumulate(attribute, XML.stringToValue((String)token));
|
newjo.accumulate(attribute, keepStrings ? ((String)token) :XML.stringToValue((String)token));
|
||||||
token = null;
|
token = null;
|
||||||
} else {
|
} else {
|
||||||
newjo.accumulate(attribute, "");
|
newjo.accumulate(attribute, "");
|
||||||
|
@ -193,9 +191,8 @@ public class JSONML {
|
||||||
if (ja == null) {
|
if (ja == null) {
|
||||||
if (arrayForm) {
|
if (arrayForm) {
|
||||||
return newja;
|
return newja;
|
||||||
} else {
|
|
||||||
return newjo;
|
|
||||||
}
|
}
|
||||||
|
return newjo;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Content, between <...> and </...>
|
// Content, between <...> and </...>
|
||||||
|
@ -204,7 +201,7 @@ public class JSONML {
|
||||||
if (token != XML.GT) {
|
if (token != XML.GT) {
|
||||||
throw x.syntaxError("Misshaped tag");
|
throw x.syntaxError("Misshaped tag");
|
||||||
}
|
}
|
||||||
closeTag = (String)parse(x, arrayForm, newja);
|
closeTag = (String)parse(x, arrayForm, newja, keepStrings);
|
||||||
if (closeTag != null) {
|
if (closeTag != null) {
|
||||||
if (!closeTag.equals(tagName)) {
|
if (!closeTag.equals(tagName)) {
|
||||||
throw x.syntaxError("Mismatched '" + tagName +
|
throw x.syntaxError("Mismatched '" + tagName +
|
||||||
|
@ -217,9 +214,8 @@ public class JSONML {
|
||||||
if (ja == null) {
|
if (ja == null) {
|
||||||
if (arrayForm) {
|
if (arrayForm) {
|
||||||
return newja;
|
return newja;
|
||||||
} else {
|
|
||||||
return newjo;
|
|
||||||
}
|
}
|
||||||
|
return newjo;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -227,7 +223,7 @@ public class JSONML {
|
||||||
} else {
|
} else {
|
||||||
if (ja != null) {
|
if (ja != null) {
|
||||||
ja.put(token instanceof String
|
ja.put(token instanceof String
|
||||||
? XML.stringToValue((String)token)
|
? keepStrings ? XML.unescape((String)token) :XML.stringToValue((String)token)
|
||||||
: token);
|
: token);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -242,13 +238,35 @@ public class JSONML {
|
||||||
* attributes, then the second element will be JSONObject containing the
|
* attributes, then the second element will be JSONObject containing the
|
||||||
* name/value pairs. If the tag contains children, then strings and
|
* name/value pairs. If the tag contains children, then strings and
|
||||||
* JSONArrays will represent the child tags.
|
* JSONArrays will represent the child tags.
|
||||||
* Comments, prologs, DTDs, and <code><[ [ ]]></code> are ignored.
|
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||||
* @param string The source string.
|
* @param string The source string.
|
||||||
* @return A JSONArray containing the structured data from the XML string.
|
* @return A JSONArray containing the structured data from the XML string.
|
||||||
* @throws JSONException
|
* @throws JSONException Thrown on error converting to a JSONArray
|
||||||
*/
|
*/
|
||||||
public static JSONArray toJSONArray(String string) throws JSONException {
|
public static JSONArray toJSONArray(String string) throws JSONException {
|
||||||
return toJSONArray(new XMLTokener(string));
|
return (JSONArray)parse(new XMLTokener(string), true, null, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
|
* JSONArray using the JsonML transform. Each XML tag is represented as
|
||||||
|
* a JSONArray in which the first element is the tag name. If the tag has
|
||||||
|
* attributes, then the second element will be JSONObject containing the
|
||||||
|
* name/value pairs. If the tag contains children, then strings and
|
||||||
|
* JSONArrays will represent the child tags.
|
||||||
|
* As opposed to toJSONArray this method does not attempt to convert
|
||||||
|
* any text node or attribute value to any type
|
||||||
|
* but just leaves it as a string.
|
||||||
|
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||||
|
* @param string The source string.
|
||||||
|
* @param keepStrings If true, then values will not be coerced into boolean
|
||||||
|
* or numeric values and will instead be left as strings
|
||||||
|
* @return A JSONArray containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown on error converting to a JSONArray
|
||||||
|
*/
|
||||||
|
public static JSONArray toJSONArray(String string, boolean keepStrings) throws JSONException {
|
||||||
|
return (JSONArray)parse(new XMLTokener(string), true, null, keepStrings);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -259,13 +277,35 @@ public class JSONML {
|
||||||
* attributes, then the second element will be JSONObject containing the
|
* attributes, then the second element will be JSONObject containing the
|
||||||
* name/value pairs. If the tag contains children, then strings and
|
* name/value pairs. If the tag contains children, then strings and
|
||||||
* JSONArrays will represent the child content and tags.
|
* JSONArrays will represent the child content and tags.
|
||||||
* Comments, prologs, DTDs, and <code><[ [ ]]></code> are ignored.
|
* As opposed to toJSONArray this method does not attempt to convert
|
||||||
|
* any text node or attribute value to any type
|
||||||
|
* but just leaves it as a string.
|
||||||
|
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||||
|
* @param x An XMLTokener.
|
||||||
|
* @param keepStrings If true, then values will not be coerced into boolean
|
||||||
|
* or numeric values and will instead be left as strings
|
||||||
|
* @return A JSONArray containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown on error converting to a JSONArray
|
||||||
|
*/
|
||||||
|
public static JSONArray toJSONArray(XMLTokener x, boolean keepStrings) throws JSONException {
|
||||||
|
return (JSONArray)parse(x, true, null, keepStrings);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
|
* JSONArray using the JsonML transform. Each XML tag is represented as
|
||||||
|
* a JSONArray in which the first element is the tag name. If the tag has
|
||||||
|
* attributes, then the second element will be JSONObject containing the
|
||||||
|
* name/value pairs. If the tag contains children, then strings and
|
||||||
|
* JSONArrays will represent the child content and tags.
|
||||||
|
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||||
* @param x An XMLTokener.
|
* @param x An XMLTokener.
|
||||||
* @return A JSONArray containing the structured data from the XML string.
|
* @return A JSONArray containing the structured data from the XML string.
|
||||||
* @throws JSONException
|
* @throws JSONException Thrown on error converting to a JSONArray
|
||||||
*/
|
*/
|
||||||
public static JSONArray toJSONArray(XMLTokener x) throws JSONException {
|
public static JSONArray toJSONArray(XMLTokener x) throws JSONException {
|
||||||
return (JSONArray)parse(x, true, null);
|
return (JSONArray)parse(x, true, null, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -277,31 +317,71 @@ public class JSONML {
|
||||||
* contains children, the object will have a "childNodes" property which
|
* contains children, the object will have a "childNodes" property which
|
||||||
* will be an array of strings and JsonML JSONObjects.
|
* will be an array of strings and JsonML JSONObjects.
|
||||||
|
|
||||||
* Comments, prologs, DTDs, and <code><[ [ ]]></code> are ignored.
|
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||||
* @param x An XMLTokener of the XML source text.
|
|
||||||
* @return A JSONObject containing the structured data from the XML string.
|
|
||||||
* @throws JSONException
|
|
||||||
*/
|
|
||||||
public static JSONObject toJSONObject(XMLTokener x) throws JSONException {
|
|
||||||
return (JSONObject)parse(x, false, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Convert a well-formed (but not necessarily valid) XML string into a
|
|
||||||
* JSONObject using the JsonML transform. Each XML tag is represented as
|
|
||||||
* a JSONObject with a "tagName" property. If the tag has attributes, then
|
|
||||||
* the attributes will be in the JSONObject as properties. If the tag
|
|
||||||
* contains children, the object will have a "childNodes" property which
|
|
||||||
* will be an array of strings and JsonML JSONObjects.
|
|
||||||
|
|
||||||
* Comments, prologs, DTDs, and <code><[ [ ]]></code> are ignored.
|
|
||||||
* @param string The XML source text.
|
* @param string The XML source text.
|
||||||
* @return A JSONObject containing the structured data from the XML string.
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
* @throws JSONException
|
* @throws JSONException Thrown on error converting to a JSONObject
|
||||||
*/
|
*/
|
||||||
public static JSONObject toJSONObject(String string) throws JSONException {
|
public static JSONObject toJSONObject(String string) throws JSONException {
|
||||||
return toJSONObject(new XMLTokener(string));
|
return (JSONObject)parse(new XMLTokener(string), false, null, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
|
* JSONObject using the JsonML transform. Each XML tag is represented as
|
||||||
|
* a JSONObject with a "tagName" property. If the tag has attributes, then
|
||||||
|
* the attributes will be in the JSONObject as properties. If the tag
|
||||||
|
* contains children, the object will have a "childNodes" property which
|
||||||
|
* will be an array of strings and JsonML JSONObjects.
|
||||||
|
|
||||||
|
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||||
|
* @param string The XML source text.
|
||||||
|
* @param keepStrings If true, then values will not be coerced into boolean
|
||||||
|
* or numeric values and will instead be left as strings
|
||||||
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown on error converting to a JSONObject
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(String string, boolean keepStrings) throws JSONException {
|
||||||
|
return (JSONObject)parse(new XMLTokener(string), false, null, keepStrings);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
|
* JSONObject using the JsonML transform. Each XML tag is represented as
|
||||||
|
* a JSONObject with a "tagName" property. If the tag has attributes, then
|
||||||
|
* the attributes will be in the JSONObject as properties. If the tag
|
||||||
|
* contains children, the object will have a "childNodes" property which
|
||||||
|
* will be an array of strings and JsonML JSONObjects.
|
||||||
|
|
||||||
|
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||||
|
* @param x An XMLTokener of the XML source text.
|
||||||
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown on error converting to a JSONObject
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(XMLTokener x) throws JSONException {
|
||||||
|
return (JSONObject)parse(x, false, null, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
|
* JSONObject using the JsonML transform. Each XML tag is represented as
|
||||||
|
* a JSONObject with a "tagName" property. If the tag has attributes, then
|
||||||
|
* the attributes will be in the JSONObject as properties. If the tag
|
||||||
|
* contains children, the object will have a "childNodes" property which
|
||||||
|
* will be an array of strings and JsonML JSONObjects.
|
||||||
|
|
||||||
|
* Comments, prologs, DTDs, and <pre>{@code <[ [ ]]>}</pre> are ignored.
|
||||||
|
* @param x An XMLTokener of the XML source text.
|
||||||
|
* @param keepStrings If true, then values will not be coerced into boolean
|
||||||
|
* or numeric values and will instead be left as strings
|
||||||
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown on error converting to a JSONObject
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(XMLTokener x, boolean keepStrings) throws JSONException {
|
||||||
|
return (JSONObject)parse(x, false, null, keepStrings);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -309,18 +389,15 @@ public class JSONML {
|
||||||
* Reverse the JSONML transformation, making an XML text from a JSONArray.
|
* Reverse the JSONML transformation, making an XML text from a JSONArray.
|
||||||
* @param ja A JSONArray.
|
* @param ja A JSONArray.
|
||||||
* @return An XML string.
|
* @return An XML string.
|
||||||
* @throws JSONException
|
* @throws JSONException Thrown on error converting to a string
|
||||||
*/
|
*/
|
||||||
public static String toString(JSONArray ja) throws JSONException {
|
public static String toString(JSONArray ja) throws JSONException {
|
||||||
int i;
|
int i;
|
||||||
JSONObject jo;
|
JSONObject jo;
|
||||||
String key;
|
|
||||||
Iterator<String> keys;
|
|
||||||
int length;
|
int length;
|
||||||
Object object;
|
Object object;
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
String tagName;
|
String tagName;
|
||||||
String value;
|
|
||||||
|
|
||||||
// Emit <tagName
|
// Emit <tagName
|
||||||
|
|
||||||
|
@ -337,17 +414,16 @@ public class JSONML {
|
||||||
|
|
||||||
// Emit the attributes
|
// Emit the attributes
|
||||||
|
|
||||||
keys = jo.keys();
|
// Don't use the new entrySet API to maintain Android support
|
||||||
while (keys.hasNext()) {
|
for (final String key : jo.keySet()) {
|
||||||
key = keys.next();
|
final Object value = jo.opt(key);
|
||||||
XML.noSpace(key);
|
XML.noSpace(key);
|
||||||
value = jo.optString(key);
|
|
||||||
if (value != null) {
|
if (value != null) {
|
||||||
sb.append(' ');
|
sb.append(' ');
|
||||||
sb.append(XML.escape(key));
|
sb.append(XML.escape(key));
|
||||||
sb.append('=');
|
sb.append('=');
|
||||||
sb.append('"');
|
sb.append('"');
|
||||||
sb.append(XML.escape(value));
|
sb.append(XML.escape(value.toString()));
|
||||||
sb.append('"');
|
sb.append('"');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -393,18 +469,16 @@ public class JSONML {
|
||||||
* The other properties are attributes with string values.
|
* The other properties are attributes with string values.
|
||||||
* @param jo A JSONObject.
|
* @param jo A JSONObject.
|
||||||
* @return An XML string.
|
* @return An XML string.
|
||||||
* @throws JSONException
|
* @throws JSONException Thrown on error converting to a string
|
||||||
*/
|
*/
|
||||||
public static String toString(JSONObject jo) throws JSONException {
|
public static String toString(JSONObject jo) throws JSONException {
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
int i;
|
int i;
|
||||||
JSONArray ja;
|
JSONArray ja;
|
||||||
String key;
|
|
||||||
Iterator<String> keys;
|
|
||||||
int length;
|
int length;
|
||||||
Object object;
|
Object object;
|
||||||
String tagName;
|
String tagName;
|
||||||
String value;
|
Object value;
|
||||||
|
|
||||||
//Emit <tagName
|
//Emit <tagName
|
||||||
|
|
||||||
|
@ -419,18 +493,17 @@ public class JSONML {
|
||||||
|
|
||||||
//Emit the attributes
|
//Emit the attributes
|
||||||
|
|
||||||
keys = jo.keys();
|
// Don't use the new entrySet API to maintain Android support
|
||||||
while (keys.hasNext()) {
|
for (final String key : jo.keySet()) {
|
||||||
key = keys.next();
|
|
||||||
if (!"tagName".equals(key) && !"childNodes".equals(key)) {
|
if (!"tagName".equals(key) && !"childNodes".equals(key)) {
|
||||||
XML.noSpace(key);
|
XML.noSpace(key);
|
||||||
value = jo.optString(key);
|
value = jo.opt(key);
|
||||||
if (value != null) {
|
if (value != null) {
|
||||||
sb.append(' ');
|
sb.append(' ');
|
||||||
sb.append(XML.escape(key));
|
sb.append(XML.escape(key));
|
||||||
sb.append('=');
|
sb.append('=');
|
||||||
sb.append('"');
|
sb.append('"');
|
||||||
sb.append(XML.escape(value));
|
sb.append(XML.escape(value.toString()));
|
||||||
sb.append('"');
|
sb.append('"');
|
||||||
}
|
}
|
||||||
}
|
}
|
1345
JSONObject.java → src/main/java/org/json/JSONObject.java
Executable file → Normal file
1345
JSONObject.java → src/main/java/org/json/JSONObject.java
Executable file → Normal file
File diff suppressed because it is too large
Load diff
295
src/main/java/org/json/JSONPointer.java
Normal file
295
src/main/java/org/json/JSONPointer.java
Normal file
|
@ -0,0 +1,295 @@
|
||||||
|
package org.json;
|
||||||
|
|
||||||
|
import static java.lang.String.format;
|
||||||
|
|
||||||
|
import java.io.UnsupportedEncodingException;
|
||||||
|
import java.net.URLDecoder;
|
||||||
|
import java.net.URLEncoder;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2002 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A JSON Pointer is a simple query language defined for JSON documents by
|
||||||
|
* <a href="https://tools.ietf.org/html/rfc6901">RFC 6901</a>.
|
||||||
|
*
|
||||||
|
* In a nutshell, JSONPointer allows the user to navigate into a JSON document
|
||||||
|
* using strings, and retrieve targeted objects, like a simple form of XPATH.
|
||||||
|
* Path segments are separated by the '/' char, which signifies the root of
|
||||||
|
* the document when it appears as the first char of the string. Array
|
||||||
|
* elements are navigated using ordinals, counting from 0. JSONPointer strings
|
||||||
|
* may be extended to any arbitrary number of segments. If the navigation
|
||||||
|
* is successful, the matched item is returned. A matched item may be a
|
||||||
|
* JSONObject, a JSONArray, or a JSON value. If the JSONPointer string building
|
||||||
|
* fails, an appropriate exception is thrown. If the navigation fails to find
|
||||||
|
* a match, a JSONPointerException is thrown.
|
||||||
|
*
|
||||||
|
* @author JSON.org
|
||||||
|
* @version 2016-05-14
|
||||||
|
*/
|
||||||
|
public class JSONPointer {
|
||||||
|
|
||||||
|
// used for URL encoding and decoding
|
||||||
|
private static final String ENCODING = "utf-8";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class allows the user to build a JSONPointer in steps, using
|
||||||
|
* exactly one segment in each step.
|
||||||
|
*/
|
||||||
|
public static class Builder {
|
||||||
|
|
||||||
|
// Segments for the eventual JSONPointer string
|
||||||
|
private final List<String> refTokens = new ArrayList<String>();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a {@code JSONPointer} instance using the tokens previously set using the
|
||||||
|
* {@link #append(String)} method calls.
|
||||||
|
* @return a JSONPointer object
|
||||||
|
*/
|
||||||
|
public JSONPointer build() {
|
||||||
|
return new JSONPointer(this.refTokens);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adds an arbitrary token to the list of reference tokens. It can be any non-null value.
|
||||||
|
*
|
||||||
|
* Unlike in the case of JSON string or URI fragment representation of JSON pointers, the
|
||||||
|
* argument of this method MUST NOT be escaped. If you want to query the property called
|
||||||
|
* {@code "a~b"} then you should simply pass the {@code "a~b"} string as-is, there is no
|
||||||
|
* need to escape it as {@code "a~0b"}.
|
||||||
|
*
|
||||||
|
* @param token the new token to be appended to the list
|
||||||
|
* @return {@code this}
|
||||||
|
* @throws NullPointerException if {@code token} is null
|
||||||
|
*/
|
||||||
|
public Builder append(String token) {
|
||||||
|
if (token == null) {
|
||||||
|
throw new NullPointerException("token cannot be null");
|
||||||
|
}
|
||||||
|
this.refTokens.add(token);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adds an integer to the reference token list. Although not necessarily, mostly this token will
|
||||||
|
* denote an array index.
|
||||||
|
*
|
||||||
|
* @param arrayIndex the array index to be added to the token list
|
||||||
|
* @return {@code this}
|
||||||
|
*/
|
||||||
|
public Builder append(int arrayIndex) {
|
||||||
|
this.refTokens.add(String.valueOf(arrayIndex));
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Static factory method for {@link Builder}. Example usage:
|
||||||
|
*
|
||||||
|
* <pre><code>
|
||||||
|
* JSONPointer pointer = JSONPointer.builder()
|
||||||
|
* .append("obj")
|
||||||
|
* .append("other~key").append("another/key")
|
||||||
|
* .append("\"")
|
||||||
|
* .append(0)
|
||||||
|
* .build();
|
||||||
|
* </code></pre>
|
||||||
|
*
|
||||||
|
* @return a builder instance which can be used to construct a {@code JSONPointer} instance by chained
|
||||||
|
* {@link Builder#append(String)} calls.
|
||||||
|
*/
|
||||||
|
public static Builder builder() {
|
||||||
|
return new Builder();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Segments for the JSONPointer string
|
||||||
|
private final List<String> refTokens;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Pre-parses and initializes a new {@code JSONPointer} instance. If you want to
|
||||||
|
* evaluate the same JSON Pointer on different JSON documents then it is recommended
|
||||||
|
* to keep the {@code JSONPointer} instances due to performance considerations.
|
||||||
|
*
|
||||||
|
* @param pointer the JSON String or URI Fragment representation of the JSON pointer.
|
||||||
|
* @throws IllegalArgumentException if {@code pointer} is not a valid JSON pointer
|
||||||
|
*/
|
||||||
|
public JSONPointer(final String pointer) {
|
||||||
|
if (pointer == null) {
|
||||||
|
throw new NullPointerException("pointer cannot be null");
|
||||||
|
}
|
||||||
|
if (pointer.isEmpty() || pointer.equals("#")) {
|
||||||
|
this.refTokens = Collections.emptyList();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
String refs;
|
||||||
|
if (pointer.startsWith("#/")) {
|
||||||
|
refs = pointer.substring(2);
|
||||||
|
try {
|
||||||
|
refs = URLDecoder.decode(refs, ENCODING);
|
||||||
|
} catch (UnsupportedEncodingException e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
} else if (pointer.startsWith("/")) {
|
||||||
|
refs = pointer.substring(1);
|
||||||
|
} else {
|
||||||
|
throw new IllegalArgumentException("a JSON pointer should start with '/' or '#/'");
|
||||||
|
}
|
||||||
|
this.refTokens = new ArrayList<String>();
|
||||||
|
int slashIdx = -1;
|
||||||
|
int prevSlashIdx = 0;
|
||||||
|
do {
|
||||||
|
prevSlashIdx = slashIdx + 1;
|
||||||
|
slashIdx = refs.indexOf('/', prevSlashIdx);
|
||||||
|
if(prevSlashIdx == slashIdx || prevSlashIdx == refs.length()) {
|
||||||
|
// found 2 slashes in a row ( obj//next )
|
||||||
|
// or single slash at the end of a string ( obj/test/ )
|
||||||
|
this.refTokens.add("");
|
||||||
|
} else if (slashIdx >= 0) {
|
||||||
|
final String token = refs.substring(prevSlashIdx, slashIdx);
|
||||||
|
this.refTokens.add(unescape(token));
|
||||||
|
} else {
|
||||||
|
// last item after separator, or no separator at all.
|
||||||
|
final String token = refs.substring(prevSlashIdx);
|
||||||
|
this.refTokens.add(unescape(token));
|
||||||
|
}
|
||||||
|
} while (slashIdx >= 0);
|
||||||
|
// using split does not take into account consecutive separators or "ending nulls"
|
||||||
|
//for (String token : refs.split("/")) {
|
||||||
|
// this.refTokens.add(unescape(token));
|
||||||
|
//}
|
||||||
|
}
|
||||||
|
|
||||||
|
public JSONPointer(List<String> refTokens) {
|
||||||
|
this.refTokens = new ArrayList<String>(refTokens);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String unescape(String token) {
|
||||||
|
return token.replace("~1", "/").replace("~0", "~")
|
||||||
|
.replace("\\\"", "\"")
|
||||||
|
.replace("\\\\", "\\");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Evaluates this JSON Pointer on the given {@code document}. The {@code document}
|
||||||
|
* is usually a {@link JSONObject} or a {@link JSONArray} instance, but the empty
|
||||||
|
* JSON Pointer ({@code ""}) can be evaluated on any JSON values and in such case the
|
||||||
|
* returned value will be {@code document} itself.
|
||||||
|
*
|
||||||
|
* @param document the JSON document which should be the subject of querying.
|
||||||
|
* @return the result of the evaluation
|
||||||
|
* @throws JSONPointerException if an error occurs during evaluation
|
||||||
|
*/
|
||||||
|
public Object queryFrom(Object document) throws JSONPointerException {
|
||||||
|
if (this.refTokens.isEmpty()) {
|
||||||
|
return document;
|
||||||
|
}
|
||||||
|
Object current = document;
|
||||||
|
for (String token : this.refTokens) {
|
||||||
|
if (current instanceof JSONObject) {
|
||||||
|
current = ((JSONObject) current).opt(unescape(token));
|
||||||
|
} else if (current instanceof JSONArray) {
|
||||||
|
current = readByIndexToken(current, token);
|
||||||
|
} else {
|
||||||
|
throw new JSONPointerException(format(
|
||||||
|
"value [%s] is not an array or object therefore its key %s cannot be resolved", current,
|
||||||
|
token));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return current;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Matches a JSONArray element by ordinal position
|
||||||
|
* @param current the JSONArray to be evaluated
|
||||||
|
* @param indexToken the array index in string form
|
||||||
|
* @return the matched object. If no matching item is found a
|
||||||
|
* @throws JSONPointerException is thrown if the index is out of bounds
|
||||||
|
*/
|
||||||
|
private static Object readByIndexToken(Object current, String indexToken) throws JSONPointerException {
|
||||||
|
try {
|
||||||
|
int index = Integer.parseInt(indexToken);
|
||||||
|
JSONArray currentArr = (JSONArray) current;
|
||||||
|
if (index >= currentArr.length()) {
|
||||||
|
throw new JSONPointerException(format("index %s is out of bounds - the array has %d elements", indexToken,
|
||||||
|
Integer.valueOf(currentArr.length())));
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
return currentArr.get(index);
|
||||||
|
} catch (JSONException e) {
|
||||||
|
throw new JSONPointerException("Error reading value at index position " + index, e);
|
||||||
|
}
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
throw new JSONPointerException(format("%s is not an array index", indexToken), e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a string representing the JSONPointer path value using string
|
||||||
|
* representation
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
StringBuilder rval = new StringBuilder("");
|
||||||
|
for (String token: this.refTokens) {
|
||||||
|
rval.append('/').append(escape(token));
|
||||||
|
}
|
||||||
|
return rval.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Escapes path segment values to an unambiguous form.
|
||||||
|
* The escape char to be inserted is '~'. The chars to be escaped
|
||||||
|
* are ~, which maps to ~0, and /, which maps to ~1. Backslashes
|
||||||
|
* and double quote chars are also escaped.
|
||||||
|
* @param token the JSONPointer segment value to be escaped
|
||||||
|
* @return the escaped value for the token
|
||||||
|
*/
|
||||||
|
private static String escape(String token) {
|
||||||
|
return token.replace("~", "~0")
|
||||||
|
.replace("/", "~1")
|
||||||
|
.replace("\\", "\\\\")
|
||||||
|
.replace("\"", "\\\"");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a string representing the JSONPointer path value using URI
|
||||||
|
* fragment identifier representation
|
||||||
|
* @return a uri fragment string
|
||||||
|
*/
|
||||||
|
public String toURIFragment() {
|
||||||
|
try {
|
||||||
|
StringBuilder rval = new StringBuilder("#");
|
||||||
|
for (String token : this.refTokens) {
|
||||||
|
rval.append('/').append(URLEncoder.encode(token, ENCODING));
|
||||||
|
}
|
||||||
|
return rval.toString();
|
||||||
|
} catch (UnsupportedEncodingException e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
45
src/main/java/org/json/JSONPointerException.java
Normal file
45
src/main/java/org/json/JSONPointerException.java
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
package org.json;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2002 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The JSONPointerException is thrown by {@link JSONPointer} if an error occurs
|
||||||
|
* during evaluating a pointer.
|
||||||
|
*
|
||||||
|
* @author JSON.org
|
||||||
|
* @version 2016-05-13
|
||||||
|
*/
|
||||||
|
public class JSONPointerException extends JSONException {
|
||||||
|
private static final long serialVersionUID = 8872944667561856751L;
|
||||||
|
|
||||||
|
public JSONPointerException(String message) {
|
||||||
|
super(message);
|
||||||
|
}
|
||||||
|
|
||||||
|
public JSONPointerException(String message, Throwable cause) {
|
||||||
|
super(message, cause);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
43
src/main/java/org/json/JSONPropertyIgnore.java
Normal file
43
src/main/java/org/json/JSONPropertyIgnore.java
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
package org.json;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2018 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static java.lang.annotation.ElementType.METHOD;
|
||||||
|
import static java.lang.annotation.RetentionPolicy.RUNTIME;
|
||||||
|
|
||||||
|
import java.lang.annotation.Documented;
|
||||||
|
import java.lang.annotation.Retention;
|
||||||
|
import java.lang.annotation.Target;
|
||||||
|
|
||||||
|
@Documented
|
||||||
|
@Retention(RUNTIME)
|
||||||
|
@Target({METHOD})
|
||||||
|
/**
|
||||||
|
* Use this annotation on a getter method to override the Bean name
|
||||||
|
* parser for Bean -> JSONObject mapping. If this annotation is
|
||||||
|
* present at any level in the class hierarchy, then the method will
|
||||||
|
* not be serialized from the bean into the JSONObject.
|
||||||
|
*/
|
||||||
|
public @interface JSONPropertyIgnore { }
|
47
src/main/java/org/json/JSONPropertyName.java
Normal file
47
src/main/java/org/json/JSONPropertyName.java
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
package org.json;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2018 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static java.lang.annotation.ElementType.METHOD;
|
||||||
|
import static java.lang.annotation.RetentionPolicy.RUNTIME;
|
||||||
|
|
||||||
|
import java.lang.annotation.Documented;
|
||||||
|
import java.lang.annotation.Retention;
|
||||||
|
import java.lang.annotation.Target;
|
||||||
|
|
||||||
|
@Documented
|
||||||
|
@Retention(RUNTIME)
|
||||||
|
@Target({METHOD})
|
||||||
|
/**
|
||||||
|
* Use this annotation on a getter method to override the Bean name
|
||||||
|
* parser for Bean -> JSONObject mapping. A value set to empty string <code>""</code>
|
||||||
|
* will have the Bean parser fall back to the default field name processing.
|
||||||
|
*/
|
||||||
|
public @interface JSONPropertyName {
|
||||||
|
/**
|
||||||
|
* @return The name of the property as to be used in the JSON Object.
|
||||||
|
*/
|
||||||
|
String value();
|
||||||
|
}
|
43
src/main/java/org/json/JSONString.java
Normal file
43
src/main/java/org/json/JSONString.java
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
package org.json;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2002 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The <code>JSONString</code> interface allows a <code>toJSONString()</code>
|
||||||
|
* method so that a class can change the behavior of
|
||||||
|
* <code>JSONObject.toString()</code>, <code>JSONArray.toString()</code>,
|
||||||
|
* and <code>JSONWriter.value(</code>Object<code>)</code>. The
|
||||||
|
* <code>toJSONString</code> method will be used instead of the default behavior
|
||||||
|
* of using the Object's <code>toString()</code> method and quoting the result.
|
||||||
|
*/
|
||||||
|
public interface JSONString {
|
||||||
|
/**
|
||||||
|
* The <code>toJSONString</code> method allows a class to produce its own JSON
|
||||||
|
* serialization.
|
||||||
|
*
|
||||||
|
* @return A strictly syntactically correct JSON text.
|
||||||
|
*/
|
||||||
|
public String toJSONString();
|
||||||
|
}
|
3
JSONStringer.java → src/main/java/org/json/JSONStringer.java
Executable file → Normal file
3
JSONStringer.java → src/main/java/org/json/JSONStringer.java
Executable file → Normal file
|
@ -54,7 +54,7 @@ import java.io.StringWriter;
|
||||||
* <p>
|
* <p>
|
||||||
* This can sometimes be easier than using a JSONObject to build a string.
|
* This can sometimes be easier than using a JSONObject to build a string.
|
||||||
* @author JSON.org
|
* @author JSON.org
|
||||||
* @version 2008-09-18
|
* @version 2015-12-09
|
||||||
*/
|
*/
|
||||||
public class JSONStringer extends JSONWriter {
|
public class JSONStringer extends JSONWriter {
|
||||||
/**
|
/**
|
||||||
|
@ -72,6 +72,7 @@ public class JSONStringer extends JSONWriter {
|
||||||
* <code>endArray</code>).
|
* <code>endArray</code>).
|
||||||
* @return The JSON text.
|
* @return The JSON text.
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return this.mode == 'd' ? this.writer.toString() : null;
|
return this.mode == 'd' ? this.writer.toString() : null;
|
||||||
}
|
}
|
|
@ -39,18 +39,26 @@ SOFTWARE.
|
||||||
* @version 2014-05-03
|
* @version 2014-05-03
|
||||||
*/
|
*/
|
||||||
public class JSONTokener {
|
public class JSONTokener {
|
||||||
|
/** current read character position on the current line. */
|
||||||
private long character;
|
private long character;
|
||||||
|
/** flag to indicate if the end of the input has been found. */
|
||||||
private boolean eof;
|
private boolean eof;
|
||||||
|
/** current read index of the input. */
|
||||||
private long index;
|
private long index;
|
||||||
|
/** current line of the input. */
|
||||||
private long line;
|
private long line;
|
||||||
|
/** previous character read from the input. */
|
||||||
private char previous;
|
private char previous;
|
||||||
private Reader reader;
|
/** Reader for the input. */
|
||||||
|
private final Reader reader;
|
||||||
|
/** flag to indicate that a previous character was requested. */
|
||||||
private boolean usePrevious;
|
private boolean usePrevious;
|
||||||
|
/** the number of characters read in the previous line. */
|
||||||
|
private long characterPreviousLine;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct a JSONTokener from a Reader.
|
* Construct a JSONTokener from a Reader. The caller must close the Reader.
|
||||||
*
|
*
|
||||||
* @param reader A reader.
|
* @param reader A reader.
|
||||||
*/
|
*/
|
||||||
|
@ -63,15 +71,16 @@ public class JSONTokener {
|
||||||
this.previous = 0;
|
this.previous = 0;
|
||||||
this.index = 0;
|
this.index = 0;
|
||||||
this.character = 1;
|
this.character = 1;
|
||||||
|
this.characterPreviousLine = 0;
|
||||||
this.line = 1;
|
this.line = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct a JSONTokener from an InputStream.
|
* Construct a JSONTokener from an InputStream. The caller must close the input stream.
|
||||||
* @param inputStream The source.
|
* @param inputStream The source.
|
||||||
*/
|
*/
|
||||||
public JSONTokener(InputStream inputStream) throws JSONException {
|
public JSONTokener(InputStream inputStream) {
|
||||||
this(new InputStreamReader(inputStream));
|
this(new InputStreamReader(inputStream));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -90,17 +99,30 @@ public class JSONTokener {
|
||||||
* Back up one character. This provides a sort of lookahead capability,
|
* Back up one character. This provides a sort of lookahead capability,
|
||||||
* so that you can test for a digit or letter before attempting to parse
|
* so that you can test for a digit or letter before attempting to parse
|
||||||
* the next number or identifier.
|
* the next number or identifier.
|
||||||
|
* @throws JSONException Thrown if trying to step back more than 1 step
|
||||||
|
* or if already at the start of the string
|
||||||
*/
|
*/
|
||||||
public void back() throws JSONException {
|
public void back() throws JSONException {
|
||||||
if (this.usePrevious || this.index <= 0) {
|
if (this.usePrevious || this.index <= 0) {
|
||||||
throw new JSONException("Stepping back two steps is not supported");
|
throw new JSONException("Stepping back two steps is not supported");
|
||||||
}
|
}
|
||||||
this.index -= 1;
|
this.decrementIndexes();
|
||||||
this.character -= 1;
|
|
||||||
this.usePrevious = true;
|
this.usePrevious = true;
|
||||||
this.eof = false;
|
this.eof = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Decrements the indexes for the {@link #back()} method based on the previous character read.
|
||||||
|
*/
|
||||||
|
private void decrementIndexes() {
|
||||||
|
this.index--;
|
||||||
|
if(this.previous=='\r' || this.previous == '\n') {
|
||||||
|
this.line--;
|
||||||
|
this.character=this.characterPreviousLine ;
|
||||||
|
} else if(this.character > 0){
|
||||||
|
this.character--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the hex value of a character (base16).
|
* Get the hex value of a character (base16).
|
||||||
|
@ -121,6 +143,11 @@ public class JSONTokener {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the end of the input has been reached.
|
||||||
|
*
|
||||||
|
* @return true if at the end of the file and we didn't step back
|
||||||
|
*/
|
||||||
public boolean end() {
|
public boolean end() {
|
||||||
return this.eof && !this.usePrevious;
|
return this.eof && !this.usePrevious;
|
||||||
}
|
}
|
||||||
|
@ -130,13 +157,28 @@ public class JSONTokener {
|
||||||
* Determine if the source string still contains characters that next()
|
* Determine if the source string still contains characters that next()
|
||||||
* can consume.
|
* can consume.
|
||||||
* @return true if not yet at the end of the source.
|
* @return true if not yet at the end of the source.
|
||||||
|
* @throws JSONException thrown if there is an error stepping forward
|
||||||
|
* or backward while checking for more data.
|
||||||
*/
|
*/
|
||||||
public boolean more() throws JSONException {
|
public boolean more() throws JSONException {
|
||||||
this.next();
|
if(this.usePrevious) {
|
||||||
if (this.end()) {
|
return true;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
this.reader.mark(1);
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw new JSONException("Unable to preserve stream position", e);
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
// -1 is EOF, but next() can not consume the null character '\0'
|
||||||
|
if(this.reader.read() <= 0) {
|
||||||
|
this.eof = true;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
this.back();
|
this.reader.reset();
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw new JSONException("Unable to read the next character from the stream", e);
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -145,6 +187,7 @@ public class JSONTokener {
|
||||||
* Get the next character in the source string.
|
* Get the next character in the source string.
|
||||||
*
|
*
|
||||||
* @return The next character, or 0 if past the end of the source string.
|
* @return The next character, or 0 if past the end of the source string.
|
||||||
|
* @throws JSONException Thrown if there is an error reading the source string.
|
||||||
*/
|
*/
|
||||||
public char next() throws JSONException {
|
public char next() throws JSONException {
|
||||||
int c;
|
int c;
|
||||||
|
@ -157,26 +200,39 @@ public class JSONTokener {
|
||||||
} catch (IOException exception) {
|
} catch (IOException exception) {
|
||||||
throw new JSONException(exception);
|
throw new JSONException(exception);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
if (c <= 0) { // End of stream
|
if (c <= 0) { // End of stream
|
||||||
this.eof = true;
|
this.eof = true;
|
||||||
c = 0;
|
return 0;
|
||||||
}
|
|
||||||
}
|
|
||||||
this.index += 1;
|
|
||||||
if (this.previous == '\r') {
|
|
||||||
this.line += 1;
|
|
||||||
this.character = c == '\n' ? 0 : 1;
|
|
||||||
} else if (c == '\n') {
|
|
||||||
this.line += 1;
|
|
||||||
this.character = 0;
|
|
||||||
} else {
|
|
||||||
this.character += 1;
|
|
||||||
}
|
}
|
||||||
|
this.incrementIndexes(c);
|
||||||
this.previous = (char) c;
|
this.previous = (char) c;
|
||||||
return this.previous;
|
return this.previous;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Increments the internal indexes according to the previous character
|
||||||
|
* read and the character passed as the current character.
|
||||||
|
* @param c the current character read.
|
||||||
|
*/
|
||||||
|
private void incrementIndexes(int c) {
|
||||||
|
if(c > 0) {
|
||||||
|
this.index++;
|
||||||
|
if(c=='\r') {
|
||||||
|
this.line++;
|
||||||
|
this.characterPreviousLine = this.character;
|
||||||
|
this.character=0;
|
||||||
|
}else if (c=='\n') {
|
||||||
|
if(this.previous != '\r') {
|
||||||
|
this.line++;
|
||||||
|
this.characterPreviousLine = this.character;
|
||||||
|
}
|
||||||
|
this.character=0;
|
||||||
|
} else {
|
||||||
|
this.character++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Consume the next character, and check that it matches a specified
|
* Consume the next character, and check that it matches a specified
|
||||||
|
@ -188,9 +244,12 @@ public class JSONTokener {
|
||||||
public char next(char c) throws JSONException {
|
public char next(char c) throws JSONException {
|
||||||
char n = this.next();
|
char n = this.next();
|
||||||
if (n != c) {
|
if (n != c) {
|
||||||
|
if(n > 0) {
|
||||||
throw this.syntaxError("Expected '" + c + "' and instead saw '" +
|
throw this.syntaxError("Expected '" + c + "' and instead saw '" +
|
||||||
n + "'");
|
n + "'");
|
||||||
}
|
}
|
||||||
|
throw this.syntaxError("Expected '" + c + "' and instead saw ''");
|
||||||
|
}
|
||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -225,7 +284,7 @@ public class JSONTokener {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the next char in the string, skipping whitespace.
|
* Get the next char in the string, skipping whitespace.
|
||||||
* @throws JSONException
|
* @throws JSONException Thrown if there is an error reading the source string.
|
||||||
* @return A character, or 0 if there are no more characters.
|
* @return A character, or 0 if there are no more characters.
|
||||||
*/
|
*/
|
||||||
public char nextClean() throws JSONException {
|
public char nextClean() throws JSONException {
|
||||||
|
@ -278,7 +337,11 @@ public class JSONTokener {
|
||||||
sb.append('\r');
|
sb.append('\r');
|
||||||
break;
|
break;
|
||||||
case 'u':
|
case 'u':
|
||||||
|
try {
|
||||||
sb.append((char)Integer.parseInt(this.next(4), 16));
|
sb.append((char)Integer.parseInt(this.next(4), 16));
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
throw this.syntaxError("Illegal escape.", e);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case '"':
|
case '"':
|
||||||
case '\'':
|
case '\'':
|
||||||
|
@ -305,6 +368,8 @@ public class JSONTokener {
|
||||||
* end of line, whichever comes first.
|
* end of line, whichever comes first.
|
||||||
* @param delimiter A delimiter character.
|
* @param delimiter A delimiter character.
|
||||||
* @return A string.
|
* @return A string.
|
||||||
|
* @throws JSONException Thrown if there is an error while searching
|
||||||
|
* for the delimiter
|
||||||
*/
|
*/
|
||||||
public String nextTo(char delimiter) throws JSONException {
|
public String nextTo(char delimiter) throws JSONException {
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
|
@ -326,6 +391,8 @@ public class JSONTokener {
|
||||||
* characters or the end of line, whichever comes first.
|
* characters or the end of line, whichever comes first.
|
||||||
* @param delimiters A set of delimiter characters.
|
* @param delimiters A set of delimiter characters.
|
||||||
* @return A string, trimmed.
|
* @return A string, trimmed.
|
||||||
|
* @throws JSONException Thrown if there is an error while searching
|
||||||
|
* for the delimiter
|
||||||
*/
|
*/
|
||||||
public String nextTo(String delimiters) throws JSONException {
|
public String nextTo(String delimiters) throws JSONException {
|
||||||
char c;
|
char c;
|
||||||
|
@ -381,7 +448,9 @@ public class JSONTokener {
|
||||||
sb.append(c);
|
sb.append(c);
|
||||||
c = this.next();
|
c = this.next();
|
||||||
}
|
}
|
||||||
|
if (!this.eof) {
|
||||||
this.back();
|
this.back();
|
||||||
|
}
|
||||||
|
|
||||||
string = sb.toString().trim();
|
string = sb.toString().trim();
|
||||||
if ("".equals(string)) {
|
if ("".equals(string)) {
|
||||||
|
@ -397,6 +466,8 @@ public class JSONTokener {
|
||||||
* @param to A character to skip to.
|
* @param to A character to skip to.
|
||||||
* @return The requested character, or zero if the requested character
|
* @return The requested character, or zero if the requested character
|
||||||
* is not found.
|
* is not found.
|
||||||
|
* @throws JSONException Thrown if there is an error while searching
|
||||||
|
* for the to character
|
||||||
*/
|
*/
|
||||||
public char skipTo(char to) throws JSONException {
|
public char skipTo(char to) throws JSONException {
|
||||||
char c;
|
char c;
|
||||||
|
@ -408,13 +479,17 @@ public class JSONTokener {
|
||||||
do {
|
do {
|
||||||
c = this.next();
|
c = this.next();
|
||||||
if (c == 0) {
|
if (c == 0) {
|
||||||
|
// in some readers, reset() may throw an exception if
|
||||||
|
// the remaining portion of the input is greater than
|
||||||
|
// the mark size (1,000,000 above).
|
||||||
this.reader.reset();
|
this.reader.reset();
|
||||||
this.index = startIndex;
|
this.index = startIndex;
|
||||||
this.character = startCharacter;
|
this.character = startCharacter;
|
||||||
this.line = startLine;
|
this.line = startLine;
|
||||||
return c;
|
return 0;
|
||||||
}
|
}
|
||||||
} while (c != to);
|
} while (c != to);
|
||||||
|
this.reader.mark(1);
|
||||||
} catch (IOException exception) {
|
} catch (IOException exception) {
|
||||||
throw new JSONException(exception);
|
throw new JSONException(exception);
|
||||||
}
|
}
|
||||||
|
@ -422,7 +497,6 @@ public class JSONTokener {
|
||||||
return c;
|
return c;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Make a JSONException to signal a syntax error.
|
* Make a JSONException to signal a syntax error.
|
||||||
*
|
*
|
||||||
|
@ -433,12 +507,23 @@ public class JSONTokener {
|
||||||
return new JSONException(message + this.toString());
|
return new JSONException(message + this.toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make a JSONException to signal a syntax error.
|
||||||
|
*
|
||||||
|
* @param message The error message.
|
||||||
|
* @param causedBy The throwable that caused the error.
|
||||||
|
* @return A JSONException object, suitable for throwing
|
||||||
|
*/
|
||||||
|
public JSONException syntaxError(String message, Throwable causedBy) {
|
||||||
|
return new JSONException(message + this.toString(), causedBy);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Make a printable string of this JSONTokener.
|
* Make a printable string of this JSONTokener.
|
||||||
*
|
*
|
||||||
* @return " at {index} [character {character} line {line}]"
|
* @return " at {index} [character {character} line {line}]"
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return " at " + this.index + " [character " + this.character + " line " +
|
return " at " + this.index + " [character " + this.character + " line " +
|
||||||
this.line + "]";
|
this.line + "]";
|
131
JSONWriter.java → src/main/java/org/json/JSONWriter.java
Executable file → Normal file
131
JSONWriter.java → src/main/java/org/json/JSONWriter.java
Executable file → Normal file
|
@ -1,7 +1,8 @@
|
||||||
package org.json;
|
package org.json;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.Writer;
|
import java.util.Collection;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright (c) 2006 JSON.org
|
Copyright (c) 2006 JSON.org
|
||||||
|
@ -50,11 +51,11 @@ SOFTWARE.
|
||||||
* <p>
|
* <p>
|
||||||
* The first method called must be <code>array</code> or <code>object</code>.
|
* The first method called must be <code>array</code> or <code>object</code>.
|
||||||
* There are no methods for adding commas or colons. JSONWriter adds them for
|
* There are no methods for adding commas or colons. JSONWriter adds them for
|
||||||
* you. Objects and arrays can be nested up to 20 levels deep.
|
* you. Objects and arrays can be nested up to 200 levels deep.
|
||||||
* <p>
|
* <p>
|
||||||
* This can sometimes be easier than using a JSONObject to build a string.
|
* This can sometimes be easier than using a JSONObject to build a string.
|
||||||
* @author JSON.org
|
* @author JSON.org
|
||||||
* @version 2011-11-24
|
* @version 2016-08-08
|
||||||
*/
|
*/
|
||||||
public class JSONWriter {
|
public class JSONWriter {
|
||||||
private static final int maxdepth = 200;
|
private static final int maxdepth = 200;
|
||||||
|
@ -88,12 +89,13 @@ public class JSONWriter {
|
||||||
/**
|
/**
|
||||||
* The writer that will receive the output.
|
* The writer that will receive the output.
|
||||||
*/
|
*/
|
||||||
protected Writer writer;
|
protected Appendable writer;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Make a fresh JSONWriter. It can be used to build one JSON text.
|
* Make a fresh JSONWriter. It can be used to build one JSON text.
|
||||||
|
* @param w an appendable object
|
||||||
*/
|
*/
|
||||||
public JSONWriter(Writer w) {
|
public JSONWriter(Appendable w) {
|
||||||
this.comma = false;
|
this.comma = false;
|
||||||
this.mode = 'i';
|
this.mode = 'i';
|
||||||
this.stack = new JSONObject[maxdepth];
|
this.stack = new JSONObject[maxdepth];
|
||||||
|
@ -114,10 +116,13 @@ public class JSONWriter {
|
||||||
if (this.mode == 'o' || this.mode == 'a') {
|
if (this.mode == 'o' || this.mode == 'a') {
|
||||||
try {
|
try {
|
||||||
if (this.comma && this.mode == 'a') {
|
if (this.comma && this.mode == 'a') {
|
||||||
this.writer.write(',');
|
this.writer.append(',');
|
||||||
}
|
}
|
||||||
this.writer.write(string);
|
this.writer.append(string);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
|
// Android as of API 25 does not support this exception constructor
|
||||||
|
// however we won't worry about it. If an exception is happening here
|
||||||
|
// it will just throw a "Method not found" exception instead.
|
||||||
throw new JSONException(e);
|
throw new JSONException(e);
|
||||||
}
|
}
|
||||||
if (this.mode == 'o') {
|
if (this.mode == 'o') {
|
||||||
|
@ -150,21 +155,24 @@ public class JSONWriter {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* End something.
|
* End something.
|
||||||
* @param mode Mode
|
* @param m Mode
|
||||||
* @param c Closing character
|
* @param c Closing character
|
||||||
* @return this
|
* @return this
|
||||||
* @throws JSONException If unbalanced.
|
* @throws JSONException If unbalanced.
|
||||||
*/
|
*/
|
||||||
private JSONWriter end(char mode, char c) throws JSONException {
|
private JSONWriter end(char m, char c) throws JSONException {
|
||||||
if (this.mode != mode) {
|
if (this.mode != m) {
|
||||||
throw new JSONException(mode == 'a'
|
throw new JSONException(m == 'a'
|
||||||
? "Misplaced endArray."
|
? "Misplaced endArray."
|
||||||
: "Misplaced endObject.");
|
: "Misplaced endObject.");
|
||||||
}
|
}
|
||||||
this.pop(mode);
|
this.pop(m);
|
||||||
try {
|
try {
|
||||||
this.writer.write(c);
|
this.writer.append(c);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
|
// Android as of API 25 does not support this exception constructor
|
||||||
|
// however we won't worry about it. If an exception is happening here
|
||||||
|
// it will just throw a "Method not found" exception instead.
|
||||||
throw new JSONException(e);
|
throw new JSONException(e);
|
||||||
}
|
}
|
||||||
this.comma = true;
|
this.comma = true;
|
||||||
|
@ -205,16 +213,24 @@ public class JSONWriter {
|
||||||
}
|
}
|
||||||
if (this.mode == 'k') {
|
if (this.mode == 'k') {
|
||||||
try {
|
try {
|
||||||
this.stack[this.top - 1].putOnce(string, Boolean.TRUE);
|
JSONObject topObject = this.stack[this.top - 1];
|
||||||
if (this.comma) {
|
// don't use the built in putOnce method to maintain Android support
|
||||||
this.writer.write(',');
|
if(topObject.has(string)) {
|
||||||
|
throw new JSONException("Duplicate key \"" + string + "\"");
|
||||||
}
|
}
|
||||||
this.writer.write(JSONObject.quote(string));
|
topObject.put(string, true);
|
||||||
this.writer.write(':');
|
if (this.comma) {
|
||||||
|
this.writer.append(',');
|
||||||
|
}
|
||||||
|
this.writer.append(JSONObject.quote(string));
|
||||||
|
this.writer.append(':');
|
||||||
this.comma = false;
|
this.comma = false;
|
||||||
this.mode = 'o';
|
this.mode = 'o';
|
||||||
return this;
|
return this;
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
|
// Android as of API 25 does not support this exception constructor
|
||||||
|
// however we won't worry about it. If an exception is happening here
|
||||||
|
// it will just throw a "Method not found" exception instead.
|
||||||
throw new JSONException(e);
|
throw new JSONException(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -281,13 +297,84 @@ public class JSONWriter {
|
||||||
this.top += 1;
|
this.top += 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make a JSON text of an Object value. If the object has an
|
||||||
|
* value.toJSONString() method, then that method will be used to produce the
|
||||||
|
* JSON text. The method is required to produce a strictly conforming text.
|
||||||
|
* If the object does not contain a toJSONString method (which is the most
|
||||||
|
* common case), then a text will be produced by other means. If the value
|
||||||
|
* is an array or Collection, then a JSONArray will be made from it and its
|
||||||
|
* toJSONString method will be called. If the value is a MAP, then a
|
||||||
|
* JSONObject will be made from it and its toJSONString method will be
|
||||||
|
* called. Otherwise, the value's toString method will be called, and the
|
||||||
|
* result will be quoted.
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* Warning: This method assumes that the data structure is acyclical.
|
||||||
|
*
|
||||||
|
* @param value
|
||||||
|
* The value to be serialized.
|
||||||
|
* @return a printable, displayable, transmittable representation of the
|
||||||
|
* object, beginning with <code>{</code> <small>(left
|
||||||
|
* brace)</small> and ending with <code>}</code> <small>(right
|
||||||
|
* brace)</small>.
|
||||||
|
* @throws JSONException
|
||||||
|
* If the value is or contains an invalid number.
|
||||||
|
*/
|
||||||
|
public static String valueToString(Object value) throws JSONException {
|
||||||
|
if (value == null || value.equals(null)) {
|
||||||
|
return "null";
|
||||||
|
}
|
||||||
|
if (value instanceof JSONString) {
|
||||||
|
String object;
|
||||||
|
try {
|
||||||
|
object = ((JSONString) value).toJSONString();
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw new JSONException(e);
|
||||||
|
}
|
||||||
|
if (object != null) {
|
||||||
|
return object;
|
||||||
|
}
|
||||||
|
throw new JSONException("Bad value from toJSONString: " + object);
|
||||||
|
}
|
||||||
|
if (value instanceof Number) {
|
||||||
|
// not all Numbers may match actual JSON Numbers. i.e. Fractions or Complex
|
||||||
|
final String numberAsString = JSONObject.numberToString((Number) value);
|
||||||
|
if(JSONObject.NUMBER_PATTERN.matcher(numberAsString).matches()) {
|
||||||
|
// Close enough to a JSON number that we will return it unquoted
|
||||||
|
return numberAsString;
|
||||||
|
}
|
||||||
|
// The Number value is not a valid JSON number.
|
||||||
|
// Instead we will quote it as a string
|
||||||
|
return JSONObject.quote(numberAsString);
|
||||||
|
}
|
||||||
|
if (value instanceof Boolean || value instanceof JSONObject
|
||||||
|
|| value instanceof JSONArray) {
|
||||||
|
return value.toString();
|
||||||
|
}
|
||||||
|
if (value instanceof Map) {
|
||||||
|
Map<?, ?> map = (Map<?, ?>) value;
|
||||||
|
return new JSONObject(map).toString();
|
||||||
|
}
|
||||||
|
if (value instanceof Collection) {
|
||||||
|
Collection<?> coll = (Collection<?>) value;
|
||||||
|
return new JSONArray(coll).toString();
|
||||||
|
}
|
||||||
|
if (value.getClass().isArray()) {
|
||||||
|
return new JSONArray(value).toString();
|
||||||
|
}
|
||||||
|
if(value instanceof Enum<?>){
|
||||||
|
return JSONObject.quote(((Enum<?>)value).name());
|
||||||
|
}
|
||||||
|
return JSONObject.quote(value.toString());
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Append either the value <code>true</code> or the value
|
* Append either the value <code>true</code> or the value
|
||||||
* <code>false</code>.
|
* <code>false</code>.
|
||||||
* @param b A boolean.
|
* @param b A boolean.
|
||||||
* @return this
|
* @return this
|
||||||
* @throws JSONException
|
* @throws JSONException if a called function has an error
|
||||||
*/
|
*/
|
||||||
public JSONWriter value(boolean b) throws JSONException {
|
public JSONWriter value(boolean b) throws JSONException {
|
||||||
return this.append(b ? "true" : "false");
|
return this.append(b ? "true" : "false");
|
||||||
|
@ -300,14 +387,14 @@ public class JSONWriter {
|
||||||
* @throws JSONException If the number is not finite.
|
* @throws JSONException If the number is not finite.
|
||||||
*/
|
*/
|
||||||
public JSONWriter value(double d) throws JSONException {
|
public JSONWriter value(double d) throws JSONException {
|
||||||
return this.value(new Double(d));
|
return this.value(Double.valueOf(d));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Append a long value.
|
* Append a long value.
|
||||||
* @param l A long.
|
* @param l A long.
|
||||||
* @return this
|
* @return this
|
||||||
* @throws JSONException
|
* @throws JSONException if a called function has an error
|
||||||
*/
|
*/
|
||||||
public JSONWriter value(long l) throws JSONException {
|
public JSONWriter value(long l) throws JSONException {
|
||||||
return this.append(Long.toString(l));
|
return this.append(Long.toString(l));
|
||||||
|
@ -322,6 +409,6 @@ public class JSONWriter {
|
||||||
* @throws JSONException If the value is out of sequence.
|
* @throws JSONException If the value is out of sequence.
|
||||||
*/
|
*/
|
||||||
public JSONWriter value(Object object) throws JSONException {
|
public JSONWriter value(Object object) throws JSONException {
|
||||||
return this.append(JSONObject.valueToString(object));
|
return this.append(valueToString(object));
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -25,7 +25,6 @@ SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.util.Enumeration;
|
import java.util.Enumeration;
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -38,9 +37,11 @@ public class 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
|
||||||
* @return JSONObject
|
* @return JSONObject
|
||||||
* @throws JSONException
|
* @throws JSONException if a called function has an error
|
||||||
*/
|
*/
|
||||||
public static JSONObject toJSONObject(java.util.Properties properties) throws JSONException {
|
public static JSONObject toJSONObject(java.util.Properties properties) throws JSONException {
|
||||||
|
// can't use the new constructor for Android support
|
||||||
|
// JSONObject jo = new JSONObject(properties == null ? 0 : properties.size());
|
||||||
JSONObject jo = new JSONObject();
|
JSONObject jo = new JSONObject();
|
||||||
if (properties != null && !properties.isEmpty()) {
|
if (properties != null && !properties.isEmpty()) {
|
||||||
Enumeration<?> enumProperties = properties.propertyNames();
|
Enumeration<?> enumProperties = properties.propertyNames();
|
||||||
|
@ -56,15 +57,17 @@ public class Property {
|
||||||
* Converts the JSONObject into a property file object.
|
* Converts the JSONObject into a property file object.
|
||||||
* @param jo JSONObject
|
* @param jo JSONObject
|
||||||
* @return java.util.Properties
|
* @return java.util.Properties
|
||||||
* @throws JSONException
|
* @throws JSONException if a called function has an error
|
||||||
*/
|
*/
|
||||||
public static Properties toProperties(JSONObject jo) throws JSONException {
|
public static Properties toProperties(JSONObject jo) throws JSONException {
|
||||||
Properties properties = new Properties();
|
Properties properties = new Properties();
|
||||||
if (jo != null) {
|
if (jo != null) {
|
||||||
Iterator<String> keys = jo.keys();
|
// Don't use the new entrySet API to maintain Android support
|
||||||
while (keys.hasNext()) {
|
for (final String key : jo.keySet()) {
|
||||||
String name = keys.next();
|
Object value = jo.opt(key);
|
||||||
properties.put(name, jo.getString(name));
|
if (!JSONObject.NULL.equals(value)) {
|
||||||
|
properties.put(key, value.toString());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return properties;
|
return properties;
|
833
src/main/java/org/json/XML.java
Normal file
833
src/main/java/org/json/XML.java
Normal file
|
@ -0,0 +1,833 @@
|
||||||
|
package org.json;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2015 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.io.Reader;
|
||||||
|
import java.io.StringReader;
|
||||||
|
import java.math.BigDecimal;
|
||||||
|
import java.math.BigInteger;
|
||||||
|
import java.util.Iterator;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This provides static methods to convert an XML text into a JSONObject, and to
|
||||||
|
* covert a JSONObject into an XML text.
|
||||||
|
*
|
||||||
|
* @author JSON.org
|
||||||
|
* @version 2016-08-10
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("boxing")
|
||||||
|
public class XML {
|
||||||
|
|
||||||
|
/** The Character '&'. */
|
||||||
|
public static final Character AMP = '&';
|
||||||
|
|
||||||
|
/** The Character '''. */
|
||||||
|
public static final Character APOS = '\'';
|
||||||
|
|
||||||
|
/** The Character '!'. */
|
||||||
|
public static final Character BANG = '!';
|
||||||
|
|
||||||
|
/** The Character '='. */
|
||||||
|
public static final Character EQ = '=';
|
||||||
|
|
||||||
|
/** The Character <pre>{@code '>'. }</pre>*/
|
||||||
|
public static final Character GT = '>';
|
||||||
|
|
||||||
|
/** The Character '<'. */
|
||||||
|
public static final Character LT = '<';
|
||||||
|
|
||||||
|
/** The Character '?'. */
|
||||||
|
public static final Character QUEST = '?';
|
||||||
|
|
||||||
|
/** The Character '"'. */
|
||||||
|
public static final Character QUOT = '"';
|
||||||
|
|
||||||
|
/** The Character '/'. */
|
||||||
|
public static final Character SLASH = '/';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Null attribute name
|
||||||
|
*/
|
||||||
|
public static final String NULL_ATTR = "xsi:nil";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates an iterator for navigating Code Points in a string instead of
|
||||||
|
* characters. Once Java7 support is dropped, this can be replaced with
|
||||||
|
* <code>
|
||||||
|
* string.codePoints()
|
||||||
|
* </code>
|
||||||
|
* which is available in Java8 and above.
|
||||||
|
*
|
||||||
|
* @see <a href=
|
||||||
|
* "http://stackoverflow.com/a/21791059/6030888">http://stackoverflow.com/a/21791059/6030888</a>
|
||||||
|
*/
|
||||||
|
private static Iterable<Integer> codePointIterator(final String string) {
|
||||||
|
return new Iterable<Integer>() {
|
||||||
|
@Override
|
||||||
|
public Iterator<Integer> iterator() {
|
||||||
|
return new Iterator<Integer>() {
|
||||||
|
private int nextIndex = 0;
|
||||||
|
private int length = string.length();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean hasNext() {
|
||||||
|
return this.nextIndex < this.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Integer next() {
|
||||||
|
int result = string.codePointAt(this.nextIndex);
|
||||||
|
this.nextIndex += Character.charCount(result);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void remove() {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Replace special characters with XML escapes:
|
||||||
|
*
|
||||||
|
* <pre>{@code
|
||||||
|
* & (ampersand) is replaced by &amp;
|
||||||
|
* < (less than) is replaced by &lt;
|
||||||
|
* > (greater than) is replaced by &gt;
|
||||||
|
* " (double quote) is replaced by &quot;
|
||||||
|
* ' (single quote / apostrophe) is replaced by &apos;
|
||||||
|
* }</pre>
|
||||||
|
*
|
||||||
|
* @param string
|
||||||
|
* The string to be escaped.
|
||||||
|
* @return The escaped string.
|
||||||
|
*/
|
||||||
|
public static String escape(String string) {
|
||||||
|
StringBuilder sb = new StringBuilder(string.length());
|
||||||
|
for (final int cp : codePointIterator(string)) {
|
||||||
|
switch (cp) {
|
||||||
|
case '&':
|
||||||
|
sb.append("&");
|
||||||
|
break;
|
||||||
|
case '<':
|
||||||
|
sb.append("<");
|
||||||
|
break;
|
||||||
|
case '>':
|
||||||
|
sb.append(">");
|
||||||
|
break;
|
||||||
|
case '"':
|
||||||
|
sb.append(""");
|
||||||
|
break;
|
||||||
|
case '\'':
|
||||||
|
sb.append("'");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
if (mustEscape(cp)) {
|
||||||
|
sb.append("&#x");
|
||||||
|
sb.append(Integer.toHexString(cp));
|
||||||
|
sb.append(';');
|
||||||
|
} else {
|
||||||
|
sb.appendCodePoint(cp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param cp code point to test
|
||||||
|
* @return true if the code point is not valid for an XML
|
||||||
|
*/
|
||||||
|
private static boolean mustEscape(int cp) {
|
||||||
|
/* Valid range from https://www.w3.org/TR/REC-xml/#charsets
|
||||||
|
*
|
||||||
|
* #x9 | #xA | #xD | [#x20-#xD7FF] | [#xE000-#xFFFD] | [#x10000-#x10FFFF]
|
||||||
|
*
|
||||||
|
* any Unicode character, excluding the surrogate blocks, FFFE, and FFFF.
|
||||||
|
*/
|
||||||
|
// isISOControl is true when (cp >= 0 && cp <= 0x1F) || (cp >= 0x7F && cp <= 0x9F)
|
||||||
|
// all ISO control characters are out of range except tabs and new lines
|
||||||
|
return (Character.isISOControl(cp)
|
||||||
|
&& cp != 0x9
|
||||||
|
&& cp != 0xA
|
||||||
|
&& cp != 0xD
|
||||||
|
) || !(
|
||||||
|
// valid the range of acceptable characters that aren't control
|
||||||
|
(cp >= 0x20 && cp <= 0xD7FF)
|
||||||
|
|| (cp >= 0xE000 && cp <= 0xFFFD)
|
||||||
|
|| (cp >= 0x10000 && cp <= 0x10FFFF)
|
||||||
|
)
|
||||||
|
;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes XML escapes from the string.
|
||||||
|
*
|
||||||
|
* @param string
|
||||||
|
* string to remove escapes from
|
||||||
|
* @return string with converted entities
|
||||||
|
*/
|
||||||
|
public static String unescape(String string) {
|
||||||
|
StringBuilder sb = new StringBuilder(string.length());
|
||||||
|
for (int i = 0, length = string.length(); i < length; i++) {
|
||||||
|
char c = string.charAt(i);
|
||||||
|
if (c == '&') {
|
||||||
|
final int semic = string.indexOf(';', i);
|
||||||
|
if (semic > i) {
|
||||||
|
final String entity = string.substring(i + 1, semic);
|
||||||
|
sb.append(XMLTokener.unescapeEntity(entity));
|
||||||
|
// skip past the entity we just parsed.
|
||||||
|
i += entity.length() + 1;
|
||||||
|
} else {
|
||||||
|
// this shouldn't happen in most cases since the parser
|
||||||
|
// errors on unclosed entries.
|
||||||
|
sb.append(c);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// not part of an entity
|
||||||
|
sb.append(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Throw an exception if the string contains whitespace. Whitespace is not
|
||||||
|
* allowed in tagNames and attributes.
|
||||||
|
*
|
||||||
|
* @param string
|
||||||
|
* A string.
|
||||||
|
* @throws JSONException Thrown if the string contains whitespace or is empty.
|
||||||
|
*/
|
||||||
|
public static void noSpace(String string) throws JSONException {
|
||||||
|
int i, length = string.length();
|
||||||
|
if (length == 0) {
|
||||||
|
throw new JSONException("Empty string.");
|
||||||
|
}
|
||||||
|
for (i = 0; i < length; i += 1) {
|
||||||
|
if (Character.isWhitespace(string.charAt(i))) {
|
||||||
|
throw new JSONException("'" + string
|
||||||
|
+ "' contains a space character.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Scan the content following the named tag, attaching it to the context.
|
||||||
|
*
|
||||||
|
* @param x
|
||||||
|
* The XMLTokener containing the source string.
|
||||||
|
* @param context
|
||||||
|
* The JSONObject that will include the new material.
|
||||||
|
* @param name
|
||||||
|
* The tag name.
|
||||||
|
* @return true if the close tag is processed.
|
||||||
|
* @throws JSONException
|
||||||
|
*/
|
||||||
|
private static boolean parse(XMLTokener x, JSONObject context, String name, XMLParserConfiguration config)
|
||||||
|
throws JSONException {
|
||||||
|
char c;
|
||||||
|
int i;
|
||||||
|
JSONObject jsonObject = null;
|
||||||
|
String string;
|
||||||
|
String tagName;
|
||||||
|
Object token;
|
||||||
|
|
||||||
|
// Test for and skip past these forms:
|
||||||
|
// <!-- ... -->
|
||||||
|
// <! ... >
|
||||||
|
// <![ ... ]]>
|
||||||
|
// <? ... ?>
|
||||||
|
// Report errors for these forms:
|
||||||
|
// <>
|
||||||
|
// <=
|
||||||
|
// <<
|
||||||
|
|
||||||
|
token = x.nextToken();
|
||||||
|
|
||||||
|
// <!
|
||||||
|
|
||||||
|
if (token == BANG) {
|
||||||
|
c = x.next();
|
||||||
|
if (c == '-') {
|
||||||
|
if (x.next() == '-') {
|
||||||
|
x.skipPast("-->");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
x.back();
|
||||||
|
} else if (c == '[') {
|
||||||
|
token = x.nextToken();
|
||||||
|
if ("CDATA".equals(token)) {
|
||||||
|
if (x.next() == '[') {
|
||||||
|
string = x.nextCDATA();
|
||||||
|
if (string.length() > 0) {
|
||||||
|
context.accumulate(config.cDataTagName, string);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw x.syntaxError("Expected 'CDATA['");
|
||||||
|
}
|
||||||
|
i = 1;
|
||||||
|
do {
|
||||||
|
token = x.nextMeta();
|
||||||
|
if (token == null) {
|
||||||
|
throw x.syntaxError("Missing '>' after '<!'.");
|
||||||
|
} else if (token == LT) {
|
||||||
|
i += 1;
|
||||||
|
} else if (token == GT) {
|
||||||
|
i -= 1;
|
||||||
|
}
|
||||||
|
} while (i > 0);
|
||||||
|
return false;
|
||||||
|
} else if (token == QUEST) {
|
||||||
|
|
||||||
|
// <?
|
||||||
|
x.skipPast("?>");
|
||||||
|
return false;
|
||||||
|
} else if (token == SLASH) {
|
||||||
|
|
||||||
|
// Close tag </
|
||||||
|
|
||||||
|
token = x.nextToken();
|
||||||
|
if (name == null) {
|
||||||
|
throw x.syntaxError("Mismatched close tag " + token);
|
||||||
|
}
|
||||||
|
if (!token.equals(name)) {
|
||||||
|
throw x.syntaxError("Mismatched " + name + " and " + token);
|
||||||
|
}
|
||||||
|
if (x.nextToken() != GT) {
|
||||||
|
throw x.syntaxError("Misshaped close tag");
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
|
||||||
|
} else if (token instanceof Character) {
|
||||||
|
throw x.syntaxError("Misshaped tag");
|
||||||
|
|
||||||
|
// Open tag <
|
||||||
|
|
||||||
|
} else {
|
||||||
|
tagName = (String) token;
|
||||||
|
token = null;
|
||||||
|
jsonObject = new JSONObject();
|
||||||
|
boolean nilAttributeFound = false;
|
||||||
|
for (;;) {
|
||||||
|
if (token == null) {
|
||||||
|
token = x.nextToken();
|
||||||
|
}
|
||||||
|
// attribute = value
|
||||||
|
if (token instanceof String) {
|
||||||
|
string = (String) token;
|
||||||
|
token = x.nextToken();
|
||||||
|
if (token == EQ) {
|
||||||
|
token = x.nextToken();
|
||||||
|
if (!(token instanceof String)) {
|
||||||
|
throw x.syntaxError("Missing value");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (config.convertNilAttributeToNull
|
||||||
|
&& NULL_ATTR.equals(string)
|
||||||
|
&& Boolean.parseBoolean((String) token)) {
|
||||||
|
nilAttributeFound = true;
|
||||||
|
} else if (!nilAttributeFound) {
|
||||||
|
jsonObject.accumulate(string,
|
||||||
|
config.keepStrings
|
||||||
|
? ((String) token)
|
||||||
|
: stringToValue((String) token));
|
||||||
|
}
|
||||||
|
token = null;
|
||||||
|
} else {
|
||||||
|
jsonObject.accumulate(string, "");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
} else if (token == SLASH) {
|
||||||
|
// Empty tag <.../>
|
||||||
|
if (x.nextToken() != GT) {
|
||||||
|
throw x.syntaxError("Misshaped tag");
|
||||||
|
}
|
||||||
|
if (nilAttributeFound) {
|
||||||
|
context.accumulate(tagName, JSONObject.NULL);
|
||||||
|
} else if (jsonObject.length() > 0) {
|
||||||
|
context.accumulate(tagName, jsonObject);
|
||||||
|
} else {
|
||||||
|
context.accumulate(tagName, "");
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
|
||||||
|
} else if (token == GT) {
|
||||||
|
// Content, between <...> and </...>
|
||||||
|
for (;;) {
|
||||||
|
token = x.nextContent();
|
||||||
|
if (token == null) {
|
||||||
|
if (tagName != null) {
|
||||||
|
throw x.syntaxError("Unclosed tag " + tagName);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
} else if (token instanceof String) {
|
||||||
|
string = (String) token;
|
||||||
|
if (string.length() > 0) {
|
||||||
|
jsonObject.accumulate(config.cDataTagName,
|
||||||
|
config.keepStrings ? string : stringToValue(string));
|
||||||
|
}
|
||||||
|
|
||||||
|
} else if (token == LT) {
|
||||||
|
// Nested element
|
||||||
|
if (parse(x, jsonObject, tagName, config)) {
|
||||||
|
if (jsonObject.length() == 0) {
|
||||||
|
context.accumulate(tagName, "");
|
||||||
|
} else if (jsonObject.length() == 1
|
||||||
|
&& jsonObject.opt(config.cDataTagName) != null) {
|
||||||
|
context.accumulate(tagName, jsonObject.opt(config.cDataTagName));
|
||||||
|
} else {
|
||||||
|
context.accumulate(tagName, jsonObject);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
throw x.syntaxError("Misshaped tag");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 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;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* direct copy of {@link JSONObject#stringToNumber(String)} to maintain Android support.
|
||||||
|
*/
|
||||||
|
private static Number stringToNumber(final String val) throws NumberFormatException {
|
||||||
|
char initial = val.charAt(0);
|
||||||
|
if ((initial >= '0' && initial <= '9') || initial == '-') {
|
||||||
|
// decimal representation
|
||||||
|
if (isDecimalNotation(val)) {
|
||||||
|
// Use a BigDecimal all the time so we keep the original
|
||||||
|
// representation. BigDecimal doesn't support -0.0, ensure we
|
||||||
|
// keep that by forcing a decimal.
|
||||||
|
try {
|
||||||
|
BigDecimal bd = new BigDecimal(val);
|
||||||
|
if(initial == '-' && BigDecimal.ZERO.compareTo(bd)==0) {
|
||||||
|
return Double.valueOf(-0.0);
|
||||||
|
}
|
||||||
|
return bd;
|
||||||
|
} catch (NumberFormatException retryAsDouble) {
|
||||||
|
// this is to support "Hex Floats" like this: 0x1.0P-1074
|
||||||
|
try {
|
||||||
|
Double d = Double.valueOf(val);
|
||||||
|
if(d.isNaN() || d.isInfinite()) {
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
return d;
|
||||||
|
} catch (NumberFormatException ignore) {
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// block items like 00 01 etc. Java number parsers treat these as Octal.
|
||||||
|
if(initial == '0' && val.length() > 1) {
|
||||||
|
char at1 = val.charAt(1);
|
||||||
|
if(at1 >= '0' && at1 <= '9') {
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
} else if (initial == '-' && val.length() > 2) {
|
||||||
|
char at1 = val.charAt(1);
|
||||||
|
char at2 = val.charAt(2);
|
||||||
|
if(at1 == '0' && at2 >= '0' && at2 <= '9') {
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// integer representation.
|
||||||
|
// This will narrow any values to the smallest reasonable Object representation
|
||||||
|
// (Integer, Long, or BigInteger)
|
||||||
|
|
||||||
|
// BigInteger down conversion: We use a similar bitLenth compare as
|
||||||
|
// BigInteger#intValueExact uses. Increases GC, but objects hold
|
||||||
|
// only what they need. i.e. Less runtime overhead if the value is
|
||||||
|
// long lived.
|
||||||
|
BigInteger bi = new BigInteger(val);
|
||||||
|
if(bi.bitLength() <= 31){
|
||||||
|
return Integer.valueOf(bi.intValue());
|
||||||
|
}
|
||||||
|
if(bi.bitLength() <= 63){
|
||||||
|
return Long.valueOf(bi.longValue());
|
||||||
|
}
|
||||||
|
return bi;
|
||||||
|
}
|
||||||
|
throw new NumberFormatException("val ["+val+"] is not a valid number.");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* direct copy of {@link JSONObject#isDecimalNotation(String)} to maintain Android support.
|
||||||
|
*/
|
||||||
|
private static boolean isDecimalNotation(final String val) {
|
||||||
|
return val.indexOf('.') > -1 || val.indexOf('e') > -1
|
||||||
|
|| val.indexOf('E') > -1 || "-0".equals(val);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
|
* JSONObject. Some information may be lost in this transformation because
|
||||||
|
* JSON is a data format and XML is a document format. XML uses elements,
|
||||||
|
* attributes, and content text, while JSON uses unordered collections of
|
||||||
|
* name/value pairs and arrays of values. JSON does not does not like to
|
||||||
|
* distinguish between elements and attributes. Sequences of similar
|
||||||
|
* elements are represented as JSONArrays. Content text may be placed in a
|
||||||
|
* "content" member. Comments, prologs, DTDs, and <pre>{@code
|
||||||
|
* <[ [ ]]>}</pre>
|
||||||
|
* are ignored.
|
||||||
|
*
|
||||||
|
* @param string
|
||||||
|
* The source string.
|
||||||
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown if there is an errors while parsing the string
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(String string) throws JSONException {
|
||||||
|
return toJSONObject(string, XMLParserConfiguration.ORIGINAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML into a
|
||||||
|
* JSONObject. Some information may be lost in this transformation because
|
||||||
|
* JSON is a data format and XML is a document format. XML uses elements,
|
||||||
|
* attributes, and content text, while JSON uses unordered collections of
|
||||||
|
* name/value pairs and arrays of values. JSON does not does not like to
|
||||||
|
* distinguish between elements and attributes. Sequences of similar
|
||||||
|
* elements are represented as JSONArrays. Content text may be placed in a
|
||||||
|
* "content" member. Comments, prologs, DTDs, and <pre>{@code
|
||||||
|
* <[ [ ]]>}</pre>
|
||||||
|
* are ignored.
|
||||||
|
*
|
||||||
|
* @param reader The XML source reader.
|
||||||
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown if there is an errors while parsing the string
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(Reader reader) throws JSONException {
|
||||||
|
return toJSONObject(reader, XMLParserConfiguration.ORIGINAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML into a
|
||||||
|
* JSONObject. Some information may be lost in this transformation because
|
||||||
|
* JSON is a data format and XML is a document format. XML uses elements,
|
||||||
|
* attributes, and content text, while JSON uses unordered collections of
|
||||||
|
* name/value pairs and arrays of values. JSON does not does not like to
|
||||||
|
* distinguish between elements and attributes. Sequences of similar
|
||||||
|
* elements are represented as JSONArrays. Content text may be placed in a
|
||||||
|
* "content" member. Comments, prologs, DTDs, and <pre>{@code
|
||||||
|
* <[ [ ]]>}</pre>
|
||||||
|
* are ignored.
|
||||||
|
*
|
||||||
|
* All values are converted as strings, for 1, 01, 29.0 will not be coerced to
|
||||||
|
* numbers but will instead be the exact value as seen in the XML document.
|
||||||
|
*
|
||||||
|
* @param reader The XML source reader.
|
||||||
|
* @param keepStrings If true, then values will not be coerced into boolean
|
||||||
|
* or numeric values and will instead be left as strings
|
||||||
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown if there is an errors while parsing the string
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(Reader reader, boolean keepStrings) throws JSONException {
|
||||||
|
if(keepStrings) {
|
||||||
|
return toJSONObject(reader, XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
}
|
||||||
|
return toJSONObject(reader, XMLParserConfiguration.ORIGINAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML into a
|
||||||
|
* JSONObject. Some information may be lost in this transformation because
|
||||||
|
* JSON is a data format and XML is a document format. XML uses elements,
|
||||||
|
* attributes, and content text, while JSON uses unordered collections of
|
||||||
|
* name/value pairs and arrays of values. JSON does not does not like to
|
||||||
|
* distinguish between elements and attributes. Sequences of similar
|
||||||
|
* elements are represented as JSONArrays. Content text may be placed in a
|
||||||
|
* "content" member. Comments, prologs, DTDs, and <pre>{@code
|
||||||
|
* <[ [ ]]>}</pre>
|
||||||
|
* are ignored.
|
||||||
|
*
|
||||||
|
* All values are converted as strings, for 1, 01, 29.0 will not be coerced to
|
||||||
|
* numbers but will instead be the exact value as seen in the XML document.
|
||||||
|
*
|
||||||
|
* @param reader The XML source reader.
|
||||||
|
* @param config Configuration options for the parser
|
||||||
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown if there is an errors while parsing the string
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(Reader reader, XMLParserConfiguration config) throws JSONException {
|
||||||
|
JSONObject jo = new JSONObject();
|
||||||
|
XMLTokener x = new XMLTokener(reader);
|
||||||
|
while (x.more()) {
|
||||||
|
x.skipPast("<");
|
||||||
|
if(x.more()) {
|
||||||
|
parse(x, jo, null, config);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return jo;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
|
* JSONObject. Some information may be lost in this transformation because
|
||||||
|
* JSON is a data format and XML is a document format. XML uses elements,
|
||||||
|
* attributes, and content text, while JSON uses unordered collections of
|
||||||
|
* name/value pairs and arrays of values. JSON does not does not like to
|
||||||
|
* distinguish between elements and attributes. Sequences of similar
|
||||||
|
* elements are represented as JSONArrays. Content text may be placed in a
|
||||||
|
* "content" member. Comments, prologs, DTDs, and <pre>{@code
|
||||||
|
* <[ [ ]]>}</pre>
|
||||||
|
* are ignored.
|
||||||
|
*
|
||||||
|
* All values are converted as strings, for 1, 01, 29.0 will not be coerced to
|
||||||
|
* numbers but will instead be the exact value as seen in the XML document.
|
||||||
|
*
|
||||||
|
* @param string
|
||||||
|
* The source string.
|
||||||
|
* @param keepStrings If true, then values will not be coerced into boolean
|
||||||
|
* or numeric values and will instead be left as strings
|
||||||
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown if there is an errors while parsing the string
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(String string, boolean keepStrings) throws JSONException {
|
||||||
|
return toJSONObject(new StringReader(string), keepStrings);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a well-formed (but not necessarily valid) XML string into a
|
||||||
|
* JSONObject. Some information may be lost in this transformation because
|
||||||
|
* JSON is a data format and XML is a document format. XML uses elements,
|
||||||
|
* attributes, and content text, while JSON uses unordered collections of
|
||||||
|
* name/value pairs and arrays of values. JSON does not does not like to
|
||||||
|
* distinguish between elements and attributes. Sequences of similar
|
||||||
|
* elements are represented as JSONArrays. Content text may be placed in a
|
||||||
|
* "content" member. Comments, prologs, DTDs, and <pre>{@code
|
||||||
|
* <[ [ ]]>}</pre>
|
||||||
|
* are ignored.
|
||||||
|
*
|
||||||
|
* All values are converted as strings, for 1, 01, 29.0 will not be coerced to
|
||||||
|
* numbers but will instead be the exact value as seen in the XML document.
|
||||||
|
*
|
||||||
|
* @param string
|
||||||
|
* The source string.
|
||||||
|
* @param config Configuration options for the parser.
|
||||||
|
* @return A JSONObject containing the structured data from the XML string.
|
||||||
|
* @throws JSONException Thrown if there is an errors while parsing the string
|
||||||
|
*/
|
||||||
|
public static JSONObject toJSONObject(String string, XMLParserConfiguration config) throws JSONException {
|
||||||
|
return toJSONObject(new StringReader(string), config);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a JSONObject into a well-formed, element-normal XML string.
|
||||||
|
*
|
||||||
|
* @param object
|
||||||
|
* A JSONObject.
|
||||||
|
* @return A string.
|
||||||
|
* @throws JSONException Thrown if there is an error parsing the string
|
||||||
|
*/
|
||||||
|
public static String toString(Object object) throws JSONException {
|
||||||
|
return toString(object, null, XMLParserConfiguration.ORIGINAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a JSONObject into a well-formed, element-normal XML string.
|
||||||
|
*
|
||||||
|
* @param object
|
||||||
|
* A JSONObject.
|
||||||
|
* @param tagName
|
||||||
|
* The optional name of the enclosing tag.
|
||||||
|
* @return A string.
|
||||||
|
* @throws JSONException Thrown if there is an error parsing the string
|
||||||
|
*/
|
||||||
|
public static String toString(final Object object, final String tagName) {
|
||||||
|
return toString(object, tagName, XMLParserConfiguration.ORIGINAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a JSONObject into a well-formed, element-normal XML string.
|
||||||
|
*
|
||||||
|
* @param object
|
||||||
|
* A JSONObject.
|
||||||
|
* @param tagName
|
||||||
|
* The optional name of the enclosing tag.
|
||||||
|
* @param config
|
||||||
|
* Configuration that can control output to XML.
|
||||||
|
* @return A string.
|
||||||
|
* @throws JSONException Thrown if there is an error parsing the string
|
||||||
|
*/
|
||||||
|
public static String toString(final Object object, final String tagName, final XMLParserConfiguration config)
|
||||||
|
throws JSONException {
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
JSONArray ja;
|
||||||
|
JSONObject jo;
|
||||||
|
String string;
|
||||||
|
|
||||||
|
if (object instanceof JSONObject) {
|
||||||
|
|
||||||
|
// Emit <tagName>
|
||||||
|
if (tagName != null) {
|
||||||
|
sb.append('<');
|
||||||
|
sb.append(tagName);
|
||||||
|
sb.append('>');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Loop thru the keys.
|
||||||
|
// don't use the new entrySet accessor to maintain Android Support
|
||||||
|
jo = (JSONObject) object;
|
||||||
|
for (final String key : jo.keySet()) {
|
||||||
|
Object value = jo.opt(key);
|
||||||
|
if (value == null) {
|
||||||
|
value = "";
|
||||||
|
} else if (value.getClass().isArray()) {
|
||||||
|
value = new JSONArray(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Emit content in body
|
||||||
|
if (key.equals(config.cDataTagName)) {
|
||||||
|
if (value instanceof JSONArray) {
|
||||||
|
ja = (JSONArray) value;
|
||||||
|
int jaLength = ja.length();
|
||||||
|
// don't use the new iterator API to maintain support for Android
|
||||||
|
for (int i = 0; i < jaLength; i++) {
|
||||||
|
if (i > 0) {
|
||||||
|
sb.append('\n');
|
||||||
|
}
|
||||||
|
Object val = ja.opt(i);
|
||||||
|
sb.append(escape(val.toString()));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
sb.append(escape(value.toString()));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Emit an array of similar keys
|
||||||
|
|
||||||
|
} else if (value instanceof JSONArray) {
|
||||||
|
ja = (JSONArray) value;
|
||||||
|
int jaLength = ja.length();
|
||||||
|
// don't use the new iterator API to maintain support for Android
|
||||||
|
for (int i = 0; i < jaLength; i++) {
|
||||||
|
Object val = ja.opt(i);
|
||||||
|
if (val instanceof JSONArray) {
|
||||||
|
sb.append('<');
|
||||||
|
sb.append(key);
|
||||||
|
sb.append('>');
|
||||||
|
sb.append(toString(val, null, config));
|
||||||
|
sb.append("</");
|
||||||
|
sb.append(key);
|
||||||
|
sb.append('>');
|
||||||
|
} else {
|
||||||
|
sb.append(toString(val, key, config));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if ("".equals(value)) {
|
||||||
|
sb.append('<');
|
||||||
|
sb.append(key);
|
||||||
|
sb.append("/>");
|
||||||
|
|
||||||
|
// Emit a new tag <k>
|
||||||
|
|
||||||
|
} else {
|
||||||
|
sb.append(toString(value, key, config));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (tagName != null) {
|
||||||
|
|
||||||
|
// Emit the </tagName> close tag
|
||||||
|
sb.append("</");
|
||||||
|
sb.append(tagName);
|
||||||
|
sb.append('>');
|
||||||
|
}
|
||||||
|
return sb.toString();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
if (object != null && (object instanceof JSONArray || object.getClass().isArray())) {
|
||||||
|
if(object.getClass().isArray()) {
|
||||||
|
ja = new JSONArray(object);
|
||||||
|
} else {
|
||||||
|
ja = (JSONArray) object;
|
||||||
|
}
|
||||||
|
int jaLength = ja.length();
|
||||||
|
// don't use the new iterator API to maintain support for Android
|
||||||
|
for (int i = 0; i < jaLength; i++) {
|
||||||
|
Object val = ja.opt(i);
|
||||||
|
// XML does not have good support for arrays. If an array
|
||||||
|
// appears in a place where XML is lacking, synthesize an
|
||||||
|
// <array> element.
|
||||||
|
sb.append(toString(val, tagName == null ? "array" : tagName, config));
|
||||||
|
}
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
string = (object == null) ? "null" : escape(object.toString());
|
||||||
|
return (tagName == null) ? "\"" + string + "\""
|
||||||
|
: (string.length() == 0) ? "<" + tagName + "/>" : "<" + tagName
|
||||||
|
+ ">" + string + "</" + tagName + ">";
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
107
src/main/java/org/json/XMLParserConfiguration.java
Normal file
107
src/main/java/org/json/XMLParserConfiguration.java
Normal file
|
@ -0,0 +1,107 @@
|
||||||
|
package org.json;
|
||||||
|
/*
|
||||||
|
Copyright (c) 2002 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configuration object for the XML parser.
|
||||||
|
* @author AylwardJ
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class XMLParserConfiguration {
|
||||||
|
/** Original Configuration of the XML Parser. */
|
||||||
|
public static final XMLParserConfiguration ORIGINAL = new XMLParserConfiguration();
|
||||||
|
/** Original configuration of the XML Parser except that values are kept as strings. */
|
||||||
|
public static final XMLParserConfiguration KEEP_STRINGS = new XMLParserConfiguration(true);
|
||||||
|
/**
|
||||||
|
* When parsing the XML into JSON, specifies if values should be kept as strings (true), or if
|
||||||
|
* they should try to be guessed into JSON values (numeric, boolean, string)
|
||||||
|
*/
|
||||||
|
public final boolean keepStrings;
|
||||||
|
/**
|
||||||
|
* The name of the key in a JSON Object that indicates a CDATA section. Historically this has
|
||||||
|
* been the value "content" but can be changed. Use <code>null</code> to indicate no CDATA
|
||||||
|
* processing.
|
||||||
|
*/
|
||||||
|
public final String cDataTagName;
|
||||||
|
/**
|
||||||
|
* When parsing the XML into JSON, specifies if values with attribute xsi:nil="true"
|
||||||
|
* should be kept as attribute(false), or they should be converted to null(true)
|
||||||
|
*/
|
||||||
|
public final boolean convertNilAttributeToNull;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Default parser configuration. Does not keep strings, and the CDATA Tag Name is "content".
|
||||||
|
*/
|
||||||
|
public XMLParserConfiguration () {
|
||||||
|
this(false, "content", false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configure the parser string processing and use the default CDATA Tag Name as "content".
|
||||||
|
* @param keepStrings <code>true</code> to parse all values as string.
|
||||||
|
* <code>false</code> to try and convert XML string values into a JSON value.
|
||||||
|
*/
|
||||||
|
public XMLParserConfiguration (final boolean keepStrings) {
|
||||||
|
this(keepStrings, "content", false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configure the parser string processing to try and convert XML values to JSON values and
|
||||||
|
* use the passed CDATA Tag Name the processing value. Pass <code>null</code> to
|
||||||
|
* disable CDATA processing
|
||||||
|
* @param cDataTagName<code>null</code> to disable CDATA processing. Any other value
|
||||||
|
* to use that value as the JSONObject key name to process as CDATA.
|
||||||
|
*/
|
||||||
|
public XMLParserConfiguration (final String cDataTagName) {
|
||||||
|
this(false, cDataTagName, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configure the parser to use custom settings.
|
||||||
|
* @param keepStrings <code>true</code> to parse all values as string.
|
||||||
|
* <code>false</code> to try and convert XML string values into a JSON value.
|
||||||
|
* @param cDataTagName<code>null</code> to disable CDATA processing. Any other value
|
||||||
|
* to use that value as the JSONObject key name to process as CDATA.
|
||||||
|
*/
|
||||||
|
public XMLParserConfiguration (final boolean keepStrings, final String cDataTagName) {
|
||||||
|
this.keepStrings = keepStrings;
|
||||||
|
this.cDataTagName = cDataTagName;
|
||||||
|
this.convertNilAttributeToNull = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configure the parser to use custom settings.
|
||||||
|
* @param keepStrings <code>true</code> to parse all values as string.
|
||||||
|
* <code>false</code> to try and convert XML string values into a JSON value.
|
||||||
|
* @param cDataTagName <code>null</code> to disable CDATA processing. Any other value
|
||||||
|
* to use that value as the JSONObject key name to process as CDATA.
|
||||||
|
* @param convertNilAttributeToNull <code>true</code> to parse values with attribute xsi:nil="true" as null.
|
||||||
|
* <code>false</code> to parse values with attribute xsi:nil="true" as {"xsi:nil":true}.
|
||||||
|
*/
|
||||||
|
public XMLParserConfiguration (final boolean keepStrings, final String cDataTagName, final boolean convertNilAttributeToNull) {
|
||||||
|
this.keepStrings = keepStrings;
|
||||||
|
this.cDataTagName = cDataTagName;
|
||||||
|
this.convertNilAttributeToNull = convertNilAttributeToNull;
|
||||||
|
}
|
||||||
|
}
|
93
XMLTokener.java → src/main/java/org/json/XMLTokener.java
Executable file → Normal file
93
XMLTokener.java → src/main/java/org/json/XMLTokener.java
Executable file → Normal file
|
@ -24,11 +24,13 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
SOFTWARE.
|
SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import java.io.Reader;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The XMLTokener extends the JSONTokener to provide additional methods
|
* The XMLTokener extends the JSONTokener to provide additional methods
|
||||||
* for the parsing of XML texts.
|
* for the parsing of XML texts.
|
||||||
* @author JSON.org
|
* @author JSON.org
|
||||||
* @version 2014-05-03
|
* @version 2015-12-09
|
||||||
*/
|
*/
|
||||||
public class XMLTokener extends JSONTokener {
|
public class XMLTokener extends JSONTokener {
|
||||||
|
|
||||||
|
@ -47,6 +49,14 @@ public class XMLTokener extends JSONTokener {
|
||||||
entity.put("quot", XML.QUOT);
|
entity.put("quot", XML.QUOT);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct an XMLTokener from a Reader.
|
||||||
|
* @param r A source reader.
|
||||||
|
*/
|
||||||
|
public XMLTokener(Reader r) {
|
||||||
|
super(r);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct an XMLTokener from a string.
|
* Construct an XMLTokener from a string.
|
||||||
* @param s A source string.
|
* @param s A source string.
|
||||||
|
@ -64,11 +74,8 @@ public class XMLTokener extends JSONTokener {
|
||||||
char c;
|
char c;
|
||||||
int i;
|
int i;
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
for (;;) {
|
while (more()) {
|
||||||
c = next();
|
c = next();
|
||||||
if (end()) {
|
|
||||||
throw syntaxError("Unclosed CDATA");
|
|
||||||
}
|
|
||||||
sb.append(c);
|
sb.append(c);
|
||||||
i = sb.length() - 3;
|
i = sb.length() - 3;
|
||||||
if (i >= 0 && sb.charAt(i) == ']' &&
|
if (i >= 0 && sb.charAt(i) == ']' &&
|
||||||
|
@ -77,17 +84,19 @@ public class XMLTokener extends JSONTokener {
|
||||||
return sb.toString();
|
return sb.toString();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
throw syntaxError("Unclosed CDATA");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the next XML outer token, trimming whitespace. There are two kinds
|
* Get the next XML outer token, trimming whitespace. There are two kinds
|
||||||
* of tokens: the '<' character which begins a markup tag, and the content
|
* of tokens: the <pre>{@code '<' }</pre> character which begins a markup
|
||||||
|
* tag, and the content
|
||||||
* text between markup tags.
|
* text between markup tags.
|
||||||
*
|
*
|
||||||
* @return A string, or a '<' Character, or null if there is no more
|
* @return A string, or a <pre>{@code '<' }</pre> Character, or null if
|
||||||
* source text.
|
* there is no more source text.
|
||||||
* @throws JSONException
|
* @throws JSONException if a called function has an error
|
||||||
*/
|
*/
|
||||||
public Object nextContent() throws JSONException {
|
public Object nextContent() throws JSONException {
|
||||||
char c;
|
char c;
|
||||||
|
@ -103,7 +112,10 @@ public class XMLTokener extends JSONTokener {
|
||||||
}
|
}
|
||||||
sb = new StringBuilder();
|
sb = new StringBuilder();
|
||||||
for (;;) {
|
for (;;) {
|
||||||
if (c == '<' || c == 0) {
|
if (c == 0) {
|
||||||
|
return sb.toString().trim();
|
||||||
|
}
|
||||||
|
if (c == '<') {
|
||||||
back();
|
back();
|
||||||
return sb.toString().trim();
|
return sb.toString().trim();
|
||||||
}
|
}
|
||||||
|
@ -118,13 +130,15 @@ public class XMLTokener extends JSONTokener {
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* <pre>{@code
|
||||||
* Return the next entity. These entities are translated to Characters:
|
* Return the next entity. These entities are translated to Characters:
|
||||||
* <code>& ' > < "</code>.
|
* & ' > < ".
|
||||||
|
* }</pre>
|
||||||
* @param ampersand An ampersand character.
|
* @param ampersand An ampersand character.
|
||||||
* @return A Character or an entity String if the entity is not recognized.
|
* @return A Character or an entity String if the entity is not recognized.
|
||||||
* @throws JSONException If missing ';' in XML entity.
|
* @throws JSONException If missing ';' in XML entity.
|
||||||
*/
|
*/
|
||||||
public Object nextEntity(char ampersand) throws JSONException {
|
public Object nextEntity(@SuppressWarnings("unused") char ampersand) throws JSONException {
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
for (;;) {
|
for (;;) {
|
||||||
char c = next();
|
char c = next();
|
||||||
|
@ -137,17 +151,49 @@ public class XMLTokener extends JSONTokener {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
String string = sb.toString();
|
String string = sb.toString();
|
||||||
Object object = entity.get(string);
|
return unescapeEntity(string);
|
||||||
return object != null ? object : ampersand + string + ";";
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Unescape an XML entity encoding;
|
||||||
|
* @param e entity (only the actual entity value, not the preceding & or ending ;
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
static String unescapeEntity(String e) {
|
||||||
|
// validate
|
||||||
|
if (e == null || e.isEmpty()) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
// if our entity is an encoded unicode point, parse it.
|
||||||
|
if (e.charAt(0) == '#') {
|
||||||
|
int cp;
|
||||||
|
if (e.charAt(1) == 'x') {
|
||||||
|
// hex encoded unicode
|
||||||
|
cp = Integer.parseInt(e.substring(2), 16);
|
||||||
|
} else {
|
||||||
|
// decimal encoded unicode
|
||||||
|
cp = Integer.parseInt(e.substring(1));
|
||||||
|
}
|
||||||
|
return new String(new int[] {cp},0,1);
|
||||||
|
}
|
||||||
|
Character knownEntity = entity.get(e);
|
||||||
|
if(knownEntity==null) {
|
||||||
|
// we don't know the entity so keep it encoded
|
||||||
|
return '&' + e + ';';
|
||||||
|
}
|
||||||
|
return knownEntity.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* <pre>{@code
|
||||||
* Returns the next XML meta token. This is used for skipping over <!...>
|
* Returns the next XML meta token. This is used for skipping over <!...>
|
||||||
* and <?...?> structures.
|
* and <?...?> structures.
|
||||||
* @return Syntax characters (<code>< > / = ! ?</code>) are returned as
|
* }</pre>
|
||||||
|
* @return <pre>{@code Syntax characters (< > / = ! ?) are returned as
|
||||||
* Character, and strings and names are returned as Boolean. We don't care
|
* Character, and strings and names are returned as Boolean. We don't care
|
||||||
* what the values actually are.
|
* what the values actually are.
|
||||||
|
* }</pre>
|
||||||
* @throws JSONException If a string is not properly closed or if the XML
|
* @throws JSONException If a string is not properly closed or if the XML
|
||||||
* is badly structured.
|
* is badly structured.
|
||||||
*/
|
*/
|
||||||
|
@ -192,6 +238,7 @@ public class XMLTokener extends JSONTokener {
|
||||||
}
|
}
|
||||||
switch (c) {
|
switch (c) {
|
||||||
case 0:
|
case 0:
|
||||||
|
throw syntaxError("Unterminated string");
|
||||||
case '<':
|
case '<':
|
||||||
case '>':
|
case '>':
|
||||||
case '/':
|
case '/':
|
||||||
|
@ -209,10 +256,12 @@ public class XMLTokener extends JSONTokener {
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* <pre>{@code
|
||||||
* Get the next XML Token. These tokens are found inside of angle
|
* Get the next XML Token. These tokens are found inside of angle
|
||||||
* brackets. It may be one of these characters: <code>/ > = ! ?</code> or it
|
* brackets. It may be one of these characters: / > = ! ? or it
|
||||||
* may be a string wrapped in single quotes or double quotes, or it may be a
|
* may be a string wrapped in single quotes or double quotes, or it may be a
|
||||||
* name.
|
* name.
|
||||||
|
* }</pre>
|
||||||
* @return a String or a Character.
|
* @return a String or a Character.
|
||||||
* @throws JSONException If the XML is not well formed.
|
* @throws JSONException If the XML is not well formed.
|
||||||
*/
|
*/
|
||||||
|
@ -296,9 +345,11 @@ public class XMLTokener extends JSONTokener {
|
||||||
* Skip characters until past the requested string.
|
* Skip characters until past the requested string.
|
||||||
* If it is not found, we are left at the end of the source with a result of false.
|
* If it is not found, we are left at the end of the source with a result of false.
|
||||||
* @param to A string to skip past.
|
* @param to A string to skip past.
|
||||||
* @throws JSONException
|
|
||||||
*/
|
*/
|
||||||
public boolean skipPast(String to) throws JSONException {
|
// The Android implementation of JSONTokener has a public method of public void skipPast(String to)
|
||||||
|
// even though ours does not have that method, to have API compatibility, our method in the subclass
|
||||||
|
// should match.
|
||||||
|
public void skipPast(String to) {
|
||||||
boolean b;
|
boolean b;
|
||||||
char c;
|
char c;
|
||||||
int i;
|
int i;
|
||||||
|
@ -315,7 +366,7 @@ public class XMLTokener extends JSONTokener {
|
||||||
for (i = 0; i < length; i += 1) {
|
for (i = 0; i < length; i += 1) {
|
||||||
c = next();
|
c = next();
|
||||||
if (c == 0) {
|
if (c == 0) {
|
||||||
return false;
|
return;
|
||||||
}
|
}
|
||||||
circle[i] = c;
|
circle[i] = c;
|
||||||
}
|
}
|
||||||
|
@ -342,14 +393,14 @@ public class XMLTokener extends JSONTokener {
|
||||||
/* If we exit the loop with b intact, then victory is ours. */
|
/* If we exit the loop with b intact, then victory is ours. */
|
||||||
|
|
||||||
if (b) {
|
if (b) {
|
||||||
return true;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get the next character. If there isn't one, then defeat is ours. */
|
/* Get the next character. If there isn't one, then defeat is ours. */
|
||||||
|
|
||||||
c = next();
|
c = next();
|
||||||
if (c == 0) {
|
if (c == 0) {
|
||||||
return false;
|
return;
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
* Shove the character in the circle buffer and advance the
|
* Shove the character in the circle buffer and advance the
|
324
src/test/java/org/json/junit/CDLTest.java
Normal file
324
src/test/java/org/json/junit/CDLTest.java
Normal file
|
@ -0,0 +1,324 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import org.json.JSONException;
|
||||||
|
import org.json.JSONObject;
|
||||||
|
import org.json.JSONArray;
|
||||||
|
import org.json.CDL;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for CDL.java.
|
||||||
|
* CDL provides an application level API, but it is not used by the
|
||||||
|
* reference app. To test it, strings will be converted to JSON-Java classes
|
||||||
|
* and then converted back.
|
||||||
|
*/
|
||||||
|
public class CDLTest {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
String lines = new String(
|
||||||
|
"Col 1, Col 2, \tCol 3, Col 4, Col 5, Col 6, Col 7\n" +
|
||||||
|
"val1, val2, val3, val4, val5, val6, val7\n" +
|
||||||
|
"1, 2, 3, 4\t, 5, 6, 7\n" +
|
||||||
|
"true, false, true, true, false, false, false\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"
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* CDL.toJSONArray() adds all values as strings, with no filtering or
|
||||||
|
* conversions. For testing, this means that the expected JSONObject
|
||||||
|
* values all must be quoted in the cases where the JSONObject parsing
|
||||||
|
* might normally convert the value into a non-string.
|
||||||
|
*/
|
||||||
|
String expectedLines = new String(
|
||||||
|
"[{Col 1:val1, Col 2:val2, Col 3:val3, Col 4:val4, Col 5:val5, Col 6:val6, Col 7:val7}, "+
|
||||||
|
"{Col 1:\"1\", Col 2:\"2\", Col 3:\"3\", Col 4:\"4\", Col 5:\"5\", Col 6:\"6\", Col 7:\"7\"}, "+
|
||||||
|
"{Col 1:\"true\", Col 2:\"false\", Col 3:\"true\", Col 4:\"true\", Col 5:\"false\", Col 6:\"false\", Col 7:\"false\"}, "+
|
||||||
|
"{Col 1:\"0.23\", Col 2:\"57.42\", Col 3:\"5e27\", Col 4:\"-234.879\", Col 5:\"2.34e5\", Col 6:\"0.0\", Col 7:\"9e-3\"}, "+
|
||||||
|
"{Col 1:\"va\tl1\", Col 2:\"v\bal2\", Col 3:val3, Col 4:\"val\f4\", Col 5:val5, Col 6:va\'l6, Col 7:val7}]");
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to create a JSONArray from a null string.
|
||||||
|
* Expect a NullPointerException.
|
||||||
|
*/
|
||||||
|
@Test(expected=NullPointerException.class)
|
||||||
|
public void exceptionOnNullString() {
|
||||||
|
String nullStr = null;
|
||||||
|
CDL.toJSONArray(nullStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to create a JSONArray from a string with unbalanced quotes
|
||||||
|
* in column title line. Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void unbalancedQuoteInName() {
|
||||||
|
String badLine = "Col1, \"Col2\nVal1, Val2";
|
||||||
|
try {
|
||||||
|
CDL.toJSONArray(badLine);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Missing close quote '\"'. at 12 [character 0 line 2]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to create a JSONArray from a string with unbalanced quotes
|
||||||
|
* in value line. Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void unbalancedQuoteInValue() {
|
||||||
|
String badLine = "Col1, Col2\n\"Val1, Val2";
|
||||||
|
try {
|
||||||
|
CDL.toJSONArray(badLine);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Missing close quote '\"'. at 22 [character 11 line 2]",
|
||||||
|
e.getMessage());
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to create a JSONArray from a string with null char
|
||||||
|
* in column title line. Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void nullInName() {
|
||||||
|
String badLine = "C\0ol1, Col2\nVal1, Val2";
|
||||||
|
try {
|
||||||
|
CDL.toJSONArray(badLine);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Bad character 'o' (111). at 2 [character 3 line 1]",
|
||||||
|
e.getMessage());
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempt to create a JSONArray with unbalanced quotes and a properly escaped doubled quote.
|
||||||
|
* Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void unbalancedEscapedQuote(){
|
||||||
|
String badLine = "Col1, Col2\n\"Val1, \"\"Val2\"\"";
|
||||||
|
try {
|
||||||
|
CDL.toJSONArray(badLine);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Missing close quote '\"'. at 26 [character 15 line 2]",
|
||||||
|
e.getMessage());
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Assert that there is no error for a single escaped quote within a properly embedded quote.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void singleEscapedQuote(){
|
||||||
|
String singleEscape = "Col1, Col2\nVal1, \"\"\"Val2\"";
|
||||||
|
JSONArray jsonArray = CDL.toJSONArray(singleEscape);
|
||||||
|
|
||||||
|
String cdlStr = CDL.toString(jsonArray);
|
||||||
|
assertTrue(cdlStr.contains("Col1"));
|
||||||
|
assertTrue(cdlStr.contains("Col2"));
|
||||||
|
assertTrue(cdlStr.contains("Val1"));
|
||||||
|
assertTrue(cdlStr.contains("\"Val2"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Assert that there is no error for a single escaped quote within a properly
|
||||||
|
* embedded quote when not the last value.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void singleEscapedQuoteMiddleString(){
|
||||||
|
String singleEscape = "Col1, Col2\nVal1, \"\"\"Val2\"\nVal 3,Val 4";
|
||||||
|
JSONArray jsonArray = CDL.toJSONArray(singleEscape);
|
||||||
|
|
||||||
|
String cdlStr = CDL.toString(jsonArray);
|
||||||
|
assertTrue(cdlStr.contains("Col1"));
|
||||||
|
assertTrue(cdlStr.contains("Col2"));
|
||||||
|
assertTrue(cdlStr.contains("Val1"));
|
||||||
|
assertTrue(cdlStr.contains("\"Val2"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempt to create a JSONArray with an escape quote and no enclosing quotes.
|
||||||
|
* Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void badEscapedQuote(){
|
||||||
|
String badLine = "Col1, Col2\nVal1, \"\"Val2";
|
||||||
|
|
||||||
|
try {
|
||||||
|
CDL.toJSONArray(badLine);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
System.out.println("Message" + e.getMessage());
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Bad character 'V' (86). at 20 [character 9 line 2]",
|
||||||
|
e.getMessage());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* call toString with a null array
|
||||||
|
*/
|
||||||
|
@Test(expected=NullPointerException.class)
|
||||||
|
public void nullJSONArrayToString() {
|
||||||
|
CDL.toString((JSONArray)null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a JSONArray from an empty string
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void emptyString() {
|
||||||
|
String emptyStr = "";
|
||||||
|
JSONArray jsonArray = CDL.toJSONArray(emptyStr);
|
||||||
|
assertTrue("CDL should return null when the input string is empty",
|
||||||
|
jsonArray == null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a JSONArray with only 1 row
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void onlyColumnNames() {
|
||||||
|
String columnNameStr = "col1, col2, col3";
|
||||||
|
JSONArray jsonArray = CDL.toJSONArray(columnNameStr);
|
||||||
|
assertNull("CDL should return null when only 1 row is given",
|
||||||
|
jsonArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a JSONArray from string containing only whitespace and commas
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void emptyLinesToJSONArray() {
|
||||||
|
String str = " , , , \n , , , ";
|
||||||
|
JSONArray jsonArray = CDL.toJSONArray(str);
|
||||||
|
assertNull("JSONArray should be null for no content",
|
||||||
|
jsonArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* call toString with a null array
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void emptyJSONArrayToString() {
|
||||||
|
JSONArray jsonArray = new JSONArray();
|
||||||
|
String str = CDL.toString(jsonArray);
|
||||||
|
assertNull("CDL should return null for toString(null)",
|
||||||
|
str);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* call toString with a null arrays for names and values
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void nullJSONArraysToString() {
|
||||||
|
String str = CDL.toString(null, null);
|
||||||
|
assertNull("CDL should return null for toString(null)",
|
||||||
|
str);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Given a JSONArray that was not built by CDL, some chars may be
|
||||||
|
* found that would otherwise be filtered out by CDL.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void checkSpecialChars() {
|
||||||
|
JSONArray jsonArray = new JSONArray();
|
||||||
|
JSONObject jsonObject = new JSONObject();
|
||||||
|
jsonArray.put(jsonObject);
|
||||||
|
// \r will be filtered from name
|
||||||
|
jsonObject.put("Col \r1", "V1");
|
||||||
|
// \r will be filtered from value
|
||||||
|
jsonObject.put("Col 2", "V2\r");
|
||||||
|
assertTrue("expected length should be 1",jsonArray.length() == 1);
|
||||||
|
String cdlStr = CDL.toString(jsonArray);
|
||||||
|
jsonObject = jsonArray.getJSONObject(0);
|
||||||
|
assertTrue(cdlStr.contains("\"Col 1\""));
|
||||||
|
assertTrue(cdlStr.contains("Col 2"));
|
||||||
|
assertTrue(cdlStr.contains("V1"));
|
||||||
|
assertTrue(cdlStr.contains("\"V2\""));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a JSONArray from a string of lines
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void textToJSONArray() {
|
||||||
|
JSONArray jsonArray = CDL.toJSONArray(this.lines);
|
||||||
|
JSONArray expectedJsonArray = new JSONArray(this.expectedLines);
|
||||||
|
Util.compareActualVsExpectedJsonArrays(jsonArray, expectedJsonArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a JSONArray from a JSONArray of titles and a
|
||||||
|
* string of value lines
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void jsonArrayToJSONArray() {
|
||||||
|
String nameArrayStr = "[Col1, Col2]";
|
||||||
|
String values = "V1, V2";
|
||||||
|
JSONArray nameJSONArray = new JSONArray(nameArrayStr);
|
||||||
|
JSONArray jsonArray = CDL.toJSONArray(nameJSONArray, values);
|
||||||
|
JSONArray expectedJsonArray = new JSONArray("[{Col1:V1,Col2:V2}]");
|
||||||
|
Util.compareActualVsExpectedJsonArrays(jsonArray, expectedJsonArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a JSONArray from a string of lines,
|
||||||
|
* then convert to string and then back to JSONArray
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void textToJSONArrayAndBackToString() {
|
||||||
|
JSONArray jsonArray = CDL.toJSONArray(this.lines);
|
||||||
|
String jsonStr = CDL.toString(jsonArray);
|
||||||
|
JSONArray finalJsonArray = CDL.toJSONArray(jsonStr);
|
||||||
|
JSONArray expectedJsonArray = new JSONArray(this.expectedLines);
|
||||||
|
Util.compareActualVsExpectedJsonArrays(finalJsonArray, expectedJsonArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
210
src/test/java/org/json/junit/CookieListTest.java
Normal file
210
src/test/java/org/json/junit/CookieListTest.java
Normal file
|
@ -0,0 +1,210 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
import org.json.*;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import com.jayway.jsonpath.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HTTP cookie specification RFC6265: http://tools.ietf.org/html/rfc6265
|
||||||
|
* <p>
|
||||||
|
* A cookie list is a JSONObject whose members are presumed to be cookie
|
||||||
|
* name/value pairs. Entries are unescaped while being added, and escaped in
|
||||||
|
* the toString() output.
|
||||||
|
* Unescaping means to convert %hh hex strings to the ascii equivalent
|
||||||
|
* and converting '+' to ' '.
|
||||||
|
* Escaping converts '+', '%', '=', ';' and ascii control chars to %hh hex strings.
|
||||||
|
* <p>
|
||||||
|
* CookieList should not be considered as just a list of Cookie objects:<br>
|
||||||
|
* - CookieList stores a cookie name/value pair as a single entry; Cookie stores
|
||||||
|
* it as 2 entries (key="name" and key="value").<br>
|
||||||
|
* - CookieList requires multiple name/value pairs as input; Cookie allows the
|
||||||
|
* 'secure' name with no associated value<br>
|
||||||
|
* - CookieList has no special handling for attribute name/value pairs.<br>
|
||||||
|
*/
|
||||||
|
public class CookieListTest {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to create a CookieList from a null string.
|
||||||
|
* Expects a NullPointerException.
|
||||||
|
*/
|
||||||
|
@Test(expected=NullPointerException.class)
|
||||||
|
public void nullCookieListException() {
|
||||||
|
String cookieStr = null;
|
||||||
|
CookieList.toJSONObject(cookieStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to create a CookieList from a malformed string.
|
||||||
|
* Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void malFormedCookieListException() {
|
||||||
|
String cookieStr = "thisCookieHasNoEqualsChar";
|
||||||
|
try {
|
||||||
|
CookieList.toJSONObject(cookieStr);
|
||||||
|
fail("should throw an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
/**
|
||||||
|
* Not sure of the missing char, but full string compare fails
|
||||||
|
*/
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Expected '=' and instead saw '' at 25 [character 26 line 1]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a CookieList from an empty string.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void emptyStringCookieList() {
|
||||||
|
String cookieStr = "";
|
||||||
|
JSONObject jsonObject = CookieList.toJSONObject(cookieStr);
|
||||||
|
assertTrue(jsonObject.isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* CookieList with the simplest cookie - a name/value pair with no delimiter.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void simpleCookieList() {
|
||||||
|
String cookieStr = "SID=31d4d96e407aad42";
|
||||||
|
JSONObject jsonObject = CookieList.toJSONObject(cookieStr);
|
||||||
|
// validate JSON content
|
||||||
|
Object doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("Expected 1 top level item", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 1);
|
||||||
|
assertTrue("expected 31d4d96e407aad42", "31d4d96e407aad42".equals(jsonObject.query("/SID")));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* CookieList with a single a cookie which has a name/value pair and delimiter.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void simpleCookieListWithDelimiter() {
|
||||||
|
String cookieStr = "SID=31d4d96e407aad42;";
|
||||||
|
JSONObject jsonObject = CookieList.toJSONObject(cookieStr);
|
||||||
|
// validate JSON content
|
||||||
|
Object doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("Expected 1 top level item", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 1);
|
||||||
|
assertTrue("expected 31d4d96e407aad42", "31d4d96e407aad42".equals(jsonObject.query("/SID")));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* CookieList with multiple cookies consisting of name/value pairs
|
||||||
|
* with delimiters.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void multiPartCookieList() {
|
||||||
|
String cookieStr =
|
||||||
|
"name1=myCookieValue1; "+
|
||||||
|
" name2=myCookieValue2;"+
|
||||||
|
"name3=myCookieValue3;"+
|
||||||
|
" name4=myCookieValue4; "+
|
||||||
|
"name5=myCookieValue5;"+
|
||||||
|
" name6=myCookieValue6;";
|
||||||
|
JSONObject jsonObject = CookieList.toJSONObject(cookieStr);
|
||||||
|
// validate JSON content
|
||||||
|
Object doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("Expected 6 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 6);
|
||||||
|
assertTrue("expected myCookieValue1", "myCookieValue1".equals(jsonObject.query("/name1")));
|
||||||
|
assertTrue("expected myCookieValue2", "myCookieValue2".equals(jsonObject.query("/name2")));
|
||||||
|
assertTrue("expected myCookieValue3", "myCookieValue3".equals(jsonObject.query("/name3")));
|
||||||
|
assertTrue("expected myCookieValue4", "myCookieValue4".equals(jsonObject.query("/name4")));
|
||||||
|
assertTrue("expected myCookieValue5", "myCookieValue5".equals(jsonObject.query("/name5")));
|
||||||
|
assertTrue("expected myCookieValue6", "myCookieValue6".equals(jsonObject.query("/name6")));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* CookieList from a JSONObject with valid key and null value
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void convertCookieListWithNullValueToString() {
|
||||||
|
JSONObject jsonObject = new JSONObject();
|
||||||
|
jsonObject.put("key", JSONObject.NULL);
|
||||||
|
String cookieToStr = CookieList.toString(jsonObject);
|
||||||
|
assertTrue("toString() should be empty", "".equals(cookieToStr));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* CookieList with multiple entries converted to a JSON document.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void convertCookieListToString() {
|
||||||
|
String cookieStr =
|
||||||
|
"name1=myCookieValue1; "+
|
||||||
|
" name2=myCookieValue2;"+
|
||||||
|
"name3=myCookieValue3;"+
|
||||||
|
" name4=myCookieValue4; "+
|
||||||
|
"name5=myCookieValue5;"+
|
||||||
|
" name6=myCookieValue6;";
|
||||||
|
JSONObject jsonObject = CookieList.toJSONObject(cookieStr);
|
||||||
|
// exercise CookieList.toString()
|
||||||
|
String cookieListString = CookieList.toString(jsonObject);
|
||||||
|
// have to convert it back for validation
|
||||||
|
jsonObject = CookieList.toJSONObject(cookieListString);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
Object doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("Expected 6 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 6);
|
||||||
|
assertTrue("expected myCookieValue1", "myCookieValue1".equals(jsonObject.query("/name1")));
|
||||||
|
assertTrue("expected myCookieValue2", "myCookieValue2".equals(jsonObject.query("/name2")));
|
||||||
|
assertTrue("expected myCookieValue3", "myCookieValue3".equals(jsonObject.query("/name3")));
|
||||||
|
assertTrue("expected myCookieValue4", "myCookieValue4".equals(jsonObject.query("/name4")));
|
||||||
|
assertTrue("expected myCookieValue5", "myCookieValue5".equals(jsonObject.query("/name5")));
|
||||||
|
assertTrue("expected myCookieValue6", "myCookieValue6".equals(jsonObject.query("/name6")));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* CookieList with multiple entries and some '+' chars and URL-encoded
|
||||||
|
* values converted to a JSON document.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void convertEncodedCookieListToString() {
|
||||||
|
String cookieStr =
|
||||||
|
"name1=myCookieValue1; "+
|
||||||
|
" name2=my+Cookie+Value+2;"+
|
||||||
|
"name3=my%2BCookie%26Value%3B3%3D;"+
|
||||||
|
" name4=my%25CookieValue4; "+
|
||||||
|
"name5=myCookieValue5;"+
|
||||||
|
" name6=myCookieValue6;";
|
||||||
|
JSONObject jsonObject = CookieList.toJSONObject(cookieStr);
|
||||||
|
// validate JSON content
|
||||||
|
Object doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("Expected 6 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 6);
|
||||||
|
assertTrue("expected myCookieValue1", "myCookieValue1".equals(jsonObject.query("/name1")));
|
||||||
|
assertTrue("expected my Cookie Value 2", "my Cookie Value 2".equals(jsonObject.query("/name2")));
|
||||||
|
assertTrue("expected my+Cookie&Value;3=", "my+Cookie&Value;3=".equals(jsonObject.query("/name3")));
|
||||||
|
assertTrue("expected my%CookieValue4", "my%CookieValue4".equals(jsonObject.query("/name4")));
|
||||||
|
assertTrue("expected my%CookieValue5", "myCookieValue5".equals(jsonObject.query("/name5")));
|
||||||
|
assertTrue("expected myCookieValue6", "myCookieValue6".equals(jsonObject.query("/name6")));
|
||||||
|
}
|
||||||
|
}
|
262
src/test/java/org/json/junit/CookieTest.java
Normal file
262
src/test/java/org/json/junit/CookieTest.java
Normal file
|
@ -0,0 +1,262 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
|
import org.json.*;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HTTP cookie specification: RFC6265
|
||||||
|
* <p>
|
||||||
|
* At its most basic, a cookie is a name=value pair. The value may be subdivided
|
||||||
|
* into other cookies, but that is not tested here. The cookie may also include
|
||||||
|
* certain named attributes, delimited by semicolons.
|
||||||
|
* <p>
|
||||||
|
* The Cookie.toString() method emits certain attributes if present: expires,
|
||||||
|
* domain, path, secure. All but secure are name-value pairs. Other attributes
|
||||||
|
* are not included in the toString() output.
|
||||||
|
* <p>
|
||||||
|
* A JSON-Java encoded cookie escapes '+', '%', '=', ';' with %hh values.
|
||||||
|
*/
|
||||||
|
public class CookieTest {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to create a JSONObject from a null string.
|
||||||
|
* Expects a NullPointerException.
|
||||||
|
*/
|
||||||
|
@Test(expected=NullPointerException.class)
|
||||||
|
public void nullCookieException() {
|
||||||
|
String cookieStr = null;
|
||||||
|
Cookie.toJSONObject(cookieStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to create a JSONObject from a cookie string with
|
||||||
|
* no '=' char.
|
||||||
|
* Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void malFormedNameValueException() {
|
||||||
|
String cookieStr = "thisCookieHasNoEqualsChar";
|
||||||
|
try {
|
||||||
|
Cookie.toJSONObject(cookieStr);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Expected '=' and instead saw '' at 25 [character 26 line 1]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to create a JSONObject from a cookie string
|
||||||
|
* with embedded ';' char.
|
||||||
|
* Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void booleanAttribute() {
|
||||||
|
String cookieStr = "this=Cookie;myAttribute";
|
||||||
|
JSONObject jo = Cookie.toJSONObject(cookieStr);
|
||||||
|
assertTrue("has key 'name'", jo.has("name"));
|
||||||
|
assertTrue("has key 'value'", jo.has("value"));
|
||||||
|
assertTrue("has key 'myAttribute'", jo.has("myattribute"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to create a JSONObject from an empty cookie string.<br>
|
||||||
|
* Note: Cookie throws an exception, but CookieList does not.<br>
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void emptyStringCookieException() {
|
||||||
|
String cookieStr = "";
|
||||||
|
try {
|
||||||
|
Cookie.toJSONObject(cookieStr);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Cookies must have a 'name'",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* Attempts to create a JSONObject from an cookie string where the name is blank.<br>
|
||||||
|
* Note: Cookie throws an exception, but CookieList does not.<br>
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void emptyNameCookieException() {
|
||||||
|
String cookieStr = " = value ";
|
||||||
|
try {
|
||||||
|
Cookie.toJSONObject(cookieStr);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Cookies must have a 'name'",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cookie from a simple name/value pair with no delimiter
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void simpleCookie() {
|
||||||
|
String cookieStr = "SID=31d4d96e407aad42";
|
||||||
|
String expectedCookieStr = "{\"name\":\"SID\",\"value\":\"31d4d96e407aad42\"}";
|
||||||
|
JSONObject jsonObject = Cookie.toJSONObject(cookieStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedCookieStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Store a cookie with all of the supported attributes in a
|
||||||
|
* JSONObject. The secure attribute, which has no value, is treated
|
||||||
|
* as a boolean.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void multiPartCookie() {
|
||||||
|
String cookieStr =
|
||||||
|
"PH=deleted; "+
|
||||||
|
" expires=Wed, 19-Mar-2014 17:53:53 GMT;"+
|
||||||
|
"path=/; "+
|
||||||
|
" domain=.yahoo.com;"+
|
||||||
|
"secure";
|
||||||
|
String expectedCookieStr =
|
||||||
|
"{"+
|
||||||
|
"\"name\":\"PH\","+
|
||||||
|
"\"value\":\"deleted\","+
|
||||||
|
"\"path\":\"/\","+
|
||||||
|
"\"expires\":\"Wed, 19-Mar-2014 17:53:53 GMT\","+
|
||||||
|
"\"domain\":\".yahoo.com\","+
|
||||||
|
"\"secure\":true"+
|
||||||
|
"}";
|
||||||
|
JSONObject jsonObject = Cookie.toJSONObject(cookieStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedCookieStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cookie.toString() will emit the non-standard "thiswont=beIncluded"
|
||||||
|
* attribute, and the attribute is still stored in the JSONObject.
|
||||||
|
* This test confirms both behaviors.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void convertCookieToString() {
|
||||||
|
String cookieStr =
|
||||||
|
"PH=deleted; "+
|
||||||
|
" expires=Wed, 19-Mar-2014 17:53:53 GMT;"+
|
||||||
|
"path=/; "+
|
||||||
|
" domain=.yahoo.com;"+
|
||||||
|
"thisWont=beIncluded;"+
|
||||||
|
"secure";
|
||||||
|
String expectedCookieStr =
|
||||||
|
"{\"thiswont\":\"beIncluded\","+
|
||||||
|
"\"path\":\"/\","+
|
||||||
|
"\"expires\":\"Wed, 19-Mar-2014 17:53:53 GMT\","+
|
||||||
|
"\"domain\":\".yahoo.com\","+
|
||||||
|
"\"name\":\"PH\","+
|
||||||
|
"\"secure\":true,"+
|
||||||
|
"\"value\":\"deleted\"}";
|
||||||
|
// Add the nonstandard attribute to the expected cookie string
|
||||||
|
String expectedDirectCompareCookieStr = expectedCookieStr;
|
||||||
|
// convert all strings into JSONObjects
|
||||||
|
JSONObject jsonObject = Cookie.toJSONObject(cookieStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedCookieStr);
|
||||||
|
JSONObject expectedDirectCompareJsonObject =
|
||||||
|
new JSONObject(expectedDirectCompareCookieStr);
|
||||||
|
// emit the string
|
||||||
|
String cookieToStr = Cookie.toString(jsonObject);
|
||||||
|
// create a final JSONObject from the string
|
||||||
|
JSONObject finalJsonObject = Cookie.toJSONObject(cookieToStr);
|
||||||
|
// JSONObject should contain the nonstandard string
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedDirectCompareJsonObject);
|
||||||
|
// JSONObject -> string -> JSONObject should not contain the nonstandard string
|
||||||
|
Util.compareActualVsExpectedJsonObjects(finalJsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A string may be URL-encoded when converting to JSONObject.
|
||||||
|
* If found, '+' is converted to ' ', and %hh hex strings are converted
|
||||||
|
* to their ascii char equivalents. This test confirms the decoding
|
||||||
|
* behavior.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void convertEncodedCookieToString() {
|
||||||
|
String cookieStr =
|
||||||
|
"PH=deleted; "+
|
||||||
|
" expires=Wed,+19-Mar-2014+17:53:53+GMT;"+
|
||||||
|
"path=/%2Bthis/is%26/a/spec%3Bsegment%3D; "+
|
||||||
|
" domain=.yahoo.com;"+
|
||||||
|
"secure";
|
||||||
|
String expectedCookieStr =
|
||||||
|
"{\"path\":\"/+this/is&/a/spec;segment=\","+
|
||||||
|
"\"expires\":\"Wed, 19-Mar-2014 17:53:53 GMT\","+
|
||||||
|
"\"domain\":\".yahoo.com\","+
|
||||||
|
"\"name\":\"PH\","+
|
||||||
|
"\"secure\":true,"+
|
||||||
|
"\"value\":\"deleted\"}";
|
||||||
|
JSONObject jsonObject = Cookie.toJSONObject(cookieStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedCookieStr);
|
||||||
|
String cookieToStr = Cookie.toString(jsonObject);
|
||||||
|
JSONObject finalJsonObject = Cookie.toJSONObject(cookieToStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(finalJsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A public API method performs a URL encoding for selected chars
|
||||||
|
* in a string. Control chars, '+', '%', '=', ';' are all encoded
|
||||||
|
* as %hh hex strings. The string is also trimmed.
|
||||||
|
* This test confirms that behavior.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void escapeString() {
|
||||||
|
String str = " +%\r\n\t\b%=;;; ";
|
||||||
|
String expectedStr = "%2b%25%0d%0a%09%08%25%3d%3b%3b%3b";
|
||||||
|
String actualStr = Cookie.escape(str);
|
||||||
|
assertTrue("expect escape() to encode correctly. Actual: " +actualStr+
|
||||||
|
" expected: " +expectedStr, expectedStr.equals(actualStr));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A public API method performs URL decoding for strings.
|
||||||
|
* '+' is converted to space and %hh hex strings are converted to
|
||||||
|
* their ascii equivalent values. The string is not trimmed.
|
||||||
|
* This test confirms that behavior.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void unescapeString() {
|
||||||
|
String str = " +%2b%25%0d%0a%09%08%25%3d%3b%3b%3b+ ";
|
||||||
|
String expectedStr = " +%\r\n\t\b%=;;; ";
|
||||||
|
String actualStr = Cookie.unescape(str);
|
||||||
|
assertTrue("expect unescape() to decode correctly. Actual: " +actualStr+
|
||||||
|
" expected: " +expectedStr, expectedStr.equals(actualStr));
|
||||||
|
}
|
||||||
|
}
|
453
src/test/java/org/json/junit/EnumTest.java
Normal file
453
src/test/java/org/json/junit/EnumTest.java
Normal file
|
@ -0,0 +1,453 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertNull;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
|
import java.util.EnumSet;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.json.JSONArray;
|
||||||
|
import org.json.JSONObject;
|
||||||
|
import org.json.junit.data.MyEnum;
|
||||||
|
import org.json.junit.data.MyEnumClass;
|
||||||
|
import org.json.junit.data.MyEnumField;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import com.jayway.jsonpath.Configuration;
|
||||||
|
import com.jayway.jsonpath.JsonPath;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Enums are not explicitly supported in JSON-Java. But because enums act like
|
||||||
|
* classes, all required behavior is already be present in some form.
|
||||||
|
* These tests explore how enum serialization works with JSON-Java.
|
||||||
|
*/
|
||||||
|
public class EnumTest {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* To serialize an enum by its getters, use the JSONObject Object constructor.
|
||||||
|
* The JSONObject ctor handles enum like any other bean. A JSONobject
|
||||||
|
* is created whose entries are the getter name/value pairs.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void jsonObjectFromEnum() {
|
||||||
|
// If there are no getters then the object is empty.
|
||||||
|
MyEnum myEnum = MyEnum.VAL2;
|
||||||
|
JSONObject jsonObject = new JSONObject(myEnum);
|
||||||
|
assertTrue("simple enum has no getters", jsonObject.isEmpty());
|
||||||
|
|
||||||
|
// enum with a getters should create a non-empty object
|
||||||
|
MyEnumField myEnumField = MyEnumField.VAL2;
|
||||||
|
jsonObject = new JSONObject(myEnumField);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
Object doc = Configuration.defaultConfiguration().jsonProvider()
|
||||||
|
.parse(jsonObject.toString());
|
||||||
|
assertTrue("expecting 2 items in top level object", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 2);
|
||||||
|
assertTrue("expecting val 2", "val 2".equals(jsonObject.query("/value")));
|
||||||
|
assertTrue("expecting 2", Integer.valueOf(2).equals(jsonObject.query("/intVal")));
|
||||||
|
|
||||||
|
/**
|
||||||
|
* class which contains enum instances. Each enum should be stored
|
||||||
|
* in its own JSONObject
|
||||||
|
*/
|
||||||
|
MyEnumClass myEnumClass = new MyEnumClass();
|
||||||
|
myEnumClass.setMyEnum(MyEnum.VAL1);
|
||||||
|
myEnumClass.setMyEnumField(MyEnumField.VAL3);
|
||||||
|
jsonObject = new JSONObject(myEnumClass);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("expected 2 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 2);
|
||||||
|
assertTrue("expected 2 myEnumField items", "VAL3".equals((JsonPath.read(doc, "$.myEnumField"))));
|
||||||
|
assertTrue("expected 0 myEnum items", "VAL1".equals((JsonPath.read(doc, "$.myEnum"))));
|
||||||
|
|
||||||
|
assertTrue("expecting MyEnumField.VAL3", MyEnumField.VAL3.equals(jsonObject.query("/myEnumField")));
|
||||||
|
assertTrue("expecting MyEnum.VAL1", MyEnum.VAL1.equals(jsonObject.query("/myEnum")));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* To serialize an enum by its set of allowed values, use getNames()
|
||||||
|
* and the the JSONObject Object with names constructor.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void jsonObjectFromEnumWithNames() {
|
||||||
|
String [] names;
|
||||||
|
JSONObject jsonObject;
|
||||||
|
|
||||||
|
MyEnum myEnum = MyEnum.VAL1;
|
||||||
|
names = JSONObject.getNames(myEnum);
|
||||||
|
// The values will be MyEnum fields
|
||||||
|
jsonObject = new JSONObject(myEnum, names);
|
||||||
|
|
||||||
|
// validate JSON object
|
||||||
|
Object doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("expected 3 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 3);
|
||||||
|
assertTrue("expected VAL1", MyEnum.VAL1.equals(jsonObject.query("/VAL1")));
|
||||||
|
assertTrue("expected VAL2", MyEnum.VAL2.equals(jsonObject.query("/VAL2")));
|
||||||
|
assertTrue("expected VAL3", MyEnum.VAL3.equals(jsonObject.query("/VAL3")));
|
||||||
|
|
||||||
|
MyEnumField myEnumField = MyEnumField.VAL3;
|
||||||
|
names = JSONObject.getNames(myEnumField);
|
||||||
|
// The values will be MyEnmField fields
|
||||||
|
jsonObject = new JSONObject(myEnumField, names);
|
||||||
|
doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("expected 3 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 3);
|
||||||
|
assertTrue("expected VAL1", MyEnumField.VAL1.equals(jsonObject.query("/VAL1")));
|
||||||
|
assertTrue("expected VAL2", MyEnumField.VAL2.equals(jsonObject.query("/VAL2")));
|
||||||
|
assertTrue("expected VAL3", MyEnumField.VAL3.equals(jsonObject.query("/VAL3")));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Verify that enums are handled consistently between JSONArray and JSONObject
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void verifyEnumConsistency(){
|
||||||
|
JSONObject jo = new JSONObject();
|
||||||
|
|
||||||
|
jo.put("value", MyEnumField.VAL2);
|
||||||
|
String expected="{\"value\":\"VAL2\"}";
|
||||||
|
String actual = jo.toString();
|
||||||
|
assertTrue("Expected "+expected+" but actual was "+actual, expected.equals(actual));
|
||||||
|
|
||||||
|
jo.accumulate("value", MyEnumField.VAL1);
|
||||||
|
expected="{\"value\":[\"VAL2\",\"VAL1\"]}";
|
||||||
|
actual = jo.toString();
|
||||||
|
assertTrue("Expected "+expected+" but actual was "+actual, expected.equals(actual));
|
||||||
|
|
||||||
|
jo.remove("value");
|
||||||
|
jo.append("value", MyEnumField.VAL1);
|
||||||
|
expected="{\"value\":[\"VAL1\"]}";
|
||||||
|
actual = jo.toString();
|
||||||
|
assertTrue("Expected "+expected+" but actual was "+actual, expected.equals(actual));
|
||||||
|
|
||||||
|
jo.put("value", EnumSet.of(MyEnumField.VAL2));
|
||||||
|
expected="{\"value\":[\"VAL2\"]}";
|
||||||
|
actual = jo.toString();
|
||||||
|
assertTrue("Expected "+expected+" but actual was "+actual, expected.equals(actual));
|
||||||
|
|
||||||
|
JSONArray ja = new JSONArray();
|
||||||
|
ja.put(MyEnumField.VAL2);
|
||||||
|
jo.put("value", ja);
|
||||||
|
actual = jo.toString();
|
||||||
|
assertTrue("Expected "+expected+" but actual was "+actual, expected.equals(actual));
|
||||||
|
|
||||||
|
jo.put("value", new MyEnumField[]{MyEnumField.VAL2});
|
||||||
|
actual = jo.toString();
|
||||||
|
assertTrue("Expected "+expected+" but actual was "+actual, expected.equals(actual));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* To serialize by assigned value, use the put() methods. The value
|
||||||
|
* will be stored as a enum type.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void enumPut() {
|
||||||
|
JSONObject jsonObject = new JSONObject();
|
||||||
|
MyEnum myEnum = MyEnum.VAL2;
|
||||||
|
jsonObject.put("myEnum", myEnum);
|
||||||
|
MyEnumField myEnumField = MyEnumField.VAL1;
|
||||||
|
jsonObject.putOnce("myEnumField", myEnumField);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
Object doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("expected 2 top level objects", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 2);
|
||||||
|
assertTrue("expected VAL2", MyEnum.VAL2.equals(jsonObject.query("/myEnum")));
|
||||||
|
assertTrue("expected VAL1", MyEnumField.VAL1.equals(jsonObject.query("/myEnumField")));
|
||||||
|
|
||||||
|
JSONArray jsonArray = new JSONArray();
|
||||||
|
jsonArray.put(myEnum);
|
||||||
|
jsonArray.put(1, myEnumField);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonArray.toString());
|
||||||
|
assertTrue("expected 2 top level objects", ((List<?>)(JsonPath.read(doc, "$"))).size() == 2);
|
||||||
|
assertTrue("expected VAL2", MyEnum.VAL2.equals(jsonArray.query("/0")));
|
||||||
|
assertTrue("expected VAL1", MyEnumField.VAL1.equals(jsonArray.query("/1")));
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Leaving these tests because they exercise get, opt, and remove
|
||||||
|
*/
|
||||||
|
assertTrue("expecting myEnum value", MyEnum.VAL2.equals(jsonArray.get(0)));
|
||||||
|
assertTrue("expecting myEnumField value", MyEnumField.VAL1.equals(jsonArray.opt(1)));
|
||||||
|
assertTrue("expecting myEnumField value", MyEnumField.VAL1.equals(jsonArray.remove(1)));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The default action of valueToString() is to call object.toString().
|
||||||
|
* For enums, this means the assigned value will be returned as a string.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void enumValueToString() {
|
||||||
|
String expectedStr1 = "\"VAL1\"";
|
||||||
|
String expectedStr2 = "\"VAL1\"";
|
||||||
|
MyEnum myEnum = MyEnum.VAL1;
|
||||||
|
MyEnumField myEnumField = MyEnumField.VAL1;
|
||||||
|
MyEnumClass myEnumClass = new MyEnumClass();
|
||||||
|
|
||||||
|
String str1 = JSONObject.valueToString(myEnum);
|
||||||
|
assertTrue("actual myEnum: "+str1+" expected: "+expectedStr1,
|
||||||
|
str1.equals(expectedStr1));
|
||||||
|
String str2 = JSONObject.valueToString(myEnumField);
|
||||||
|
assertTrue("actual myEnumField: "+str2+" expected: "+expectedStr2,
|
||||||
|
str2.equals(expectedStr2));
|
||||||
|
|
||||||
|
/**
|
||||||
|
* However, an enum within another class will not be rendered
|
||||||
|
* unless that class overrides default toString()
|
||||||
|
*/
|
||||||
|
String expectedStr3 = "\"org.json.junit.data.MyEnumClass@";
|
||||||
|
myEnumClass.setMyEnum(MyEnum.VAL1);
|
||||||
|
myEnumClass.setMyEnumField(MyEnumField.VAL1);
|
||||||
|
String str3 = JSONObject.valueToString(myEnumClass);
|
||||||
|
assertTrue("actual myEnumClass: "+str3+" expected: "+expectedStr3,
|
||||||
|
str3.startsWith(expectedStr3));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* In whatever form the enum was added to the JSONObject or JSONArray,
|
||||||
|
* json[Object|Array].toString should serialize it in a reasonable way.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void enumToString() {
|
||||||
|
MyEnum myEnum = MyEnum.VAL2;
|
||||||
|
JSONObject jsonObject = new JSONObject(myEnum);
|
||||||
|
String expectedStr = "{}";
|
||||||
|
assertTrue("myEnum toString() should be empty", expectedStr.equals(jsonObject.toString()));
|
||||||
|
|
||||||
|
MyEnumField myEnumField = MyEnumField.VAL2;
|
||||||
|
jsonObject = new JSONObject(myEnumField);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
Object doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("expected 2 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 2);
|
||||||
|
assertTrue("expected val 2", "val 2".equals(jsonObject.query("/value")));
|
||||||
|
assertTrue("expected 2", Integer.valueOf(2).equals(jsonObject.query("/intVal")));
|
||||||
|
|
||||||
|
MyEnumClass myEnumClass = new MyEnumClass();
|
||||||
|
myEnumClass.setMyEnum(MyEnum.VAL1);
|
||||||
|
myEnumClass.setMyEnumField(MyEnumField.VAL3);
|
||||||
|
jsonObject = new JSONObject(myEnumClass);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("expected 2 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 2);
|
||||||
|
assertTrue("expected VAL3", "VAL3".equals((JsonPath.read(doc, "$.myEnumField"))));
|
||||||
|
assertTrue("expected VAL1", "VAL1".equals((JsonPath.read(doc, "$.myEnum"))));
|
||||||
|
|
||||||
|
String [] names = JSONObject.getNames(myEnum);
|
||||||
|
jsonObject = new JSONObject(myEnum, names);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("expected 3 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 3);
|
||||||
|
assertTrue("expected VAL1", MyEnum.VAL1.equals(jsonObject.query("/VAL1")));
|
||||||
|
assertTrue("expected VAL2", MyEnum.VAL2.equals(jsonObject.query("/VAL2")));
|
||||||
|
assertTrue("expected VAL3", MyEnum.VAL3.equals(jsonObject.query("/VAL3")));
|
||||||
|
|
||||||
|
names = JSONObject.getNames(myEnumField);
|
||||||
|
jsonObject = new JSONObject(myEnumField, names);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("expected 3 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 3);
|
||||||
|
assertTrue("expected VAL1", MyEnumField.VAL1.equals(jsonObject.query("/VAL1")));
|
||||||
|
assertTrue("expected VAL2", MyEnumField.VAL2.equals(jsonObject.query("/VAL2")));
|
||||||
|
assertTrue("expected VAL3", MyEnumField.VAL3.equals(jsonObject.query("/VAL3")));
|
||||||
|
|
||||||
|
expectedStr = "{\"myEnum\":\"VAL2\", \"myEnumField\":\"VAL2\"}";
|
||||||
|
jsonObject = new JSONObject();
|
||||||
|
jsonObject.putOpt("myEnum", myEnum);
|
||||||
|
jsonObject.putOnce("myEnumField", myEnumField);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("expected 2 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 2);
|
||||||
|
assertTrue("expected VAL2", MyEnum.VAL2.equals(jsonObject.query("/myEnum")));
|
||||||
|
assertTrue("expected VAL2", MyEnumField.VAL2.equals(jsonObject.query("/myEnumField")));
|
||||||
|
|
||||||
|
JSONArray jsonArray = new JSONArray();
|
||||||
|
jsonArray.put(myEnum);
|
||||||
|
jsonArray.put(1, myEnumField);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonArray.toString());
|
||||||
|
assertTrue("expected 2 top level items", ((List<?>)(JsonPath.read(doc, "$"))).size() == 2);
|
||||||
|
assertTrue("expected VAL2", MyEnum.VAL2.equals(jsonArray.query("/0")));
|
||||||
|
assertTrue("expected VAL2", MyEnumField.VAL2.equals(jsonArray.query("/1")));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wrap should handle enums exactly as a value type like Integer, Boolean, or String.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void wrap() {
|
||||||
|
assertTrue("simple enum has no getters", JSONObject.wrap(MyEnum.VAL2) instanceof MyEnum);
|
||||||
|
|
||||||
|
MyEnumField myEnumField = MyEnumField.VAL2;
|
||||||
|
JSONObject jsonObject = new JSONObject();
|
||||||
|
jsonObject.put("enum",myEnumField);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
Object doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("expected 1 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 1);
|
||||||
|
assertTrue("expected VAL2", MyEnumField.VAL2.equals(jsonObject.query("/enum")));
|
||||||
|
|
||||||
|
MyEnumClass myEnumClass = new MyEnumClass();
|
||||||
|
myEnumClass.setMyEnum(MyEnum.VAL1);
|
||||||
|
myEnumClass.setMyEnumField(MyEnumField.VAL3);
|
||||||
|
jsonObject = (JSONObject)JSONObject.wrap(myEnumClass);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("expected 2 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 2);
|
||||||
|
assertTrue("expected VAL3", "VAL3".equals((JsonPath.read(doc, "$.myEnumField"))));
|
||||||
|
assertTrue("expected VAL1", "VAL1".equals((JsonPath.read(doc, "$.myEnum"))));
|
||||||
|
|
||||||
|
assertTrue("expecting MyEnumField.VAL3", MyEnumField.VAL3.equals(jsonObject.query("/myEnumField")));
|
||||||
|
assertTrue("expecting MyEnum.VAL1", MyEnum.VAL1.equals(jsonObject.query("/myEnum")));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* It was determined that some API methods should be added to
|
||||||
|
* support enums:<br>
|
||||||
|
* JSONObject.getEnum(class, key)<br>
|
||||||
|
* JSONObject.optEnum(class, key)<br>
|
||||||
|
* JSONObject.optEnum(class, key, default)<br>
|
||||||
|
* JSONArray.getEnum(class, index)<br>
|
||||||
|
* JSONArray.optEnum(class, index)<br>
|
||||||
|
* JSONArray.optEnum(class, index, default)<br>
|
||||||
|
* <p>
|
||||||
|
* Exercise these enum API methods on JSONObject and JSONArray
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void enumAPI() {
|
||||||
|
MyEnumClass myEnumClass = new MyEnumClass();
|
||||||
|
myEnumClass.setMyEnum(MyEnum.VAL1);
|
||||||
|
MyEnumField myEnumField = MyEnumField.VAL2;
|
||||||
|
|
||||||
|
JSONObject jsonObject = new JSONObject();
|
||||||
|
jsonObject.put("strKey", "value");
|
||||||
|
jsonObject.put("strKey2", "VAL1");
|
||||||
|
jsonObject.put("enumKey", myEnumField);
|
||||||
|
jsonObject.put("enumClassKey", myEnumClass);
|
||||||
|
|
||||||
|
// get a plain old enum
|
||||||
|
MyEnumField actualEnum = jsonObject.getEnum(MyEnumField.class, "enumKey");
|
||||||
|
assertTrue("get myEnumField", actualEnum == MyEnumField.VAL2);
|
||||||
|
|
||||||
|
// try to get the wrong value
|
||||||
|
try {
|
||||||
|
actualEnum = jsonObject.getEnum(MyEnumField.class, "strKey");
|
||||||
|
assertTrue("should throw an exception for wrong key", false);
|
||||||
|
} catch (Exception ignored) {}
|
||||||
|
|
||||||
|
// get a class that contains an enum
|
||||||
|
MyEnumClass actualEnumClass = (MyEnumClass)jsonObject.get("enumClassKey");
|
||||||
|
assertTrue("get enum", actualEnumClass.getMyEnum() == MyEnum.VAL1);
|
||||||
|
|
||||||
|
// opt a plain old enum
|
||||||
|
actualEnum = jsonObject.optEnum(MyEnumField.class, "enumKey");
|
||||||
|
assertTrue("opt myEnumField", actualEnum == MyEnumField.VAL2);
|
||||||
|
|
||||||
|
// opt the wrong value
|
||||||
|
actualEnum = jsonObject.optEnum(MyEnumField.class, "strKey");
|
||||||
|
assertTrue("opt null", actualEnum == null);
|
||||||
|
|
||||||
|
// opt a class that contains an enum
|
||||||
|
actualEnumClass = (MyEnumClass)jsonObject.opt("enumClassKey");
|
||||||
|
assertTrue("get enum", actualEnumClass.getMyEnum() == MyEnum.VAL1);
|
||||||
|
|
||||||
|
// opt with default a plain old enum
|
||||||
|
actualEnum = jsonObject.optEnum(MyEnumField.class, "enumKey", null);
|
||||||
|
assertTrue("opt myEnumField", actualEnum == MyEnumField.VAL2);
|
||||||
|
|
||||||
|
// opt with default the wrong value
|
||||||
|
actualEnum = jsonObject.optEnum(MyEnumField.class, "strKey", null);
|
||||||
|
assertNull("opt null", actualEnum);
|
||||||
|
|
||||||
|
// opt with default the string value
|
||||||
|
actualEnum = jsonObject.optEnum(MyEnumField.class, "strKey2", null);
|
||||||
|
assertEquals(MyEnumField.VAL1, actualEnum);
|
||||||
|
|
||||||
|
// opt with default an index that does not exist
|
||||||
|
actualEnum = jsonObject.optEnum(MyEnumField.class, "noKey", null);
|
||||||
|
assertNull("opt null", actualEnum);
|
||||||
|
|
||||||
|
assertNull("Expected Null when the enum class is null",
|
||||||
|
jsonObject.optEnum(null, "enumKey"));
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Exercise the proposed enum API methods on JSONArray
|
||||||
|
*/
|
||||||
|
JSONArray jsonArray = new JSONArray();
|
||||||
|
jsonArray.put("value");
|
||||||
|
jsonArray.put(myEnumField);
|
||||||
|
jsonArray.put(myEnumClass);
|
||||||
|
|
||||||
|
// get a plain old enum
|
||||||
|
actualEnum = jsonArray.getEnum(MyEnumField.class, 1);
|
||||||
|
assertTrue("get myEnumField", actualEnum == MyEnumField.VAL2);
|
||||||
|
|
||||||
|
// try to get the wrong value
|
||||||
|
try {
|
||||||
|
actualEnum = jsonArray.getEnum(MyEnumField.class, 0);
|
||||||
|
assertTrue("should throw an exception for wrong index", false);
|
||||||
|
} catch (Exception ignored) {}
|
||||||
|
|
||||||
|
// get a class that contains an enum
|
||||||
|
actualEnumClass = (MyEnumClass)jsonArray.get(2);
|
||||||
|
assertTrue("get enum", actualEnumClass.getMyEnum() == MyEnum.VAL1);
|
||||||
|
|
||||||
|
// opt a plain old enum
|
||||||
|
actualEnum = jsonArray.optEnum(MyEnumField.class, 1);
|
||||||
|
assertTrue("opt myEnumField", actualEnum == MyEnumField.VAL2);
|
||||||
|
|
||||||
|
// opt the wrong value
|
||||||
|
actualEnum = jsonArray.optEnum(MyEnumField.class, 0);
|
||||||
|
assertTrue("opt null", actualEnum == null);
|
||||||
|
|
||||||
|
// opt a class that contains an enum
|
||||||
|
actualEnumClass = (MyEnumClass)jsonArray.opt(2);
|
||||||
|
assertTrue("get enum", actualEnumClass.getMyEnum() == MyEnum.VAL1);
|
||||||
|
|
||||||
|
// opt with default a plain old enum
|
||||||
|
actualEnum = jsonArray.optEnum(MyEnumField.class, 1, null);
|
||||||
|
assertTrue("opt myEnumField", actualEnum == MyEnumField.VAL2);
|
||||||
|
|
||||||
|
// opt with default the wrong value
|
||||||
|
actualEnum = jsonArray.optEnum(MyEnumField.class, 0, null);
|
||||||
|
assertTrue("opt null", actualEnum == null);
|
||||||
|
|
||||||
|
// opt with default an index that does not exist
|
||||||
|
actualEnum = jsonArray.optEnum(MyEnumField.class, 3, null);
|
||||||
|
assertTrue("opt null", actualEnum == null);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
220
src/test/java/org/json/junit/HTTPTest.java
Normal file
220
src/test/java/org/json/junit/HTTPTest.java
Normal file
|
@ -0,0 +1,220 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
|
import org.json.*;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Unit tests for JSON-Java HTTP.java. See RFC7230.
|
||||||
|
*/
|
||||||
|
public class HTTPTest {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempt to call HTTP.toJSONObject() with a null string
|
||||||
|
* Expects a NUllPointerException.
|
||||||
|
*/
|
||||||
|
@Test(expected=NullPointerException.class)
|
||||||
|
public void nullHTTPException() {
|
||||||
|
String httpStr = null;
|
||||||
|
HTTP.toJSONObject(httpStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempt to call HTTP.toJSONObject() with a string containing
|
||||||
|
* an empty object. Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void notEnoughHTTPException() {
|
||||||
|
String httpStr = "{}";
|
||||||
|
JSONObject jsonObject = new JSONObject(httpStr);
|
||||||
|
try {
|
||||||
|
HTTP.toString(jsonObject);
|
||||||
|
assertTrue("Expected to throw exception", false);
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Expecting an exception message",
|
||||||
|
"Not enough material for an HTTP header.".equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Calling HTTP.toJSONObject() with an empty string will result in a
|
||||||
|
* populated JSONObject with keys but no values for Request-URI, Method,
|
||||||
|
* and HTTP-Version.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void emptyStringHTTPRequest() {
|
||||||
|
String httpStr = "";
|
||||||
|
String expectedHTTPStr = "{\"Request-URI\":\"\",\"Method\":\"\",\"HTTP-Version\":\"\"}";
|
||||||
|
JSONObject jsonObject = HTTP.toJSONObject(httpStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedHTTPStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Call HTTP.toJSONObject() with a Request-URI, Method,
|
||||||
|
* and HTTP-Version.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void simpleHTTPRequest() {
|
||||||
|
String httpStr = "GET /hello.txt HTTP/1.1";
|
||||||
|
String expectedHTTPStr =
|
||||||
|
"{\"Request-URI\":\"/hello.txt\",\"Method\":\"GET\",\"HTTP-Version\":\"HTTP/1.1\"}";
|
||||||
|
JSONObject jsonObject = HTTP.toJSONObject(httpStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedHTTPStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Call HTTP.toJSONObject() with a response string containing a
|
||||||
|
* HTTP-Version, Status-Code, and Reason.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void simpleHTTPResponse() {
|
||||||
|
String httpStr = "HTTP/1.1 200 OK";
|
||||||
|
String expectedHTTPStr =
|
||||||
|
"{\"HTTP-Version\":\"HTTP/1.1\",\"Status-Code\":\"200\",\"Reason-Phrase\":\"OK\"}";
|
||||||
|
JSONObject jsonObject = HTTP.toJSONObject(httpStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedHTTPStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Call HTTP.toJSONObject() with a full request string including
|
||||||
|
* request headers.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void extendedHTTPRequest() {
|
||||||
|
String httpStr =
|
||||||
|
"POST /enlighten/calais.asmx HTTP/1.1\n"+
|
||||||
|
"Host: api.opencalais.com\n"+
|
||||||
|
"Content-Type: text/xml; charset=utf-8\n"+
|
||||||
|
"Content-Length: 100\n"+
|
||||||
|
"SOAPAction: \"http://clearforest.com/Enlighten\"";
|
||||||
|
String expectedHTTPStr =
|
||||||
|
"{"+
|
||||||
|
"\"Request-URI\":\"/enlighten/calais.asmx\","+
|
||||||
|
"\"Host\":\"api.opencalais.com\","+
|
||||||
|
"\"Method\":\"POST\","+
|
||||||
|
"\"HTTP-Version\":\"HTTP/1.1\","+
|
||||||
|
"\"Content-Length\":\"100\","+
|
||||||
|
"\"Content-Type\":\"text/xml; charset=utf-8\"}";
|
||||||
|
JSONObject jsonObject = HTTP.toJSONObject(httpStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedHTTPStr);
|
||||||
|
/**
|
||||||
|
* Not too easy for JSONObject to parse a string with embedded quotes.
|
||||||
|
* For the sake of the test, add it here.
|
||||||
|
*/
|
||||||
|
expectedJsonObject.put("SOAPAction","\"http://clearforest.com/Enlighten\"");
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Call HTTP.toJSONObject() with a full response string including
|
||||||
|
* response headers.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void extendedHTTPResponse() {
|
||||||
|
String httpStr =
|
||||||
|
"HTTP/1.1 200 OK\n"+
|
||||||
|
"Content-Type: text/xml; charset=utf-8\n"+
|
||||||
|
"Content-Length: 100\n";
|
||||||
|
String expectedHTTPStr =
|
||||||
|
"{\"HTTP-Version\":\"HTTP/1.1\","+
|
||||||
|
"\"Status-Code\":\"200\","+
|
||||||
|
"\"Content-Length\":\"100\","+
|
||||||
|
"\"Reason-Phrase\":\"OK\","+
|
||||||
|
"\"Content-Type\":\"text/xml; charset=utf-8\"}";
|
||||||
|
JSONObject jsonObject = HTTP.toJSONObject(httpStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedHTTPStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Call HTTP.toJSONObject() with a full POST request string including
|
||||||
|
* response headers, then convert it back into an HTTP string.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void convertHTTPRequestToString() {
|
||||||
|
String httpStr =
|
||||||
|
"POST /enlighten/calais.asmx HTTP/1.1\n"+
|
||||||
|
"Host: api.opencalais.com\n"+
|
||||||
|
"Content-Type: text/xml; charset=utf-8\n"+
|
||||||
|
"Content-Length: 100";
|
||||||
|
String expectedHTTPStr =
|
||||||
|
"{"+
|
||||||
|
"\"Request-URI\":\"/enlighten/calais.asmx\","+
|
||||||
|
"\"Host\":\"api.opencalais.com\","+
|
||||||
|
"\"Method\":\"POST\","+
|
||||||
|
"\"HTTP-Version\":\"HTTP/1.1\","+
|
||||||
|
"\"Content-Length\":\"100\","+
|
||||||
|
"\"Content-Type\":\"text/xml; charset=utf-8\"}";
|
||||||
|
JSONObject jsonObject = HTTP.toJSONObject(httpStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedHTTPStr);
|
||||||
|
String httpToStr = HTTP.toString(jsonObject);
|
||||||
|
/**
|
||||||
|
* JSONObject objects to crlfs and any trailing chars.
|
||||||
|
* For the sake of the test, simplify the resulting string
|
||||||
|
*/
|
||||||
|
httpToStr = httpToStr.replaceAll("("+HTTP.CRLF+HTTP.CRLF+")", "");
|
||||||
|
httpToStr = httpToStr.replaceAll(HTTP.CRLF, "\n");
|
||||||
|
JSONObject finalJsonObject = HTTP.toJSONObject(httpToStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(finalJsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Call HTTP.toJSONObject() with a full response string including
|
||||||
|
* response headers, then convert it back into an HTTP string.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void convertHTTPResponseToString() {
|
||||||
|
String httpStr =
|
||||||
|
"HTTP/1.1 200 OK\n"+
|
||||||
|
"Content-Type: text/xml; charset=utf-8\n"+
|
||||||
|
"Content-Length: 100\n";
|
||||||
|
String expectedHTTPStr =
|
||||||
|
"{\"HTTP-Version\":\"HTTP/1.1\","+
|
||||||
|
"\"Status-Code\":\"200\","+
|
||||||
|
"\"Content-Length\":\"100\","+
|
||||||
|
"\"Reason-Phrase\":\"OK\","+
|
||||||
|
"\"Content-Type\":\"text/xml; charset=utf-8\"}";
|
||||||
|
JSONObject jsonObject = HTTP.toJSONObject(httpStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedHTTPStr);
|
||||||
|
String httpToStr = HTTP.toString(jsonObject);
|
||||||
|
/**
|
||||||
|
* JSONObject objects to crlfs and any trailing chars.
|
||||||
|
* For the sake of the test, simplify the resulting string
|
||||||
|
*/
|
||||||
|
httpToStr = httpToStr.replaceAll("("+HTTP.CRLF+HTTP.CRLF+")", "");
|
||||||
|
httpToStr = httpToStr.replaceAll(HTTP.CRLF, "\n");
|
||||||
|
JSONObject finalJsonObject = HTTP.toJSONObject(httpToStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(finalJsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
}
|
1181
src/test/java/org/json/junit/JSONArrayTest.java
Normal file
1181
src/test/java/org/json/junit/JSONArrayTest.java
Normal file
File diff suppressed because it is too large
Load diff
856
src/test/java/org/json/junit/JSONMLTest.java
Normal file
856
src/test/java/org/json/junit/JSONMLTest.java
Normal file
|
@ -0,0 +1,856 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
|
import org.json.*;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for org.json.JSONML.java
|
||||||
|
*
|
||||||
|
* Certain inputs are expected to result in exceptions. These tests are
|
||||||
|
* executed first. JSONML provides an API to:
|
||||||
|
* Convert an XML string into a JSONArray or a JSONObject.
|
||||||
|
* Convert a JSONArray or JSONObject into an XML string.
|
||||||
|
* Both fromstring and tostring operations operations should be symmetrical
|
||||||
|
* within the limits of JSONML.
|
||||||
|
* It should be possible to perform the following operations, which should
|
||||||
|
* result in the original string being recovered, within the limits of the
|
||||||
|
* underlying classes:
|
||||||
|
* Convert a string -> JSONArray -> string -> JSONObject -> string
|
||||||
|
* Convert a string -> JSONObject -> string -> JSONArray -> string
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class JSONMLTest {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to transform a null XML string to JSON.
|
||||||
|
* Expects a NullPointerException
|
||||||
|
*/
|
||||||
|
@Test(expected=NullPointerException.class)
|
||||||
|
public void nullXMLException() {
|
||||||
|
String xmlStr = null;
|
||||||
|
JSONML.toJSONArray(xmlStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to transform an empty string to JSON.
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void emptyXMLException() {
|
||||||
|
String xmlStr = "";
|
||||||
|
try {
|
||||||
|
JSONML.toJSONArray(xmlStr);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Bad XML at 0 [character 1 line 1]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to call JSONML.toString() with a null JSONArray.
|
||||||
|
* Expects a NullPointerException.
|
||||||
|
*/
|
||||||
|
@Test(expected=NullPointerException.class)
|
||||||
|
public void nullJSONXMLException() {
|
||||||
|
/**
|
||||||
|
* Tries to convert a null JSONArray to XML.
|
||||||
|
*/
|
||||||
|
JSONArray jsonArray= null;
|
||||||
|
JSONML.toString(jsonArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to call JSONML.toString() with a null JSONArray.
|
||||||
|
* Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void emptyJSONXMLException() {
|
||||||
|
/**
|
||||||
|
* Tries to convert an empty JSONArray to XML.
|
||||||
|
*/
|
||||||
|
JSONArray jsonArray = new JSONArray();
|
||||||
|
try {
|
||||||
|
JSONML.toString(jsonArray);
|
||||||
|
assertTrue("Expecting an exception", false);
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Expecting an exception message",
|
||||||
|
"JSONArray[0] not found.".
|
||||||
|
equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to transform an non-XML string to JSON.
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void nonXMLException() {
|
||||||
|
/**
|
||||||
|
* Attempts to transform a nonXML string to JSON
|
||||||
|
*/
|
||||||
|
String xmlStr = "{ \"this is\": \"not xml\"}";
|
||||||
|
try {
|
||||||
|
JSONML.toJSONArray(xmlStr);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Bad XML at 23 [character 24 line 1]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to transform a JSON document with XML content that
|
||||||
|
* does not follow JSONML conventions (element name is not first value
|
||||||
|
* in a nested JSONArray) to a JSONArray then back to string.
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void emptyTagException() {
|
||||||
|
/**
|
||||||
|
* jsonArrayStr is used to build a JSONArray which is then
|
||||||
|
* turned into XML. For this transformation, all arrays represent
|
||||||
|
* elements and the first array entry is the name of the element.
|
||||||
|
* In this case, one of the arrays does not have a name
|
||||||
|
*/
|
||||||
|
String jsonArrayStr =
|
||||||
|
"[\"addresses\","+
|
||||||
|
"{\"xsi:noNamespaceSchemaLocation\":\"test.xsd\","+
|
||||||
|
"\"xmlns:xsi\":\"http://www.w3.org/2001/XMLSchema-instance\"},"+
|
||||||
|
// this array has no name
|
||||||
|
"["+
|
||||||
|
"[\"name\"],"+
|
||||||
|
"[\"nocontent\"],"+
|
||||||
|
"\">\""+
|
||||||
|
"]"+
|
||||||
|
"]";
|
||||||
|
JSONArray jsonArray = new JSONArray(jsonArrayStr);
|
||||||
|
try {
|
||||||
|
JSONML.toString(jsonArray);
|
||||||
|
assertTrue("Expecting an exception", false);
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"JSONArray[0] is not a String.",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to transform a JSON document with XML content that
|
||||||
|
* does not follow JSONML conventions (element tag has an embedded space)
|
||||||
|
* to a JSONArray then back to string. Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void spaceInTagException() {
|
||||||
|
/**
|
||||||
|
* jsonArrayStr is used to build a JSONArray which is then
|
||||||
|
* turned into XML. For this transformation, all arrays represent
|
||||||
|
* elements and the first array entry is the name of the element.
|
||||||
|
* In this case, one of the element names has an embedded space,
|
||||||
|
* which is not allowed.
|
||||||
|
*/
|
||||||
|
String jsonArrayStr =
|
||||||
|
"[\"addresses\","+
|
||||||
|
"{\"xsi:noNamespaceSchemaLocation\":\"test.xsd\","+
|
||||||
|
"\"xmlns:xsi\":\"http://www.w3.org/2001/XMLSchema-instance\"},"+
|
||||||
|
// this array has an invalid name
|
||||||
|
"[\"addr esses\","+
|
||||||
|
"[\"name\"],"+
|
||||||
|
"[\"nocontent\"],"+
|
||||||
|
"\">\""+
|
||||||
|
"]"+
|
||||||
|
"]";
|
||||||
|
JSONArray jsonArray = new JSONArray(jsonArrayStr);
|
||||||
|
try {
|
||||||
|
JSONML.toString(jsonArray);
|
||||||
|
assertTrue("Expecting an exception", false);
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Expecting an exception message",
|
||||||
|
"'addr esses' contains a space character.".
|
||||||
|
equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to transform a malformed XML document
|
||||||
|
* (element tag has a frontslash) to a JSONArray.\
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void invalidSlashInTagException() {
|
||||||
|
/**
|
||||||
|
* xmlStr contains XML text which is transformed into a JSONArray.
|
||||||
|
* In this case, the XML is invalid because the 'name' element
|
||||||
|
* contains an invalid frontslash.
|
||||||
|
*/
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/x>\n"+
|
||||||
|
" <street>abc street</street>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
JSONML.toJSONArray(xmlStr);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Misshaped tag at 176 [character 14 line 4]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Malformed XML text (invalid tagname) is transformed into a JSONArray.
|
||||||
|
* Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void invalidBangInTagException() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/>\n"+
|
||||||
|
" <!>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
JSONML.toJSONArray(xmlStr);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Misshaped meta tag at 215 [character 12 line 7]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Malformed XML text (invalid tagname, no close bracket) is transformed\
|
||||||
|
* into a JSONArray. Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void invalidBangNoCloseInTagException() {
|
||||||
|
/**
|
||||||
|
* xmlStr contains XML text which is transformed into a JSONArray.
|
||||||
|
* In this case, the XML is invalid because an element
|
||||||
|
* starts with '!' and has no closing tag
|
||||||
|
*/
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/>\n"+
|
||||||
|
" <!\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
JSONML.toJSONArray(xmlStr);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Misshaped meta tag at 214 [character 12 line 7]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Malformed XML text (tagname with no close bracket) is transformed\
|
||||||
|
* into a JSONArray. Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void noCloseStartTagException() {
|
||||||
|
/**
|
||||||
|
* xmlStr contains XML text which is transformed into a JSONArray.
|
||||||
|
* In this case, the XML is invalid because an element
|
||||||
|
* has no closing '>'.
|
||||||
|
*/
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/>\n"+
|
||||||
|
" <abc\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
JSONML.toJSONArray(xmlStr);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Misplaced '<' at 194 [character 5 line 6]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Malformed XML text (endtag with no name) is transformed\
|
||||||
|
* into a JSONArray. Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void noCloseEndTagException() {
|
||||||
|
/**
|
||||||
|
* xmlStr contains XML text which is transformed into a JSONArray.
|
||||||
|
* In this case, the XML is invalid because an element
|
||||||
|
* has no name after the closing tag '</'.
|
||||||
|
*/
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/>\n"+
|
||||||
|
" <abc/>\n"+
|
||||||
|
" </>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
JSONML.toJSONArray(xmlStr);
|
||||||
|
assertTrue("Expecting an exception", false);
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Expecting an exception message",
|
||||||
|
"Expected a closing name instead of '>'.".
|
||||||
|
equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Malformed XML text (endtag with no close bracket) is transformed\
|
||||||
|
* into a JSONArray. Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void noCloseEndBraceException() {
|
||||||
|
/**
|
||||||
|
* xmlStr contains XML text which is transformed into a JSONArray.
|
||||||
|
* In this case, the XML is invalid because an element
|
||||||
|
* has '>' after the closing tag '</' and name.
|
||||||
|
*/
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation=\"test.xsd\">\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/>\n"+
|
||||||
|
" <abc/>\n"+
|
||||||
|
" </address\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
JSONML.toJSONArray(xmlStr);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Misplaced '<' at 206 [character 1 line 7]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Malformed XML text (incomplete CDATA string) is transformed\
|
||||||
|
* into a JSONArray. Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void invalidCDATABangInTagException() {
|
||||||
|
/**
|
||||||
|
* xmlStr contains XML text which is transformed into a JSONArray.
|
||||||
|
* In this case, the XML is invalid because an element
|
||||||
|
* does not have a complete CDATA string.
|
||||||
|
*/
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name>Joe Tester</name>\n"+
|
||||||
|
" <![[]>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
JSONML.toJSONArray(xmlStr);
|
||||||
|
fail("Expecting an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Expected 'CDATA[' at 204 [character 11 line 5]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert an XML document into a JSONArray, then use JSONML.toString()
|
||||||
|
* to convert it into a string. This string is then converted back into
|
||||||
|
* a JSONArray. Both JSONArrays are compared against a control to
|
||||||
|
* confirm the contents.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void toJSONArray() {
|
||||||
|
/**
|
||||||
|
* xmlStr contains XML text which is transformed into a JSONArray.
|
||||||
|
* Each element becomes a JSONArray:
|
||||||
|
* 1st entry = elementname
|
||||||
|
* 2nd entry = attributes object (if present)
|
||||||
|
* 3rd entry = content (if present)
|
||||||
|
* 4th entry = child element JSONArrays (if present)
|
||||||
|
* The result is compared against an expected JSONArray.
|
||||||
|
* The transformed JSONArray is then transformed back into a string
|
||||||
|
* which is used to create a final JSONArray, which is also compared
|
||||||
|
* against the expected JSONArray.
|
||||||
|
*/
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
"xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
"<address attr1=\"attrValue1\" attr2=\"attrValue2\" attr3=\"attrValue3\">\n"+
|
||||||
|
"<name nameType=\"mine\">myName</name>\n"+
|
||||||
|
"<nocontent/>>\n"+
|
||||||
|
"</address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
String expectedStr =
|
||||||
|
"[\"addresses\","+
|
||||||
|
"{\"xsi:noNamespaceSchemaLocation\":\"test.xsd\","+
|
||||||
|
"\"xmlns:xsi\":\"http://www.w3.org/2001/XMLSchema-instance\"},"+
|
||||||
|
"[\"address\","+
|
||||||
|
"{\"attr1\":\"attrValue1\",\"attr2\":\"attrValue2\",\"attr3\":\"attrValue3\"},"+
|
||||||
|
"[\"name\", {\"nameType\":\"mine\"},\"myName\"],"+
|
||||||
|
"[\"nocontent\"],"+
|
||||||
|
"\">\""+
|
||||||
|
"]"+
|
||||||
|
"]";
|
||||||
|
JSONArray jsonArray = JSONML.toJSONArray(xmlStr);
|
||||||
|
JSONArray expectedJsonArray = new JSONArray(expectedStr);
|
||||||
|
String xmlToStr = JSONML.toString(jsonArray);
|
||||||
|
JSONArray finalJsonArray = JSONML.toJSONArray(xmlToStr);
|
||||||
|
Util.compareActualVsExpectedJsonArrays(jsonArray, expectedJsonArray);
|
||||||
|
Util.compareActualVsExpectedJsonArrays(finalJsonArray, expectedJsonArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert an XML document into a JSONObject. Use JSONML.toString() to
|
||||||
|
* convert it back into a string, and then re-convert it into a JSONObject.
|
||||||
|
* Both JSONObjects are compared against a control JSONObject to confirm
|
||||||
|
* the contents.
|
||||||
|
* <p>
|
||||||
|
* Next convert the XML document into a JSONArray. Use JSONML.toString() to
|
||||||
|
* convert it back into a string, and then re-convert it into a JSONArray.
|
||||||
|
* Both JSONArrays are compared against a control JSONArray to confirm
|
||||||
|
* the contents.
|
||||||
|
* <p>
|
||||||
|
* This test gives a comprehensive example of how the JSONML
|
||||||
|
* transformations work.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void toJSONObjectToJSONArray() {
|
||||||
|
/**
|
||||||
|
* xmlStr contains XML text which is transformed into a JSONObject,
|
||||||
|
* restored to XML, transformed into a JSONArray, and then restored
|
||||||
|
* to XML again. Both JSONObject and JSONArray should contain the same
|
||||||
|
* information and should produce the same XML, allowing for non-ordered
|
||||||
|
* attributes.
|
||||||
|
*
|
||||||
|
* Transformation to JSONObject:
|
||||||
|
* The elementName is stored as a string where key="tagName"
|
||||||
|
* Attributes are simply stored as key/value pairs
|
||||||
|
* If the element has either content or child elements, they are stored
|
||||||
|
* in a jsonArray with key="childNodes".
|
||||||
|
*
|
||||||
|
* Transformation to JSONArray:
|
||||||
|
* 1st entry = elementname
|
||||||
|
* 2nd entry = attributes object (if present)
|
||||||
|
* 3rd entry = content (if present)
|
||||||
|
* 4th entry = child element JSONArrays (if present)
|
||||||
|
*/
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
"xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
"<address addrType=\"my address\">\n"+
|
||||||
|
"<name nameType=\"my name\">Joe Tester</name>\n"+
|
||||||
|
"<street><![CDATA[Baker street 5]]></street>\n"+
|
||||||
|
"<NothingHere except=\"an attribute\"/>\n"+
|
||||||
|
"<TrueValue>true</TrueValue>\n"+
|
||||||
|
"<FalseValue>false</FalseValue>\n"+
|
||||||
|
"<NullValue>null</NullValue>\n"+
|
||||||
|
"<PositiveValue>42</PositiveValue>\n"+
|
||||||
|
"<NegativeValue>-23</NegativeValue>\n"+
|
||||||
|
"<DoubleValue>-23.45</DoubleValue>\n"+
|
||||||
|
"<Nan>-23x.45</Nan>\n"+
|
||||||
|
"<ArrayOfNum>\n"+
|
||||||
|
"<value>1</value>\n"+
|
||||||
|
"<value>2</value>\n"+
|
||||||
|
"<value><subValue svAttr=\"svValue\">abc</subValue></value>\n"+
|
||||||
|
"<value>3</value>\n"+
|
||||||
|
"<value>4.1</value>\n"+
|
||||||
|
"<value>5.2</value>\n"+
|
||||||
|
"</ArrayOfNum>\n"+
|
||||||
|
"</address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
|
||||||
|
String expectedJSONObjectStr =
|
||||||
|
"{"+
|
||||||
|
"\"xsi:noNamespaceSchemaLocation\":\"test.xsd\","+
|
||||||
|
"\"childNodes\":["+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":["+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[\"Joe Tester\"],"+
|
||||||
|
"\"nameType\":\"my name\","+
|
||||||
|
"\"tagName\":\"name\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[\"Baker street 5\"],"+
|
||||||
|
"\"tagName\":\"street\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"tagName\":\"NothingHere\","+
|
||||||
|
"\"except\":\"an attribute\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[true],"+
|
||||||
|
"\"tagName\":\"TrueValue\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[false],"+
|
||||||
|
"\"tagName\":\"FalseValue\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[null],"+
|
||||||
|
"\"tagName\":\"NullValue\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[42],"+
|
||||||
|
"\"tagName\":\"PositiveValue\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[-23],"+
|
||||||
|
"\"tagName\":\"NegativeValue\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[-23.45],"+
|
||||||
|
"\"tagName\":\"DoubleValue\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[\"-23x.45\"],"+
|
||||||
|
"\"tagName\":\"Nan\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":["+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[1],"+
|
||||||
|
"\"tagName\":\"value\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[2],"+
|
||||||
|
"\"tagName\":\"value\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":["+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[\"abc\"],"+
|
||||||
|
"\"svAttr\":\"svValue\","+
|
||||||
|
"\"tagName\":\"subValue\""+
|
||||||
|
"}"+
|
||||||
|
"],"+
|
||||||
|
"\"tagName\":\"value\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[3],"+
|
||||||
|
"\"tagName\":\"value\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[4.1],"+
|
||||||
|
"\"tagName\":\"value\""+
|
||||||
|
"},"+
|
||||||
|
"{"+
|
||||||
|
"\"childNodes\":[5.2],"+
|
||||||
|
"\"tagName\":\"value\""+
|
||||||
|
"}"+
|
||||||
|
"],"+
|
||||||
|
"\"tagName\":\"ArrayOfNum\""+
|
||||||
|
"}"+
|
||||||
|
"],"+
|
||||||
|
"\"addrType\":\"my address\","+
|
||||||
|
"\"tagName\":\"address\""+
|
||||||
|
"}"+
|
||||||
|
"],"+
|
||||||
|
"\"xmlns:xsi\":\"http://www.w3.org/2001/XMLSchema-instance\","+
|
||||||
|
"\"tagName\":\"addresses\""+
|
||||||
|
"}";
|
||||||
|
|
||||||
|
String expectedJSONArrayStr =
|
||||||
|
"["+
|
||||||
|
"\"addresses\","+
|
||||||
|
"{"+
|
||||||
|
"\"xsi:noNamespaceSchemaLocation\":\"test.xsd\","+
|
||||||
|
"\"xmlns:xsi\":\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
"},"+
|
||||||
|
"["+
|
||||||
|
"\"address\","+
|
||||||
|
"{"+
|
||||||
|
"\"addrType\":\"my address\""+
|
||||||
|
"},"+
|
||||||
|
"["+
|
||||||
|
"\"name\","+
|
||||||
|
"{"+
|
||||||
|
"\"nameType\":\"my name\""+
|
||||||
|
"},"+
|
||||||
|
"\"Joe Tester\""+
|
||||||
|
"],"+
|
||||||
|
"[\"street\",\"Baker street 5\"],"+
|
||||||
|
"["+
|
||||||
|
"\"NothingHere\","+
|
||||||
|
"{\"except\":\"an attribute\"}"+
|
||||||
|
"],"+
|
||||||
|
"[\"TrueValue\",true],"+
|
||||||
|
"[\"FalseValue\",false],"+
|
||||||
|
"[\"NullValue\",null],"+
|
||||||
|
"[\"PositiveValue\",42],"+
|
||||||
|
"[\"NegativeValue\",-23],"+
|
||||||
|
"[\"DoubleValue\",-23.45],"+
|
||||||
|
"[\"Nan\",\"-23x.45\"],"+
|
||||||
|
"["+
|
||||||
|
"\"ArrayOfNum\","+
|
||||||
|
"[\"value\",1],"+
|
||||||
|
"[\"value\",2],"+
|
||||||
|
"[\"value\","+
|
||||||
|
"["+
|
||||||
|
"\"subValue\","+
|
||||||
|
"{\"svAttr\":\"svValue\"},"+
|
||||||
|
"\"abc\""+
|
||||||
|
"],"+
|
||||||
|
"],"+
|
||||||
|
"[\"value\",3],"+
|
||||||
|
"[\"value\",4.1],"+
|
||||||
|
"[\"value\",5.2]"+
|
||||||
|
"]"+
|
||||||
|
"]"+
|
||||||
|
"]";
|
||||||
|
|
||||||
|
// make a JSONObject and make sure it looks as expected
|
||||||
|
JSONObject jsonObject = JSONML.toJSONObject(xmlStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedJSONObjectStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
|
||||||
|
// restore the XML, then make another JSONObject and make sure it
|
||||||
|
// looks as expected
|
||||||
|
String jsonObjectXmlToStr = JSONML.toString(jsonObject);
|
||||||
|
JSONObject finalJsonObject = JSONML.toJSONObject(jsonObjectXmlToStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(finalJsonObject, expectedJsonObject);
|
||||||
|
|
||||||
|
// create a JSON array from the original string and make sure it
|
||||||
|
// looks as expected
|
||||||
|
JSONArray jsonArray = JSONML.toJSONArray(xmlStr);
|
||||||
|
JSONArray expectedJsonArray = new JSONArray(expectedJSONArrayStr);
|
||||||
|
Util.compareActualVsExpectedJsonArrays(jsonArray,expectedJsonArray);
|
||||||
|
|
||||||
|
// restore the XML, then make another JSONArray and make sure it
|
||||||
|
// looks as expected
|
||||||
|
String jsonArrayXmlToStr = JSONML.toString(jsonArray);
|
||||||
|
JSONArray finalJsonArray = JSONML.toJSONArray(jsonArrayXmlToStr);
|
||||||
|
Util.compareActualVsExpectedJsonArrays(finalJsonArray, expectedJsonArray);
|
||||||
|
|
||||||
|
// lastly, confirm the restored JSONObject XML and JSONArray XML look
|
||||||
|
// reasonably similar
|
||||||
|
JSONObject jsonObjectFromObject = JSONML.toJSONObject(jsonObjectXmlToStr);
|
||||||
|
JSONObject jsonObjectFromArray = JSONML.toJSONObject(jsonArrayXmlToStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObjectFromObject, jsonObjectFromArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert an XML document which contains embedded comments into
|
||||||
|
* a JSONArray. Use JSONML.toString() to turn it into a string, then
|
||||||
|
* reconvert it into a JSONArray. Compare both JSONArrays to a control
|
||||||
|
* JSONArray to confirm the contents.
|
||||||
|
* <p>
|
||||||
|
* This test shows how XML comments are handled.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void commentsInXML() {
|
||||||
|
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<!-- this is a comment -->\n"+
|
||||||
|
"<addresses>\n"+
|
||||||
|
"<address>\n"+
|
||||||
|
"<!-- <!--[CDATA[ this is -- <another> comment ]] -->\n"+
|
||||||
|
"<name>Joe Tester</name>\n"+
|
||||||
|
"<!-- this is a - multi line \n"+
|
||||||
|
"comment -->\n"+
|
||||||
|
"<street>Baker street 5</street>\n"+
|
||||||
|
"</address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
String expectedStr =
|
||||||
|
"[\"addresses\","+
|
||||||
|
"[\"address\","+
|
||||||
|
"[\"name\",\"Joe Tester\"],"+
|
||||||
|
"[\"street\",\"Baker street 5\"]"+
|
||||||
|
"]"+
|
||||||
|
"]";
|
||||||
|
JSONArray jsonArray = JSONML.toJSONArray(xmlStr);
|
||||||
|
JSONArray expectedJsonArray = new JSONArray(expectedStr);
|
||||||
|
String xmlToStr = JSONML.toString(jsonArray);
|
||||||
|
JSONArray finalJsonArray = JSONML.toJSONArray(xmlToStr);
|
||||||
|
Util.compareActualVsExpectedJsonArrays(jsonArray, expectedJsonArray);
|
||||||
|
Util.compareActualVsExpectedJsonArrays(finalJsonArray, expectedJsonArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSON string with lost leading zero and converted "True" to true. See test
|
||||||
|
* result in comment below.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testToJSONArray_jsonOutput() {
|
||||||
|
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
||||||
|
final String expectedJsonString = "[\"root\",[\"id\",\"01\"],[\"id\",1],[\"id\",\"00\"],[\"id\",0],[\"item\",{\"id\":\"01\"}],[\"title\",true]]";
|
||||||
|
final JSONArray actualJsonOutput = JSONML.toJSONArray(originalXml, false);
|
||||||
|
assertEquals(expectedJsonString, actualJsonOutput.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSON string cannot be reverted to original xml when type guessing is used.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testToJSONArray_reversibility() {
|
||||||
|
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
||||||
|
final String revertedXml = JSONML.toString(JSONML.toJSONArray(originalXml, false));
|
||||||
|
assertNotEquals(revertedXml, originalXml);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSON string cannot be reverted to original xml when type guessing is used.
|
||||||
|
* When we force all the values as string, the original text comes back.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testToJSONArray_reversibility2() {
|
||||||
|
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
||||||
|
final String expectedJsonString = "[\"root\",[\"id\",\"01\"],[\"id\",\"1\"],[\"id\",\"00\"],[\"id\",\"0\"],[\"item\",{\"id\":\"01\"}],[\"title\",\"True\"]]";
|
||||||
|
final JSONArray json = JSONML.toJSONArray(originalXml,true);
|
||||||
|
assertEquals(expectedJsonString, json.toString());
|
||||||
|
|
||||||
|
final String reverseXml = JSONML.toString(json);
|
||||||
|
assertEquals(originalXml, reverseXml);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSON can be reverted to original xml.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testToJSONArray_reversibility3() {
|
||||||
|
final String originalXml = "<readResult><errors someAttr=\"arrtValue\"><code>400</code></errors><errors><code>402</code></errors></readResult>";
|
||||||
|
final JSONArray jsonArray = JSONML.toJSONArray(originalXml, false);
|
||||||
|
final String revertedXml = JSONML.toString(jsonArray);
|
||||||
|
assertEquals(revertedXml, originalXml);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSON string cannot be reverted to original xml. See test result in
|
||||||
|
* comment below.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testToJSONObject_reversibility() {
|
||||||
|
final String originalXml = "<readResult><errors someAttr=\"arrtValue\"><code>400</code></errors><errors><code>402</code></errors></readResult>";
|
||||||
|
final JSONObject originalObject=JSONML.toJSONObject(originalXml,false);
|
||||||
|
final String originalJson = originalObject.toString();
|
||||||
|
final String xml = JSONML.toString(originalObject);
|
||||||
|
final JSONObject revertedObject = JSONML.toJSONObject(xml, false);
|
||||||
|
final String newJson = revertedObject.toString();
|
||||||
|
assertTrue("JSON Objects are not similar",originalObject.similar(revertedObject));
|
||||||
|
assertEquals("original JSON does not equal the new JSON",originalJson, newJson);
|
||||||
|
}
|
||||||
|
|
||||||
|
// these tests do not pass for the following reasons:
|
||||||
|
// 1. Our XML parser does not handle generic HTML entities, only valid XML entities. Hence
|
||||||
|
// or other HTML specific entities would fail on reversability
|
||||||
|
// 2. Our JSON implementation for storing the XML attributes uses the standard unordered map.
|
||||||
|
// This means that <tag attr1="v1" attr2="v2" /> can not be reversed reliably.
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Test texts taken from jsonml.org. Currently our implementation FAILS this conversion but shouldn't.
|
||||||
|
// * Technically JsonML should be able to transform any valid xhtml document, but ours only supports
|
||||||
|
// * standard XML entities, not HTML entities.
|
||||||
|
// */
|
||||||
|
// @Test
|
||||||
|
// public void testAttributeConversionReversabilityHTML() {
|
||||||
|
// final String originalXml = "<table class=\"MyTable\" style=\"background-color:yellow\"><tr><td class=\"MyTD\" style=\"border:1px solid black\">#5D28D1</td><td class=\"MyTD\" style=\"background-color:red\">Example text here</td></tr><tr><td class=\"MyTD\" style=\"border:1px solid black\">#AF44EF</td><td class=\"MyTD\" style=\"background-color:green\">127310656</td></tr><tr><td class=\"MyTD\" style=\"border:1px solid black\">#AAD034</td><td class=\"MyTD\" style=\"background-color:blue\"> <span style=\"background-color:maroon\">©</span> </td></tr></table>";
|
||||||
|
// final String expectedJsonString = "[\"table\",{\"class\" : \"MyTable\",\"style\" : \"background-color:yellow\"},[\"tr\",[\"td\",{\"class\" : \"MyTD\",\"style\" : \"border:1px solid black\"},\"#550758\"],[\"td\",{\"class\" : \"MyTD\",\"style\" : \"background-color:red\"},\"Example text here\"]],[\"tr\",[\"td\",{\"class\" : \"MyTD\",\"style\" : \"border:1px solid black\"},\"#993101\"],[\"td\",{\"class\" : \"MyTD\",\"style\" : \"background-color:green\"},\"127624015\"]],[\"tr\",[\"td\",{\"class\" : \"MyTD\",\"style\" : \"border:1px solid black\"},\"#E33D87\"],[\"td\",{\"class\" : \"MyTD\",\"style\" : \"background-color:blue\"},\"\u00A0\",[\"span\",{ \"style\" : \"background-color:maroon\" },\"\u00A9\"],\"\u00A0\"]]]";
|
||||||
|
// final JSONArray json = JSONML.toJSONArray(originalXml,true);
|
||||||
|
// final String actualJsonString = json.toString();
|
||||||
|
//
|
||||||
|
// final String reverseXml = JSONML.toString(json);
|
||||||
|
// assertNotEquals(originalXml, reverseXml);
|
||||||
|
//
|
||||||
|
// assertNotEquals(expectedJsonString, actualJsonString);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Test texts taken from jsonml.org but modified to have XML entities only.
|
||||||
|
// */
|
||||||
|
// @Test
|
||||||
|
// public void testAttributeConversionReversabilityXML() {
|
||||||
|
// final String originalXml = "<table class=\"MyTable\" style=\"background-color:yellow\"><tr><td class=\"MyTD\" style=\"border:1px solid black\">#5D28D1</td><td class=\"MyTD\" style=\"background-color:red\">Example text here</td></tr><tr><td class=\"MyTD\" style=\"border:1px solid black\">#AF44EF</td><td class=\"MyTD\" style=\"background-color:green\">127310656</td></tr><tr><td class=\"MyTD\" style=\"border:1px solid black\">#AAD034</td><td class=\"MyTD\" style=\"background-color:blue\">&<span style=\"background-color:maroon\">></span><</td></tr></table>";
|
||||||
|
// final String expectedJsonString = "[\"table\",{\"class\" : \"MyTable\",\"style\" : \"background-color:yellow\"},[\"tr\",[\"td\",{\"class\" : \"MyTD\",\"style\" : \"border:1px solid black\"},\"#550758\"],[\"td\",{\"class\" : \"MyTD\",\"style\" : \"background-color:red\"},\"Example text here\"]],[\"tr\",[\"td\",{\"class\" : \"MyTD\",\"style\" : \"border:1px solid black\"},\"#993101\"],[\"td\",{\"class\" : \"MyTD\",\"style\" : \"background-color:green\"},\"127624015\"]],[\"tr\",[\"td\",{\"class\" : \"MyTD\",\"style\" : \"border:1px solid black\"},\"#E33D87\"],[\"td\",{\"class\" : \"MyTD\",\"style\" : \"background-color:blue\"},\"&\",[\"span\",{ \"style\" : \"background-color:maroon\" },\">\"],\"<\"]]]";
|
||||||
|
// final JSONArray jsonML = JSONML.toJSONArray(originalXml,true);
|
||||||
|
// final String actualJsonString = jsonML.toString();
|
||||||
|
//
|
||||||
|
// final String reverseXml = JSONML.toString(jsonML);
|
||||||
|
// // currently not equal because the hashing of the attribute objects makes the attribute
|
||||||
|
// // order not happen the same way twice
|
||||||
|
// assertEquals(originalXml, reverseXml);
|
||||||
|
//
|
||||||
|
// assertEquals(expectedJsonString, actualJsonString);
|
||||||
|
// }
|
||||||
|
|
||||||
|
@Test (timeout = 6000)
|
||||||
|
public void testIssue484InfinteLoop1() {
|
||||||
|
try {
|
||||||
|
JSONML.toJSONObject("??*^M??|?CglR^F??`??>?w??PIlr^E??D^X^]?$?-^R?o??O?*??{OD?^FY??`2a????NM?b^Tq?:O?>S$^K?J?^FB.gUK?m^H??zE??^??!v]?^A???^[^A??^U?c??????h???s???g^Z???`?q^Dbi??:^QZl?)?}1^??k?0??:$V?$?Ovs(}J??^V????2;^QgQ?^_^A?^D?^U?Tg?K?`?h%c?hmGA?<!C*^P^Y?^X9?~?t?)??,z^XA???S}?Q??.q?j????]");
|
||||||
|
fail("Exception expected for invalid JSON.");
|
||||||
|
} catch (JSONException ex) {
|
||||||
|
assertEquals("Exception string did not match: ",
|
||||||
|
"Unterminated string at 271 [character 272 line 1]",
|
||||||
|
ex.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test (timeout = 6000)
|
||||||
|
public void testIssue484InfinteLoop2() {
|
||||||
|
try {
|
||||||
|
String input = "??*\n" +
|
||||||
|
"??|?CglR??`??>?w??PIlr??D?$?-?o??O?*??{OD?Y??`2a????NM?bq?:O?>S$?J?B.gUK?m\b??zE???!v]???????c??????h???s???g???`?qbi??:Zl?)?}1^??k?0??:$V?$?Ovs(}J??????2;gQ????Tg?K?`?h%c?hmGA?<!C*?9?~?t?)??,zA???S}?Q??.q?j????]";
|
||||||
|
JSONML.toJSONObject(input);
|
||||||
|
fail("Exception expected for invalid JSON.");
|
||||||
|
} catch (JSONException ex) {
|
||||||
|
assertEquals("Exception string did not match: ",
|
||||||
|
"Unterminated string at 242 [character 238 line 2]",
|
||||||
|
ex.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
80
src/test/java/org/json/junit/JSONObjectLocaleTest.java
Executable file
80
src/test/java/org/json/junit/JSONObjectLocaleTest.java
Executable file
|
@ -0,0 +1,80 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
import org.json.*;
|
||||||
|
import org.json.junit.data.MyLocaleBean;
|
||||||
|
import org.junit.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Note: This file is saved as UTF-8. Do not save as ASCII or the tests will
|
||||||
|
* fail.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class JSONObjectLocaleTest {
|
||||||
|
/**
|
||||||
|
* JSONObject built from a bean with locale-specific keys.
|
||||||
|
* In the Turkish alphabet, there are 2 versions of the letter "i".
|
||||||
|
* 'eh' I ı (dotless i)
|
||||||
|
* 'ee' İ i (dotted i)
|
||||||
|
* A problem can occur when parsing the public get methods for a bean.
|
||||||
|
* If the method starts with getI... then the key name will be lowercased
|
||||||
|
* to 'i' in English, and 'ı' in Turkish.
|
||||||
|
* We want the keys to be consistent regardless of locale, so JSON-Java
|
||||||
|
* lowercase operations are made to be locale-neutral by specifying
|
||||||
|
* Locale.ROOT. This causes 'I' to be universally lowercased to 'i'
|
||||||
|
* regardless of the locale currently in effect.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void jsonObjectByLocaleBean() {
|
||||||
|
|
||||||
|
MyLocaleBean myLocaleBean = new MyLocaleBean();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is just the control case which happens when the locale.ROOT
|
||||||
|
* lowercasing behavior is the same as the current locale.
|
||||||
|
*/
|
||||||
|
Locale.setDefault(new Locale("en"));
|
||||||
|
JSONObject jsonen = new JSONObject(myLocaleBean);
|
||||||
|
assertEquals("expected size 2, found: " +jsonen.length(), 2, jsonen.length());
|
||||||
|
assertEquals("expected jsonen[i] == beanI", "beanI", jsonen.getString("i"));
|
||||||
|
assertEquals("expected jsonen[id] == beanId", "beanId", jsonen.getString("id"));
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Without the JSON-Java change, these keys would be stored internally as
|
||||||
|
* starting with the letter, 'ı' (dotless i), since the lowercasing of
|
||||||
|
* the getI and getId keys would be specific to the Turkish locale.
|
||||||
|
*/
|
||||||
|
Locale.setDefault(new Locale("tr"));
|
||||||
|
JSONObject jsontr = new JSONObject(myLocaleBean);
|
||||||
|
assertEquals("expected size 2, found: " +jsontr.length(), 2, jsontr.length());
|
||||||
|
assertEquals("expected jsontr[i] == beanI", "beanI", jsontr.getString("i"));
|
||||||
|
assertEquals("expected jsontr[id] == beanId", "beanId", jsontr.getString("id"));
|
||||||
|
}
|
||||||
|
}
|
3152
src/test/java/org/json/junit/JSONObjectTest.java
Normal file
3152
src/test/java/org/json/junit/JSONObjectTest.java
Normal file
File diff suppressed because it is too large
Load diff
384
src/test/java/org/json/junit/JSONPointerTest.java
Normal file
384
src/test/java/org/json/junit/JSONPointerTest.java
Normal file
|
@ -0,0 +1,384 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertSame;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import static org.junit.Assert.fail;
|
||||||
|
|
||||||
|
import java.io.InputStream;
|
||||||
|
|
||||||
|
import org.json.JSONArray;
|
||||||
|
import org.json.JSONObject;
|
||||||
|
import org.json.JSONPointer;
|
||||||
|
import org.json.JSONPointerException;
|
||||||
|
import org.json.JSONTokener;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
public class JSONPointerTest {
|
||||||
|
|
||||||
|
private static final JSONObject document;
|
||||||
|
|
||||||
|
static {
|
||||||
|
@SuppressWarnings("resource")
|
||||||
|
InputStream resourceAsStream = JSONPointerTest.class.getClassLoader().getResourceAsStream("jsonpointer-testdoc.json");
|
||||||
|
if(resourceAsStream == null) {
|
||||||
|
throw new ExceptionInInitializerError("Unable to locate test file. Please check your development environment configuration");
|
||||||
|
}
|
||||||
|
document = new JSONObject(new JSONTokener(resourceAsStream));
|
||||||
|
}
|
||||||
|
|
||||||
|
private Object query(String pointer) {
|
||||||
|
return new JSONPointer(pointer).queryFrom(document);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void emptyPointer() {
|
||||||
|
assertSame(document, query(""));
|
||||||
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
@Test(expected = NullPointerException.class)
|
||||||
|
public void nullPointer() {
|
||||||
|
new JSONPointer((String) null);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void objectPropertyQuery() {
|
||||||
|
assertSame(document.get("foo"), query("/foo"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void arrayIndexQuery() {
|
||||||
|
assertSame(document.getJSONArray("foo").get(0), query("/foo/0"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = JSONPointerException.class)
|
||||||
|
public void stringPropOfArrayFailure() {
|
||||||
|
query("/foo/bar");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void queryByEmptyKey() {
|
||||||
|
assertSame(document.get(""), query("/"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void queryByEmptyKeySubObject() {
|
||||||
|
assertSame(document.getJSONObject("obj").getJSONObject(""), query("/obj/"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void queryByEmptyKeySubObjectSubOject() {
|
||||||
|
assertSame(
|
||||||
|
document.getJSONObject("obj").getJSONObject("").get(""),
|
||||||
|
query("/obj//")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void queryByEmptyKeySubObjectValue() {
|
||||||
|
assertSame(
|
||||||
|
document.getJSONObject("obj").getJSONObject("").get("subKey"),
|
||||||
|
query("/obj//subKey")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void slashEscaping() {
|
||||||
|
assertSame(document.get("a/b"), query("/a~1b"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void tildeEscaping() {
|
||||||
|
assertSame(document.get("m~n"), query("/m~0n"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void backslashEscaping() {
|
||||||
|
assertSame(document.get("i\\j"), query("/i\\\\j"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void quotationEscaping() {
|
||||||
|
assertSame(document.get("k\"l"), query("/k\\\\\\\"l"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void whitespaceKey() {
|
||||||
|
assertSame(document.get(" "), query("/ "));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void uriFragmentNotation() {
|
||||||
|
assertSame(document.get("foo"), query("#/foo"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void uriFragmentNotationRoot() {
|
||||||
|
assertSame(document, query("#"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void uriFragmentPercentHandling() {
|
||||||
|
assertSame(document.get("c%d"), query("#/c%25d"));
|
||||||
|
assertSame(document.get("e^f"), query("#/e%5Ef"));
|
||||||
|
assertSame(document.get("g|h"), query("#/g%7Ch"));
|
||||||
|
assertSame(document.get("m~n"), query("#/m~0n"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
@Test(expected = IllegalArgumentException.class)
|
||||||
|
public void syntaxError() {
|
||||||
|
new JSONPointer("key");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = JSONPointerException.class)
|
||||||
|
public void arrayIndexFailure() {
|
||||||
|
query("/foo/2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = JSONPointerException.class)
|
||||||
|
public void primitiveFailure() {
|
||||||
|
query("/obj/key/failure");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void builderTest() {
|
||||||
|
JSONPointer pointer = JSONPointer.builder()
|
||||||
|
.append("obj")
|
||||||
|
.append("other~key").append("another/key")
|
||||||
|
.append(0)
|
||||||
|
.build();
|
||||||
|
assertEquals("val", pointer.queryFrom(document));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = NullPointerException.class)
|
||||||
|
public void nullToken() {
|
||||||
|
JSONPointer.builder().append(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void toStringEscaping() {
|
||||||
|
JSONPointer pointer = JSONPointer.builder()
|
||||||
|
.append("obj")
|
||||||
|
.append("other~key").append("another/key")
|
||||||
|
.append("\"")
|
||||||
|
.append(0)
|
||||||
|
.build();
|
||||||
|
assertEquals("/obj/other~0key/another~1key/\\\"/0", pointer.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void emptyPointerToString() {
|
||||||
|
assertEquals("", new JSONPointer("").toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void toURIFragment() {
|
||||||
|
assertEquals("#/c%25d", new JSONPointer("/c%d").toURIFragment());
|
||||||
|
assertEquals("#/e%5Ef", new JSONPointer("/e^f").toURIFragment());
|
||||||
|
assertEquals("#/g%7Ch", new JSONPointer("/g|h").toURIFragment());
|
||||||
|
assertEquals("#/m%7En", new JSONPointer("/m~n").toURIFragment());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void tokenListIsCopiedInConstructor() {
|
||||||
|
JSONPointer.Builder b = JSONPointer.builder().append("key1");
|
||||||
|
JSONPointer jp1 = b.build();
|
||||||
|
b.append("key2");
|
||||||
|
JSONPointer jp2 = b.build();
|
||||||
|
if(jp1.toString().equals(jp2.toString())) {
|
||||||
|
fail("Oops, my pointers are sharing a backing array");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Coverage for JSONObject query(String)
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void queryFromJSONObject() {
|
||||||
|
String str = "{"+
|
||||||
|
"\"stringKey\":\"hello world!\","+
|
||||||
|
"\"arrayKey\":[0,1,2],"+
|
||||||
|
"\"objectKey\": {"+
|
||||||
|
"\"a\":\"aVal\","+
|
||||||
|
"\"b\":\"bVal\""+
|
||||||
|
"}"+
|
||||||
|
"}";
|
||||||
|
JSONObject jsonObject = new JSONObject(str);
|
||||||
|
Object obj = jsonObject.query("/stringKey");
|
||||||
|
assertTrue("Expected 'hello world!'", "hello world!".equals(obj));
|
||||||
|
obj = jsonObject.query("/arrayKey/1");
|
||||||
|
assertTrue("Expected 1", Integer.valueOf(1).equals(obj));
|
||||||
|
obj = jsonObject.query("/objectKey/b");
|
||||||
|
assertTrue("Expected bVal", "bVal".equals(obj));
|
||||||
|
try {
|
||||||
|
obj = jsonObject.query("/a/b/c");
|
||||||
|
assertTrue("Expected JSONPointerException", false);
|
||||||
|
} catch (JSONPointerException e) {
|
||||||
|
assertTrue("Expected bad key/value exception",
|
||||||
|
"value [null] is not an array or object therefore its key b cannot be resolved".
|
||||||
|
equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Coverage for JSONObject query(JSONPointer)
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void queryFromJSONObjectUsingPointer() {
|
||||||
|
String str = "{"+
|
||||||
|
"\"stringKey\":\"hello world!\","+
|
||||||
|
"\"arrayKey\":[0,1,2],"+
|
||||||
|
"\"objectKey\": {"+
|
||||||
|
"\"a\":\"aVal\","+
|
||||||
|
"\"b\":\"bVal\""+
|
||||||
|
"}"+
|
||||||
|
"}";
|
||||||
|
JSONObject jsonObject = new JSONObject(str);
|
||||||
|
Object obj = jsonObject.query(new JSONPointer("/stringKey"));
|
||||||
|
assertTrue("Expected 'hello world!'", "hello world!".equals(obj));
|
||||||
|
obj = jsonObject.query(new JSONPointer("/arrayKey/1"));
|
||||||
|
assertTrue("Expected 1", Integer.valueOf(1).equals(obj));
|
||||||
|
obj = jsonObject.query(new JSONPointer("/objectKey/b"));
|
||||||
|
assertTrue("Expected bVal", "bVal".equals(obj));
|
||||||
|
try {
|
||||||
|
obj = jsonObject.query(new JSONPointer("/a/b/c"));
|
||||||
|
assertTrue("Expected JSONPointerException", false);
|
||||||
|
} catch (JSONPointerException e) {
|
||||||
|
assertTrue("Expected bad key/value exception",
|
||||||
|
"value [null] is not an array or object therefore its key b cannot be resolved".
|
||||||
|
equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Coverage for JSONObject optQuery(JSONPointer)
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void optQueryFromJSONObjectUsingPointer() {
|
||||||
|
String str = "{"+
|
||||||
|
"\"stringKey\":\"hello world!\","+
|
||||||
|
"\"arrayKey\":[0,1,2],"+
|
||||||
|
"\"objectKey\": {"+
|
||||||
|
"\"a\":\"aVal\","+
|
||||||
|
"\"b\":\"bVal\""+
|
||||||
|
"}"+
|
||||||
|
"}";
|
||||||
|
JSONObject jsonObject = new JSONObject(str);
|
||||||
|
Object obj = jsonObject.optQuery(new JSONPointer("/stringKey"));
|
||||||
|
assertTrue("Expected 'hello world!'", "hello world!".equals(obj));
|
||||||
|
obj = jsonObject.optQuery(new JSONPointer("/arrayKey/1"));
|
||||||
|
assertTrue("Expected 1", Integer.valueOf(1).equals(obj));
|
||||||
|
obj = jsonObject.optQuery(new JSONPointer("/objectKey/b"));
|
||||||
|
assertTrue("Expected bVal", "bVal".equals(obj));
|
||||||
|
obj = jsonObject.optQuery(new JSONPointer("/a/b/c"));
|
||||||
|
assertTrue("Expected null", obj == null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Coverage for JSONArray query(String)
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void queryFromJSONArray() {
|
||||||
|
String str = "["+
|
||||||
|
"\"hello world!\","+
|
||||||
|
"[0,1,2],"+
|
||||||
|
"{"+
|
||||||
|
"\"a\":\"aVal\","+
|
||||||
|
"\"b\":\"bVal\""+
|
||||||
|
"}"+
|
||||||
|
"]";
|
||||||
|
JSONArray jsonArray = new JSONArray(str);
|
||||||
|
Object obj = jsonArray.query("/0");
|
||||||
|
assertTrue("Expected 'hello world!'", "hello world!".equals(obj));
|
||||||
|
obj = jsonArray.query("/1/1");
|
||||||
|
assertTrue("Expected 1", Integer.valueOf(1).equals(obj));
|
||||||
|
obj = jsonArray.query("/2/b");
|
||||||
|
assertTrue("Expected bVal", "bVal".equals(obj));
|
||||||
|
try {
|
||||||
|
obj = jsonArray.query("/a/b/c");
|
||||||
|
assertTrue("Expected JSONPointerException", false);
|
||||||
|
} catch (JSONPointerException e) {
|
||||||
|
assertTrue("Expected bad index exception",
|
||||||
|
"a is not an array index".equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Coverage for JSONArray query(JSONPointer)
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void queryFromJSONArrayUsingPointer() {
|
||||||
|
String str = "["+
|
||||||
|
"\"hello world!\","+
|
||||||
|
"[0,1,2],"+
|
||||||
|
"{"+
|
||||||
|
"\"a\":\"aVal\","+
|
||||||
|
"\"b\":\"bVal\""+
|
||||||
|
"}"+
|
||||||
|
"]";
|
||||||
|
JSONArray jsonArray = new JSONArray(str);
|
||||||
|
Object obj = jsonArray.query(new JSONPointer("/0"));
|
||||||
|
assertTrue("Expected 'hello world!'", "hello world!".equals(obj));
|
||||||
|
obj = jsonArray.query(new JSONPointer("/1/1"));
|
||||||
|
assertTrue("Expected 1", Integer.valueOf(1).equals(obj));
|
||||||
|
obj = jsonArray.query(new JSONPointer("/2/b"));
|
||||||
|
assertTrue("Expected bVal", "bVal".equals(obj));
|
||||||
|
try {
|
||||||
|
obj = jsonArray.query(new JSONPointer("/a/b/c"));
|
||||||
|
assertTrue("Expected JSONPointerException", false);
|
||||||
|
} catch (JSONPointerException e) {
|
||||||
|
assertTrue("Expected bad index exception",
|
||||||
|
"a is not an array index".equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Coverage for JSONArray optQuery(JSONPointer)
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void optQueryFromJSONArrayUsingPointer() {
|
||||||
|
String str = "["+
|
||||||
|
"\"hello world!\","+
|
||||||
|
"[0,1,2],"+
|
||||||
|
"{"+
|
||||||
|
"\"a\":\"aVal\","+
|
||||||
|
"\"b\":\"bVal\""+
|
||||||
|
"}"+
|
||||||
|
"]";
|
||||||
|
JSONArray jsonArray = new JSONArray(str);
|
||||||
|
Object obj = jsonArray.optQuery(new JSONPointer("/0"));
|
||||||
|
assertTrue("Expected 'hello world!'", "hello world!".equals(obj));
|
||||||
|
obj = jsonArray.optQuery(new JSONPointer("/1/1"));
|
||||||
|
assertTrue("Expected 1", Integer.valueOf(1).equals(obj));
|
||||||
|
obj = jsonArray.optQuery(new JSONPointer("/2/b"));
|
||||||
|
assertTrue("Expected bVal", "bVal".equals(obj));
|
||||||
|
obj = jsonArray.optQuery(new JSONPointer("/a/b/c"));
|
||||||
|
assertTrue("Expected null", obj == null);
|
||||||
|
}
|
||||||
|
}
|
360
src/test/java/org/json/junit/JSONStringTest.java
Normal file
360
src/test/java/org/json/junit/JSONStringTest.java
Normal file
|
@ -0,0 +1,360 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
|
import java.io.StringWriter;
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
import org.json.*;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for JSONString implementations, and the difference between
|
||||||
|
* {@link JSONObject#valueToString} and {@link JSONObject#writeValue}.
|
||||||
|
*/
|
||||||
|
public class JSONStringTest {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This tests the JSONObject.writeValue() method. We can't test directly
|
||||||
|
* due to it being a package-protected method. Instead, we can call
|
||||||
|
* JSONArray.write(), which delegates the writing of each entry to
|
||||||
|
* writeValue().
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void writeValues() throws Exception {
|
||||||
|
JSONArray jsonArray = new JSONArray();
|
||||||
|
jsonArray.put((Object)null);
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
String output = jsonArray.write(writer).toString();
|
||||||
|
assertTrue("String values should be equal", "[null]".equals(output));
|
||||||
|
|
||||||
|
jsonArray = new JSONArray();
|
||||||
|
jsonArray.put(JSONObject.NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
String output = jsonArray.write(writer).toString();
|
||||||
|
assertTrue("String values should be equal", "[null]".equals(output));
|
||||||
|
|
||||||
|
jsonArray = new JSONArray();
|
||||||
|
jsonArray.put(new JSONObject());
|
||||||
|
}
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
String output = jsonArray.write(writer).toString();
|
||||||
|
assertTrue("String values should be equal", "[{}]".equals(output));
|
||||||
|
|
||||||
|
jsonArray = new JSONArray();
|
||||||
|
jsonArray.put(new JSONArray());
|
||||||
|
}
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
String output = jsonArray.write(writer).toString();
|
||||||
|
assertTrue("String values should be equal", "[[]]".equals(output));
|
||||||
|
|
||||||
|
jsonArray = new JSONArray();
|
||||||
|
Map<?,?> singleMap = Collections.singletonMap("key1", "value1");
|
||||||
|
jsonArray.put((Object)singleMap);
|
||||||
|
}
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
String output = jsonArray.write(writer).toString();
|
||||||
|
assertTrue("String values should be equal", "[{\"key1\":\"value1\"}]".equals(output));
|
||||||
|
|
||||||
|
jsonArray = new JSONArray();
|
||||||
|
List<?> singleList = Collections.singletonList("entry1");
|
||||||
|
jsonArray.put((Object)singleList);
|
||||||
|
}
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
String output = jsonArray.write(writer).toString();
|
||||||
|
assertTrue("String values should be equal", "[[\"entry1\"]]".equals(output));
|
||||||
|
|
||||||
|
jsonArray = new JSONArray();
|
||||||
|
int[] intArray = new int[] { 1, 2, 3 };
|
||||||
|
jsonArray.put(intArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
String output = jsonArray.write(writer).toString();
|
||||||
|
assertTrue("String values should be equal", "[[1,2,3]]".equals(output));
|
||||||
|
|
||||||
|
jsonArray = new JSONArray();
|
||||||
|
jsonArray.put(24);
|
||||||
|
}
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
String output = jsonArray.write(writer).toString();
|
||||||
|
assertTrue("String values should be equal", "[24]".equals(output));
|
||||||
|
|
||||||
|
jsonArray = new JSONArray();
|
||||||
|
jsonArray.put("string value");
|
||||||
|
}
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
String output = jsonArray.write(writer).toString();
|
||||||
|
assertTrue("String values should be equal", "[\"string value\"]".equals(output));
|
||||||
|
|
||||||
|
jsonArray = new JSONArray();
|
||||||
|
jsonArray.put(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
String output = jsonArray.write(writer).toString();
|
||||||
|
assertTrue("String values should be equal", "[true]".equals(output));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This tests the JSONObject.valueToString() method. These should be
|
||||||
|
* identical to the values above, except for the enclosing [ and ].
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("boxing")
|
||||||
|
@Test
|
||||||
|
public void valuesToString() throws Exception {
|
||||||
|
|
||||||
|
String output = JSONObject.valueToString(null);
|
||||||
|
assertTrue("String values should be equal", "null".equals(output));
|
||||||
|
|
||||||
|
output = JSONObject.valueToString(JSONObject.NULL);
|
||||||
|
assertTrue("String values should be equal", "null".equals(output));
|
||||||
|
|
||||||
|
output = JSONObject.valueToString(new JSONObject());
|
||||||
|
assertTrue("String values should be equal", "{}".equals(output));
|
||||||
|
|
||||||
|
output = JSONObject.valueToString(new JSONArray());
|
||||||
|
assertTrue("String values should be equal", "[]".equals(output));
|
||||||
|
|
||||||
|
Map<?,?> singleMap = Collections.singletonMap("key1", "value1");
|
||||||
|
output = JSONObject.valueToString(singleMap);
|
||||||
|
assertTrue("String values should be equal", "{\"key1\":\"value1\"}".equals(output));
|
||||||
|
|
||||||
|
List<?> singleList = Collections.singletonList("entry1");
|
||||||
|
output = JSONObject.valueToString(singleList);
|
||||||
|
assertTrue("String values should be equal", "[\"entry1\"]".equals(output));
|
||||||
|
|
||||||
|
int[] intArray = new int[] { 1, 2, 3 };
|
||||||
|
output = JSONObject.valueToString(intArray);
|
||||||
|
assertTrue("String values should be equal", "[1,2,3]".equals(output));
|
||||||
|
|
||||||
|
output = JSONObject.valueToString(24);
|
||||||
|
assertTrue("String values should be equal", "24".equals(output));
|
||||||
|
|
||||||
|
output = JSONObject.valueToString("string value");
|
||||||
|
assertTrue("String values should be equal", "\"string value\"".equals(output));
|
||||||
|
|
||||||
|
output = JSONObject.valueToString(true);
|
||||||
|
assertTrue("String values should be equal", "true".equals(output));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test what happens when toJSONString() returns a well-formed JSON value.
|
||||||
|
* This is the usual case.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testJSONStringValue() throws Exception {
|
||||||
|
JSONStringValue jsonString = new JSONStringValue();
|
||||||
|
JSONArray jsonArray = new JSONArray();
|
||||||
|
|
||||||
|
jsonArray.put(jsonString);
|
||||||
|
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
String output = jsonArray.write(writer).toString();
|
||||||
|
assertTrue("String values should be equal", "[\"the JSON string value\"]".equals(output));
|
||||||
|
|
||||||
|
output = JSONObject.valueToString(jsonString);
|
||||||
|
assertTrue("String values should be equal", "\"the JSON string value\"".equals(output));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test what happens when toJSONString() returns null. In one case,
|
||||||
|
* use the object's toString() method. In the other, throw a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testJSONNullStringValue() throws Exception {
|
||||||
|
JSONNullStringValue jsonString = new JSONNullStringValue();
|
||||||
|
JSONArray jsonArray = new JSONArray();
|
||||||
|
|
||||||
|
jsonArray.put(jsonString);
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
String output = jsonArray.write(writer).toString();
|
||||||
|
assertTrue("String values should be equal", "[\"the toString value\"]".equals(output));
|
||||||
|
|
||||||
|
// The only different between writeValue() and valueToString():
|
||||||
|
// in this case, valueToString throws a JSONException
|
||||||
|
try {
|
||||||
|
output = JSONObject.valueToString(jsonString);
|
||||||
|
fail("Expected an exception, got a String value");
|
||||||
|
} catch (Exception e) {
|
||||||
|
assertTrue("Expected JSONException", e instanceof JSONException);
|
||||||
|
assertTrue("Exception message does not match", "Bad value from toJSONString: null".equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test what happens when toJSONString() returns an exception. In both
|
||||||
|
* cases, a JSONException is thrown, with the cause and message set from
|
||||||
|
* the original exception.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testJSONStringExceptionValue() {
|
||||||
|
JSONStringExceptionValue jsonString = new JSONStringExceptionValue();
|
||||||
|
JSONArray jsonArray = new JSONArray();
|
||||||
|
|
||||||
|
jsonArray.put(jsonString);
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
jsonArray.write(writer).toString();
|
||||||
|
fail("Expected an exception, got a String value");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Unable to write JSONArray value at index: 0", e.getMessage());
|
||||||
|
} catch(Exception e) {
|
||||||
|
fail("Expected JSONException");
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
JSONObject.valueToString(jsonString);
|
||||||
|
fail("Expected an exception, got a String value");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Exception message does not match", "the exception value".equals(e.getMessage()));
|
||||||
|
} catch(Exception e) {
|
||||||
|
fail("Expected JSONException");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test what happens when a Java object's toString() returns a String value.
|
||||||
|
* This is the usual case.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testStringValue() throws Exception {
|
||||||
|
StringValue nonJsonString = new StringValue();
|
||||||
|
JSONArray jsonArray = new JSONArray();
|
||||||
|
|
||||||
|
jsonArray.put(nonJsonString);
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
String output = jsonArray.write(writer).toString();
|
||||||
|
assertTrue("String values should be equal", "[\"the toString value for StringValue\"]".equals(output));
|
||||||
|
|
||||||
|
output = JSONObject.valueToString(nonJsonString);
|
||||||
|
assertTrue("String values should be equal", "\"the toString value for StringValue\"".equals(output));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test what happens when a Java object's toString() returns null.
|
||||||
|
* Defaults to empty string.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testNullStringValue() throws Exception {
|
||||||
|
NullStringValue nonJsonString = new NullStringValue();
|
||||||
|
JSONArray jsonArray = new JSONArray();
|
||||||
|
|
||||||
|
jsonArray.put(nonJsonString);
|
||||||
|
|
||||||
|
|
||||||
|
try (StringWriter writer = new StringWriter();) {
|
||||||
|
String output = jsonArray.write(writer).toString();
|
||||||
|
assertTrue("String values should be equal", "[\"\"]".equals(output));
|
||||||
|
|
||||||
|
output = JSONObject.valueToString(nonJsonString);
|
||||||
|
assertTrue("String values should be equal", "\"\"".equals(output));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A JSONString that returns a valid JSON string value.
|
||||||
|
*/
|
||||||
|
private static final class JSONStringValue implements JSONString {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toJSONString() {
|
||||||
|
return "\"the JSON string value\"";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "the toString value for JSONStringValue";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A JSONString that returns null when calling toJSONString().
|
||||||
|
*/
|
||||||
|
private static final class JSONNullStringValue implements JSONString {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toJSONString() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "the toString value";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A JSONString that throw an exception when calling toJSONString().
|
||||||
|
*/
|
||||||
|
private static final class JSONStringExceptionValue implements JSONString {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toJSONString() {
|
||||||
|
throw new IllegalStateException("the exception value");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "the toString value for JSONStringExceptionValue";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final class StringValue {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "the toString value for StringValue";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final class NullStringValue {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
377
src/test/java/org/json/junit/JSONStringerTest.java
Normal file
377
src/test/java/org/json/junit/JSONStringerTest.java
Normal file
|
@ -0,0 +1,377 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
|
import java.math.BigDecimal;
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
import org.json.*;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import com.jayway.jsonpath.*;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for JSON-Java JSONStringer and JSONWriter.
|
||||||
|
*/
|
||||||
|
public class JSONStringerTest {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Object with a null key.
|
||||||
|
* Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void nullKeyException() {
|
||||||
|
JSONStringer jsonStringer = new JSONStringer();
|
||||||
|
jsonStringer.object();
|
||||||
|
try {
|
||||||
|
jsonStringer.key(null);
|
||||||
|
assertTrue("Expected an exception", false);
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Expected an exception message",
|
||||||
|
"Null key.".
|
||||||
|
equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add a key with no object.
|
||||||
|
* Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void outOfSequenceException() {
|
||||||
|
JSONStringer jsonStringer = new JSONStringer();
|
||||||
|
try {
|
||||||
|
jsonStringer.key("hi");
|
||||||
|
assertTrue("Expected an exception", false);
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Expected an exception message",
|
||||||
|
"Misplaced key.".
|
||||||
|
equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Missplace an array.
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void missplacedArrayException() {
|
||||||
|
JSONStringer jsonStringer = new JSONStringer();
|
||||||
|
jsonStringer.object().endObject();
|
||||||
|
try {
|
||||||
|
jsonStringer.array();
|
||||||
|
assertTrue("Expected an exception", false);
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Expected an exception message",
|
||||||
|
"Misplaced array.".
|
||||||
|
equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Missplace an endErray.
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void missplacedEndArrayException() {
|
||||||
|
JSONStringer jsonStringer = new JSONStringer();
|
||||||
|
jsonStringer.object();
|
||||||
|
try {
|
||||||
|
jsonStringer.endArray();
|
||||||
|
assertTrue("Expected an exception", false);
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Expected an exception message",
|
||||||
|
"Misplaced endArray.".
|
||||||
|
equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Missplace an endObject.
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void missplacedEndObjectException() {
|
||||||
|
JSONStringer jsonStringer = new JSONStringer();
|
||||||
|
jsonStringer.array();
|
||||||
|
try {
|
||||||
|
jsonStringer.endObject();
|
||||||
|
assertTrue("Expected an exception", false);
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Expected an exception message",
|
||||||
|
"Misplaced endObject.".
|
||||||
|
equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Missplace an object.
|
||||||
|
* Expects a JSONException.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void missplacedObjectException() {
|
||||||
|
JSONStringer jsonStringer = new JSONStringer();
|
||||||
|
jsonStringer.object().endObject();
|
||||||
|
try {
|
||||||
|
jsonStringer.object();
|
||||||
|
assertTrue("Expected an exception", false);
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Expected an exception message",
|
||||||
|
"Misplaced object.".
|
||||||
|
equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Exceeds implementation max nesting depth.
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void exceedNestDepthException() {
|
||||||
|
try {
|
||||||
|
JSONStringer s = new JSONStringer();
|
||||||
|
s.object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object();
|
||||||
|
s.key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object().
|
||||||
|
key("k").object().key("k").object().key("k").object().key("k").object().key("k").object();
|
||||||
|
fail("Expected an exception message");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertTrue("Expected an exception message",
|
||||||
|
"Nesting too deep.".
|
||||||
|
equals(e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build a JSON doc using JSONString API calls,
|
||||||
|
* then convert to JSONObject
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void simpleObjectString() {
|
||||||
|
JSONStringer jsonStringer = new JSONStringer();
|
||||||
|
jsonStringer.object();
|
||||||
|
jsonStringer.key("trueValue").value(true);
|
||||||
|
jsonStringer.key("falseValue").value(false);
|
||||||
|
jsonStringer.key("nullValue").value(null);
|
||||||
|
jsonStringer.key("stringValue").value("hello world!");
|
||||||
|
jsonStringer.key("complexStringValue").value("h\be\tllo w\u1234orld!");
|
||||||
|
jsonStringer.key("intValue").value(42);
|
||||||
|
jsonStringer.key("doubleValue").value(-23.45e67);
|
||||||
|
jsonStringer.endObject();
|
||||||
|
String str = jsonStringer.toString();
|
||||||
|
JSONObject jsonObject = new JSONObject(str);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
Object doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("expected 7 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 7);
|
||||||
|
assertTrue("expected true", Boolean.TRUE.equals(jsonObject.query("/trueValue")));
|
||||||
|
assertTrue("expected false", Boolean.FALSE.equals(jsonObject.query("/falseValue")));
|
||||||
|
assertTrue("expected null", JSONObject.NULL.equals(jsonObject.query("/nullValue")));
|
||||||
|
assertTrue("expected hello world!", "hello world!".equals(jsonObject.query("/stringValue")));
|
||||||
|
assertTrue("expected h\be\tllo w\u1234orld!", "h\be\tllo w\u1234orld!".equals(jsonObject.query("/complexStringValue")));
|
||||||
|
assertTrue("expected 42", Integer.valueOf(42).equals(jsonObject.query("/intValue")));
|
||||||
|
assertTrue("expected -23.45e67", BigDecimal.valueOf(-23.45e67).equals(jsonObject.query("/doubleValue")));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build a JSON doc using JSONString API calls,
|
||||||
|
* then convert to JSONArray
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void simpleArrayString() {
|
||||||
|
JSONStringer jsonStringer = new JSONStringer();
|
||||||
|
jsonStringer.array();
|
||||||
|
jsonStringer.value(true);
|
||||||
|
jsonStringer.value(false);
|
||||||
|
jsonStringer.value(null);
|
||||||
|
jsonStringer.value("hello world!");
|
||||||
|
jsonStringer.value(42);
|
||||||
|
jsonStringer.value(-23.45e67);
|
||||||
|
jsonStringer.endArray();
|
||||||
|
String str = jsonStringer.toString();
|
||||||
|
JSONArray jsonArray = new JSONArray(str);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
Object doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonArray.toString());
|
||||||
|
assertTrue("expected 6 top level items", ((List<?>)(JsonPath.read(doc, "$"))).size() == 6);
|
||||||
|
assertTrue("expected true", Boolean.TRUE.equals(jsonArray.query("/0")));
|
||||||
|
assertTrue("expected false", Boolean.FALSE.equals(jsonArray.query("/1")));
|
||||||
|
assertTrue("expected null", JSONObject.NULL.equals(jsonArray.query("/2")));
|
||||||
|
assertTrue("expected hello world!", "hello world!".equals(jsonArray.query("/3")));
|
||||||
|
assertTrue("expected 42", Integer.valueOf(42).equals(jsonArray.query("/4")));
|
||||||
|
assertTrue("expected -23.45e67", BigDecimal.valueOf(-23.45e67).equals(jsonArray.query("/5")));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build a nested JSON doc using JSONString API calls, then convert to
|
||||||
|
* JSONObject. Will create a long cascade of output by reusing the
|
||||||
|
* returned values..
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void complexObjectString() {
|
||||||
|
JSONStringer jsonStringer = new JSONStringer();
|
||||||
|
jsonStringer.object().
|
||||||
|
key("trueValue").value(true).
|
||||||
|
key("falseValue").value(false).
|
||||||
|
key("nullValue").value(null).
|
||||||
|
key("stringValue").value("hello world!").
|
||||||
|
key("object2").object().
|
||||||
|
key("k1").value("v1").
|
||||||
|
key("k2").value("v2").
|
||||||
|
key("k3").value("v3").
|
||||||
|
key("array1").array().
|
||||||
|
value(1).
|
||||||
|
value(2).
|
||||||
|
object().
|
||||||
|
key("k4").value("v4").
|
||||||
|
key("k5").value("v5").
|
||||||
|
key("k6").value("v6").
|
||||||
|
key("array2").array().
|
||||||
|
value(5).
|
||||||
|
value(6).
|
||||||
|
value(7).
|
||||||
|
value(8).
|
||||||
|
endArray().
|
||||||
|
endObject().
|
||||||
|
value(3).
|
||||||
|
value(4).
|
||||||
|
endArray().
|
||||||
|
endObject().
|
||||||
|
key("complexStringValue").value("h\be\tllo w\u1234orld!").
|
||||||
|
key("intValue").value(42).
|
||||||
|
key("doubleValue").value(-23.45e67).
|
||||||
|
endObject();
|
||||||
|
String str = jsonStringer.toString();
|
||||||
|
JSONObject jsonObject = new JSONObject(str);
|
||||||
|
|
||||||
|
// validate JSON content
|
||||||
|
Object doc = Configuration.defaultConfiguration().jsonProvider().parse(jsonObject.toString());
|
||||||
|
assertTrue("expected 8 top level items", ((Map<?,?>)(JsonPath.read(doc, "$"))).size() == 8);
|
||||||
|
assertTrue("expected 4 object2 items", ((Map<?,?>)(JsonPath.read(doc, "$.object2"))).size() == 4);
|
||||||
|
assertTrue("expected 5 array1 items", ((List<?>)(JsonPath.read(doc, "$.object2.array1"))).size() == 5);
|
||||||
|
assertTrue("expected 4 array[2] items", ((Map<?,?>)(JsonPath.read(doc, "$.object2.array1[2]"))).size() == 4);
|
||||||
|
assertTrue("expected 4 array1[2].array2 items", ((List<?>)(JsonPath.read(doc, "$.object2.array1[2].array2"))).size() == 4);
|
||||||
|
assertTrue("expected true", Boolean.TRUE.equals(jsonObject.query("/trueValue")));
|
||||||
|
assertTrue("expected false", Boolean.FALSE.equals(jsonObject.query("/falseValue")));
|
||||||
|
assertTrue("expected null", JSONObject.NULL.equals(jsonObject.query("/nullValue")));
|
||||||
|
assertTrue("expected hello world!", "hello world!".equals(jsonObject.query("/stringValue")));
|
||||||
|
assertTrue("expected 42", Integer.valueOf(42).equals(jsonObject.query("/intValue")));
|
||||||
|
assertTrue("expected -23.45e67", BigDecimal.valueOf(-23.45e67).equals(jsonObject.query("/doubleValue")));
|
||||||
|
assertTrue("expected h\be\tllo w\u1234orld!", "h\be\tllo w\u1234orld!".equals(jsonObject.query("/complexStringValue")));
|
||||||
|
assertTrue("expected v1", "v1".equals(jsonObject.query("/object2/k1")));
|
||||||
|
assertTrue("expected v2", "v2".equals(jsonObject.query("/object2/k2")));
|
||||||
|
assertTrue("expected v3", "v3".equals(jsonObject.query("/object2/k3")));
|
||||||
|
assertTrue("expected 1", Integer.valueOf(1).equals(jsonObject.query("/object2/array1/0")));
|
||||||
|
assertTrue("expected 2", Integer.valueOf(2).equals(jsonObject.query("/object2/array1/1")));
|
||||||
|
assertTrue("expected v4", "v4".equals(jsonObject.query("/object2/array1/2/k4")));
|
||||||
|
assertTrue("expected v5", "v5".equals(jsonObject.query("/object2/array1/2/k5")));
|
||||||
|
assertTrue("expected v6", "v6".equals(jsonObject.query("/object2/array1/2/k6")));
|
||||||
|
assertTrue("expected 5", Integer.valueOf(5).equals(jsonObject.query("/object2/array1/2/array2/0")));
|
||||||
|
assertTrue("expected 6", Integer.valueOf(6).equals(jsonObject.query("/object2/array1/2/array2/1")));
|
||||||
|
assertTrue("expected 7", Integer.valueOf(7).equals(jsonObject.query("/object2/array1/2/array2/2")));
|
||||||
|
assertTrue("expected 8", Integer.valueOf(8).equals(jsonObject.query("/object2/array1/2/array2/3")));
|
||||||
|
assertTrue("expected 3", Integer.valueOf(3).equals(jsonObject.query("/object2/array1/3")));
|
||||||
|
assertTrue("expected 4", Integer.valueOf(4).equals(jsonObject.query("/object2/array1/4")));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
319
src/test/java/org/json/junit/JSONTokenerTest.java
Normal file
319
src/test/java/org/json/junit/JSONTokenerTest.java
Normal file
|
@ -0,0 +1,319 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import static org.junit.Assert.fail;
|
||||||
|
|
||||||
|
import java.io.BufferedReader;
|
||||||
|
import java.io.ByteArrayInputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStreamReader;
|
||||||
|
import java.io.Reader;
|
||||||
|
import java.io.StringReader;
|
||||||
|
|
||||||
|
import org.json.JSONArray;
|
||||||
|
import org.json.JSONException;
|
||||||
|
import org.json.JSONObject;
|
||||||
|
import org.json.JSONTokener;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test specific to the {@link org.json.JSONTokener} class.
|
||||||
|
* @author John Aylward
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class JSONTokenerTest {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* verify that back() fails as expected.
|
||||||
|
* @throws IOException thrown if something unexpected happens.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void verifyBackFailureZeroIndex() throws IOException {
|
||||||
|
try(Reader reader = new StringReader("some test string")) {
|
||||||
|
final JSONTokener tokener = new JSONTokener(reader);
|
||||||
|
try {
|
||||||
|
// this should fail since the index is 0;
|
||||||
|
tokener.back();
|
||||||
|
fail("Expected an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Stepping back two steps is not supported", e.getMessage());
|
||||||
|
} catch (Exception e) {
|
||||||
|
fail("Unknown Exception type " + e.getClass().getCanonicalName()+" with message "+e.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* verify that back() fails as expected.
|
||||||
|
* @throws IOException thrown if something unexpected happens.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void verifyBackFailureDoubleBack() throws IOException {
|
||||||
|
try(Reader reader = new StringReader("some test string")) {
|
||||||
|
final JSONTokener tokener = new JSONTokener(reader);
|
||||||
|
tokener.next();
|
||||||
|
tokener.back();
|
||||||
|
try {
|
||||||
|
// this should fail since the index is 0;
|
||||||
|
tokener.back();
|
||||||
|
fail("Expected an exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Stepping back two steps is not supported", e.getMessage());
|
||||||
|
} catch (Exception e) {
|
||||||
|
fail("Unknown Exception type " + e.getClass().getCanonicalName()+" with message "+e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testValid() {
|
||||||
|
checkValid("0",Number.class);
|
||||||
|
checkValid(" 0 ",Number.class);
|
||||||
|
checkValid("23",Number.class);
|
||||||
|
checkValid("23.5",Number.class);
|
||||||
|
checkValid(" 23.5 ",Number.class);
|
||||||
|
checkValid("null",null);
|
||||||
|
checkValid(" null ",null);
|
||||||
|
checkValid("true",Boolean.class);
|
||||||
|
checkValid(" true\n",Boolean.class);
|
||||||
|
checkValid("false",Boolean.class);
|
||||||
|
checkValid("\nfalse ",Boolean.class);
|
||||||
|
checkValid("{}",JSONObject.class);
|
||||||
|
checkValid(" {} ",JSONObject.class);
|
||||||
|
checkValid("{\"a\":1}",JSONObject.class);
|
||||||
|
checkValid(" {\"a\":1} ",JSONObject.class);
|
||||||
|
checkValid("[]",JSONArray.class);
|
||||||
|
checkValid(" [] ",JSONArray.class);
|
||||||
|
checkValid("[1,2]",JSONArray.class);
|
||||||
|
checkValid("\n\n[1,2]\n\n",JSONArray.class);
|
||||||
|
checkValid("1 2", String.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testErrors() {
|
||||||
|
// Check that stream can detect that a value is found after
|
||||||
|
// the first one
|
||||||
|
checkError(" { \"a\":1 } 4 ");
|
||||||
|
checkError("null \"a\"");
|
||||||
|
checkError("{} true");
|
||||||
|
}
|
||||||
|
|
||||||
|
private Object checkValid(String testStr, Class<?> aClass) {
|
||||||
|
Object result = nextValue(testStr);
|
||||||
|
|
||||||
|
// Check class of object returned
|
||||||
|
if( null == aClass ) {
|
||||||
|
if(JSONObject.NULL.equals(result)) {
|
||||||
|
// OK
|
||||||
|
} else {
|
||||||
|
throw new JSONException("Unexpected class: "+result.getClass().getSimpleName());
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if( null == result ) {
|
||||||
|
throw new JSONException("Unexpected null result");
|
||||||
|
} else if(!aClass.isAssignableFrom(result.getClass()) ) {
|
||||||
|
throw new JSONException("Unexpected class: "+result.getClass().getSimpleName());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void checkError(String testStr) {
|
||||||
|
try {
|
||||||
|
nextValue(testStr);
|
||||||
|
|
||||||
|
fail("Error should be triggered: (\""+testStr+"\")");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
// OK
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Verifies that JSONTokener can read a stream that contains a value. After
|
||||||
|
* the reading is done, check that the stream is left in the correct state
|
||||||
|
* by reading the characters after. All valid cases should reach end of stream.
|
||||||
|
* @param testStr
|
||||||
|
* @return
|
||||||
|
* @throws Exception
|
||||||
|
*/
|
||||||
|
private Object nextValue(String testStr) throws JSONException {
|
||||||
|
try(StringReader sr = new StringReader(testStr);){
|
||||||
|
JSONTokener tokener = new JSONTokener(sr);
|
||||||
|
|
||||||
|
Object result = tokener.nextValue();
|
||||||
|
|
||||||
|
if( result == null ) {
|
||||||
|
throw new JSONException("Unable to find value token in JSON stream: ("+tokener+"): "+testStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
char c = tokener.nextClean();
|
||||||
|
if( 0 != c ) {
|
||||||
|
throw new JSONException("Unexpected character found at end of JSON stream: "+c+ " ("+tokener+"): "+testStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests the failure of the skipTo method with a buffered reader. Preferably
|
||||||
|
* we'd like this not to fail but at this time we don't have a good recovery.
|
||||||
|
*
|
||||||
|
* @throws IOException thrown if something unexpected happens.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testSkipToFailureWithBufferedReader() throws IOException {
|
||||||
|
final byte[] superLongBuffer = new byte[1000001];
|
||||||
|
// fill our buffer
|
||||||
|
for(int i=0;i<superLongBuffer.length;i++) {
|
||||||
|
superLongBuffer[i] = 'A';
|
||||||
|
}
|
||||||
|
try(Reader reader = new BufferedReader(new InputStreamReader(new ByteArrayInputStream(superLongBuffer)))) {
|
||||||
|
final JSONTokener tokener = new JSONTokener(reader);
|
||||||
|
try {
|
||||||
|
// this should fail since the internal markAhead buffer is only 1,000,000
|
||||||
|
// but 'B' doesn't exist in our buffer that is 1,000,001 in size
|
||||||
|
tokener.skipTo('B');
|
||||||
|
fail("Expected exception");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Mark invalid", e.getMessage());
|
||||||
|
} catch (Exception e) {
|
||||||
|
fail("Unknown Exception type " + e.getClass().getCanonicalName()+" with message "+e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests the success of the skipTo method with a String reader.
|
||||||
|
*
|
||||||
|
* @throws IOException thrown if something unexpected happens.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testSkipToSuccessWithStringReader() throws IOException {
|
||||||
|
final StringBuilder superLongBuffer = new StringBuilder(1000001);
|
||||||
|
// fill our buffer
|
||||||
|
for(int i=0;i<superLongBuffer.length();i++) {
|
||||||
|
superLongBuffer.append('A');
|
||||||
|
}
|
||||||
|
try(Reader reader = new StringReader(superLongBuffer.toString())) {
|
||||||
|
final JSONTokener tokener = new JSONTokener(reader);
|
||||||
|
try {
|
||||||
|
// this should not fail since the internal markAhead is ignored for StringReaders
|
||||||
|
tokener.skipTo('B');
|
||||||
|
} catch (Exception e) {
|
||||||
|
fail("Unknown Exception type " + e.getClass().getCanonicalName()+" with message "+e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Verify that next and back are working properly and tracking the correct positions
|
||||||
|
* with different new line combinations.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testNextBackComboWithNewLines() {
|
||||||
|
final String testString = "this is\nA test\r\nWith some different\rNew Lines";
|
||||||
|
// ^ ^ ^ ^
|
||||||
|
// index positions 0 8 16 36
|
||||||
|
final JSONTokener tokener = new JSONTokener(testString);
|
||||||
|
assertEquals(" at 0 [character 1 line 1]", tokener.toString());
|
||||||
|
assertEquals('t',tokener.next());
|
||||||
|
assertEquals(" at 1 [character 2 line 1]", tokener.toString());
|
||||||
|
tokener.skipTo('\n');
|
||||||
|
assertEquals("skipTo() improperly modifying indexes"," at 7 [character 8 line 1]", tokener.toString());
|
||||||
|
assertEquals('\n',tokener.next());
|
||||||
|
assertEquals(" at 8 [character 0 line 2]", tokener.toString());
|
||||||
|
assertEquals('A',tokener.next());
|
||||||
|
assertEquals(" at 9 [character 1 line 2]", tokener.toString());
|
||||||
|
tokener.back();
|
||||||
|
assertEquals(" at 8 [character 0 line 2]", tokener.toString());
|
||||||
|
tokener.skipTo('\r');
|
||||||
|
assertEquals("skipTo() improperly modifying indexes"," at 14 [character 6 line 2]", tokener.toString());
|
||||||
|
// verify \r\n combo doesn't increment the line twice
|
||||||
|
assertEquals('\r', tokener.next());
|
||||||
|
assertEquals(" at 15 [character 0 line 3]", tokener.toString());
|
||||||
|
assertEquals('\n', tokener.next());
|
||||||
|
assertEquals(" at 16 [character 0 line 3]", tokener.toString());
|
||||||
|
// verify stepping back after reading the \n of an \r\n combo doesn't increment the line incorrectly
|
||||||
|
tokener.back();
|
||||||
|
assertEquals(" at 15 [character 6 line 2]", tokener.toString());
|
||||||
|
assertEquals('\n', tokener.next());
|
||||||
|
assertEquals(" at 16 [character 0 line 3]", tokener.toString());
|
||||||
|
assertEquals('W', tokener.next());
|
||||||
|
assertEquals(" at 17 [character 1 line 3]", tokener.toString());
|
||||||
|
assertEquals('i', tokener.next());
|
||||||
|
assertEquals(" at 18 [character 2 line 3]", tokener.toString());
|
||||||
|
tokener.skipTo('\r');
|
||||||
|
assertEquals("skipTo() improperly modifying indexes"," at 35 [character 19 line 3]", tokener.toString());
|
||||||
|
assertEquals('\r', tokener.next());
|
||||||
|
assertEquals(" at 36 [character 0 line 4]", tokener.toString());
|
||||||
|
tokener.back();
|
||||||
|
assertEquals(" at 35 [character 19 line 3]", tokener.toString());
|
||||||
|
assertEquals('\r', tokener.next());
|
||||||
|
assertEquals(" at 36 [character 0 line 4]", tokener.toString());
|
||||||
|
assertEquals('N', tokener.next());
|
||||||
|
assertEquals(" at 37 [character 1 line 4]", tokener.toString());
|
||||||
|
|
||||||
|
// verify we get the same data just walking though, no calls to back
|
||||||
|
final JSONTokener t2 = new JSONTokener(testString);
|
||||||
|
for(int i=0; i<7; i++) {
|
||||||
|
assertTrue(t2.toString().startsWith(" at " + i + " "));
|
||||||
|
assertEquals(testString.charAt(i), t2.next());
|
||||||
|
}
|
||||||
|
assertEquals(" at 7 [character 8 line 1]", t2.toString());
|
||||||
|
assertEquals(testString.charAt(7), t2.next());
|
||||||
|
assertEquals(" at 8 [character 0 line 2]", t2.toString());
|
||||||
|
for(int i=8; i<14; i++) {
|
||||||
|
assertTrue(t2.toString().startsWith(" at " + i + " "));
|
||||||
|
assertEquals(testString.charAt(i), t2.next());
|
||||||
|
}
|
||||||
|
assertEquals(" at 14 [character 6 line 2]", t2.toString());
|
||||||
|
assertEquals('\r', t2.next());
|
||||||
|
assertEquals(" at 15 [character 0 line 3]", t2.toString());
|
||||||
|
assertEquals('\n', t2.next());
|
||||||
|
assertEquals(" at 16 [character 0 line 3]", t2.toString());
|
||||||
|
assertEquals('W', t2.next());
|
||||||
|
assertEquals(" at 17 [character 1 line 3]", t2.toString());
|
||||||
|
for(int i=17; i<37; i++) {
|
||||||
|
assertTrue(t2.toString().startsWith(" at " + i + " "));
|
||||||
|
assertEquals(testString.charAt(i), t2.next());
|
||||||
|
}
|
||||||
|
assertEquals(" at 37 [character 1 line 4]", t2.toString());
|
||||||
|
for(int i=37; i<testString.length(); i++) {
|
||||||
|
assertTrue(t2.toString().startsWith(" at " + i + " "));
|
||||||
|
assertEquals(testString.charAt(i), t2.next());
|
||||||
|
}
|
||||||
|
assertEquals(" at "+ testString.length() +" [character 9 line 4]", t2.toString());
|
||||||
|
// end of the input
|
||||||
|
assertEquals(0, t2.next());
|
||||||
|
assertFalse(t2.more());
|
||||||
|
}
|
||||||
|
}
|
122
src/test/java/org/json/junit/PropertyTest.java
Normal file
122
src/test/java/org/json/junit/PropertyTest.java
Normal file
|
@ -0,0 +1,122 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
|
import org.json.*;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for JSON-Java Property
|
||||||
|
*/
|
||||||
|
public class PropertyTest {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSONObject from null properties object should
|
||||||
|
* result in an empty JSONObject.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNullProperties() {
|
||||||
|
Properties properties = null;
|
||||||
|
JSONObject jsonObject = Property.toJSONObject(properties);
|
||||||
|
assertTrue("jsonObject should be empty", jsonObject.isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSONObject from empty properties object should
|
||||||
|
* result in an empty JSONObject.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleEmptyProperties() {
|
||||||
|
Properties properties = new Properties();
|
||||||
|
JSONObject jsonObject = Property.toJSONObject(properties);
|
||||||
|
assertTrue("jsonObject should be empty", jsonObject.isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSONObject from simple properties object.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleProperties() {
|
||||||
|
Properties properties = new Properties();
|
||||||
|
|
||||||
|
properties.put("Illinois", "Springfield");
|
||||||
|
properties.put("Missouri", "Jefferson City");
|
||||||
|
properties.put("Washington", "Olympia");
|
||||||
|
properties.put("California", "Sacramento");
|
||||||
|
properties.put("Indiana", "Indianapolis");
|
||||||
|
|
||||||
|
JSONObject jsonObject = Property.toJSONObject(properties);
|
||||||
|
|
||||||
|
assertTrue("jsonObject should contain 5 items", jsonObject.length() == 5);
|
||||||
|
assertTrue("jsonObject should contain Illinois property",
|
||||||
|
"Springfield".equals(jsonObject.get("Illinois")));
|
||||||
|
assertTrue("jsonObject should contain Missouri property",
|
||||||
|
"Jefferson City".equals(jsonObject.get("Missouri")));
|
||||||
|
assertTrue("jsonObject should contain Washington property",
|
||||||
|
"Olympia".equals(jsonObject.get("Washington")));
|
||||||
|
assertTrue("jsonObject should contain California property",
|
||||||
|
"Sacramento".equals(jsonObject.get("California")));
|
||||||
|
assertTrue("jsonObject should contain Indiana property",
|
||||||
|
"Indianapolis".equals(jsonObject.get("Indiana")));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Null JSONObject toProperties() should result in an empty
|
||||||
|
* Properties object.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNullJSONProperty() {
|
||||||
|
JSONObject jsonObject= null;
|
||||||
|
Properties properties = Property.toProperties(jsonObject);
|
||||||
|
assertTrue("properties should be empty",
|
||||||
|
properties.size() == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Properties should convert to JSONObject, and back to
|
||||||
|
* Properties without changing.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleJSONProperty() {
|
||||||
|
Properties properties = new Properties();
|
||||||
|
|
||||||
|
properties.put("Illinois", "Springfield");
|
||||||
|
properties.put("Missouri", "Jefferson City");
|
||||||
|
properties.put("Washington", "Olympia");
|
||||||
|
properties.put("California", "Sacramento");
|
||||||
|
properties.put("Indiana", "Indianapolis");
|
||||||
|
|
||||||
|
JSONObject jsonObject = Property.toJSONObject(properties);
|
||||||
|
Properties jsonProperties = Property.toProperties(jsonObject);
|
||||||
|
|
||||||
|
assertTrue("property objects should match",
|
||||||
|
properties.equals(jsonProperties));
|
||||||
|
}
|
||||||
|
}
|
122
src/test/java/org/json/junit/Util.java
Normal file
122
src/test/java/org/json/junit/Util.java
Normal file
|
@ -0,0 +1,122 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
import org.json.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* These are helpful utility methods that perform basic comparisons
|
||||||
|
* between various objects. In most cases, the comparisons are not
|
||||||
|
* order-dependent, or else the order is known.
|
||||||
|
*/
|
||||||
|
public class Util {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compares two JSONArrays for equality.
|
||||||
|
* The arrays need not be in the same order.
|
||||||
|
* @param jsonArray created by the code to be tested
|
||||||
|
* @param expectedJsonArray created specifically for comparing
|
||||||
|
*/
|
||||||
|
public static void compareActualVsExpectedJsonArrays(JSONArray jsonArray,
|
||||||
|
JSONArray expectedJsonArray) {
|
||||||
|
assertTrue("jsonArray lengths should be equal",
|
||||||
|
jsonArray.length() == expectedJsonArray.length());
|
||||||
|
for (int i = 0; i < jsonArray.length(); ++i) {
|
||||||
|
Object value = jsonArray.get(i);
|
||||||
|
Object expectedValue = expectedJsonArray.get(i);
|
||||||
|
compareActualVsExpectedObjects(value, expectedValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compares two JSONObjects for equality. The objects need not be
|
||||||
|
* in the same order
|
||||||
|
* @param jsonObject created by the code to be tested
|
||||||
|
* @param expectedJsonObject created specifically for comparing
|
||||||
|
*/
|
||||||
|
public static void compareActualVsExpectedJsonObjects(
|
||||||
|
JSONObject jsonObject, JSONObject expectedJsonObject) {
|
||||||
|
assertTrue("jsonObjects should have the same length",
|
||||||
|
jsonObject.length() == expectedJsonObject.length());
|
||||||
|
Iterator<String> keys = jsonObject.keys();
|
||||||
|
while (keys.hasNext()) {
|
||||||
|
String key = keys.next();
|
||||||
|
Object value = jsonObject.get(key);
|
||||||
|
Object expectedValue = expectedJsonObject.get(key);
|
||||||
|
compareActualVsExpectedObjects(value, expectedValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compare two objects for equality. Might be JSONArray, JSONObject,
|
||||||
|
* or something else.
|
||||||
|
* @param value created by the code to be tested
|
||||||
|
* @param expectedValue created specifically for comparing
|
||||||
|
* @param key key to the jsonObject entry to be compared
|
||||||
|
*/
|
||||||
|
private static void compareActualVsExpectedObjects(Object value,
|
||||||
|
Object expectedValue) {
|
||||||
|
if (value instanceof JSONObject && expectedValue instanceof JSONObject) {
|
||||||
|
// Compare JSONObjects
|
||||||
|
JSONObject jsonObject = (JSONObject)value;
|
||||||
|
JSONObject expectedJsonObject = (JSONObject)expectedValue;
|
||||||
|
compareActualVsExpectedJsonObjects(
|
||||||
|
jsonObject, expectedJsonObject);
|
||||||
|
} else if (value instanceof JSONArray && expectedValue instanceof JSONArray) {
|
||||||
|
// Compare JSONArrays
|
||||||
|
JSONArray jsonArray = (JSONArray)value;
|
||||||
|
JSONArray expectedJsonArray = (JSONArray)expectedValue;
|
||||||
|
compareActualVsExpectedJsonArrays(
|
||||||
|
jsonArray, expectedJsonArray);
|
||||||
|
} else {
|
||||||
|
/**
|
||||||
|
* Compare all other types using toString(). First, the types must
|
||||||
|
* also be equal, unless both are Number type. Certain helper
|
||||||
|
* classes (e.g. XML) may create Long instead of Integer for small
|
||||||
|
* int values.
|
||||||
|
*/
|
||||||
|
if (!(value instanceof Number && expectedValue instanceof Number)) {
|
||||||
|
// Non-Number and non-matching types
|
||||||
|
assertTrue("object types should be equal for actual: "+
|
||||||
|
value.toString()+" ("+
|
||||||
|
value.getClass().toString()+") expected: "+
|
||||||
|
expectedValue.toString()+" ("+
|
||||||
|
expectedValue.getClass().toString()+")",
|
||||||
|
value.getClass().toString().equals(
|
||||||
|
expectedValue.getClass().toString()));
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Same types or both Numbers, compare by toString()
|
||||||
|
*/
|
||||||
|
assertTrue("string values should be equal for actual: "+
|
||||||
|
value.toString()+" expected: "+expectedValue.toString(),
|
||||||
|
value.toString().equals(expectedValue.toString()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
958
src/test/java/org/json/junit/XMLConfigurationTest.java
Executable file
958
src/test/java/org/json/junit/XMLConfigurationTest.java
Executable file
|
@ -0,0 +1,958 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static 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.FileReader;
|
||||||
|
import java.io.FileWriter;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.Reader;
|
||||||
|
import java.io.StringReader;
|
||||||
|
|
||||||
|
import org.json.JSONArray;
|
||||||
|
import org.json.JSONException;
|
||||||
|
import org.json.JSONObject;
|
||||||
|
import org.json.XML;
|
||||||
|
import org.json.XMLParserConfiguration;
|
||||||
|
import org.junit.Rule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.rules.TemporaryFolder;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for JSON-Java XML.java with XMLParserConfiguration.java
|
||||||
|
*/
|
||||||
|
public class XMLConfigurationTest {
|
||||||
|
/**
|
||||||
|
* JUnit supports temporary files and folders that are cleaned up after the test.
|
||||||
|
* https://garygregory.wordpress.com/2010/01/20/junit-tip-use-rules-to-manage-temporary-files-and-folders/
|
||||||
|
*/
|
||||||
|
@Rule
|
||||||
|
public TemporaryFolder testFolder = new TemporaryFolder();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSONObject from a null XML string.
|
||||||
|
* Expects a NullPointerException
|
||||||
|
*/
|
||||||
|
@Test(expected=NullPointerException.class)
|
||||||
|
public void shouldHandleNullXML() {
|
||||||
|
String xmlStr = null;
|
||||||
|
JSONObject jsonObject =
|
||||||
|
XML.toJSONObject(xmlStr, XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
assertTrue("jsonObject should be empty", jsonObject.isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Empty JSONObject from an empty XML string.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleEmptyXML() {
|
||||||
|
|
||||||
|
String xmlStr = "";
|
||||||
|
JSONObject jsonObject =
|
||||||
|
XML.toJSONObject(xmlStr, XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
assertTrue("jsonObject should be empty", jsonObject.isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Empty JSONObject from a non-XML string.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNonXML() {
|
||||||
|
String xmlStr = "{ \"this is\": \"not xml\"}";
|
||||||
|
JSONObject jsonObject =
|
||||||
|
XML.toJSONObject(xmlStr, XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
assertTrue("xml string should be empty", jsonObject.isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Invalid XML string (tag contains a frontslash).
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleInvalidSlashInTag() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/x>\n"+
|
||||||
|
" <street>abc street</street>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
XML.toJSONObject(xmlStr, XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Misshaped tag at 176 [character 14 line 4]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Invalid XML string ('!' char in tag)
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleInvalidBangInTag() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/>\n"+
|
||||||
|
" <!>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
XML.toJSONObject(xmlStr, XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Misshaped meta tag at 214 [character 12 line 7]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Invalid XML string ('!' char and no closing tag brace)
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleInvalidBangNoCloseInTag() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/>\n"+
|
||||||
|
" <!\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
XML.toJSONObject(xmlStr, XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Misshaped meta tag at 213 [character 12 line 7]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Invalid XML string (no end brace for tag)
|
||||||
|
* Expects JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNoCloseStartTag() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/>\n"+
|
||||||
|
" <abc\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
XML.toJSONObject(xmlStr, XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Misplaced '<' at 193 [character 4 line 6]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Invalid XML string (partial CDATA chars in tag name)
|
||||||
|
* Expects JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleInvalidCDATABangInTag() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name>Joe Tester</name>\n"+
|
||||||
|
" <![[]>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
XMLParserConfiguration config =
|
||||||
|
new XMLParserConfiguration("altContent");
|
||||||
|
XML.toJSONObject(xmlStr, config);
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Expected 'CDATA[' at 204 [character 11 line 5]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Null JSONObject in XML.toString()
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNullJSONXML() {
|
||||||
|
JSONObject jsonObject= null;
|
||||||
|
String actualXml = XML.toString(jsonObject, null,
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
assertEquals("generated XML does not equal expected XML","\"null\"",actualXml);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Empty JSONObject in XML.toString()
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleEmptyJSONXML() {
|
||||||
|
JSONObject jsonObject= new JSONObject();
|
||||||
|
String xmlStr = XML.toString(jsonObject, null,
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
assertTrue("xml string should be empty", xmlStr.isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* No SML start tag. The ending tag ends up being treated as content.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNoStartTag() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/>\n"+
|
||||||
|
" <nocontent/>>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":{\"address\":{\"name\":\"\",\"nocontent\":\"\",\""+
|
||||||
|
"content\":\">\"},\"xsi:noNamespaceSchemaLocation\":\"test.xsd\",\""+
|
||||||
|
"xmlns:xsi\":\"http://www.w3.org/2001/XMLSchema-instance\"}}";
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr,
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Valid XML to JSONObject
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleSimpleXML() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name>Joe Tester</name>\n"+
|
||||||
|
" <street>[CDATA[Baker street 5]</street>\n"+
|
||||||
|
" <NothingHere/>\n"+
|
||||||
|
" <TrueValue>true</TrueValue>\n"+
|
||||||
|
" <FalseValue>false</FalseValue>\n"+
|
||||||
|
" <NullValue>null</NullValue>\n"+
|
||||||
|
" <PositiveValue>42</PositiveValue>\n"+
|
||||||
|
" <NegativeValue>-23</NegativeValue>\n"+
|
||||||
|
" <DoubleValue>-23.45</DoubleValue>\n"+
|
||||||
|
" <Nan>-23x.45</Nan>\n"+
|
||||||
|
" <ArrayOfNum>1, 2, 3, 4.1, 5.2</ArrayOfNum>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":{\"address\":{\"street\":\"[CDATA[Baker street 5]\","+
|
||||||
|
"\"name\":\"Joe Tester\",\"NothingHere\":\"\",TrueValue:true,\n"+
|
||||||
|
"\"FalseValue\":false,\"NullValue\":null,\"PositiveValue\":42,\n"+
|
||||||
|
"\"NegativeValue\":-23,\"DoubleValue\":-23.45,\"Nan\":-23x.45,\n"+
|
||||||
|
"\"ArrayOfNum\":\"1, 2, 3, 4.1, 5.2\"\n"+
|
||||||
|
"},\"xsi:noNamespaceSchemaLocation\":"+
|
||||||
|
"\"test.xsd\",\"xmlns:xsi\":\"http://www.w3.org/2001/"+
|
||||||
|
"XMLSchema-instance\"}}";
|
||||||
|
|
||||||
|
XMLParserConfiguration config =
|
||||||
|
new XMLParserConfiguration("altContent");
|
||||||
|
compareStringToJSONObject(xmlStr, expectedStr, config);
|
||||||
|
compareReaderToJSONObject(xmlStr, expectedStr, config);
|
||||||
|
compareFileToJSONObject(xmlStr, expectedStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Valid XML with comments to JSONObject
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleCommentsInXML() {
|
||||||
|
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<!-- this is a comment -->\n"+
|
||||||
|
"<addresses>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <![CDATA[ this is -- <another> comment ]]>\n"+
|
||||||
|
" <name>Joe Tester</name>\n"+
|
||||||
|
" <!-- this is a - multi line \n"+
|
||||||
|
" comment -->\n"+
|
||||||
|
" <street>Baker street 5</street>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
XMLParserConfiguration config =
|
||||||
|
new XMLParserConfiguration("altContent");
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr, config);
|
||||||
|
String expectedStr = "{\"addresses\":{\"address\":{\"street\":\"Baker "+
|
||||||
|
"street 5\",\"name\":\"Joe Tester\",\"altContent\":\" this is -- "+
|
||||||
|
"<another> comment \"}}}";
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Valid XML to XML.toString()
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleToString() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name>[CDATA[Joe & T > e < s " t ' er]]</name>\n"+
|
||||||
|
" <street>Baker street 5</street>\n"+
|
||||||
|
" <ArrayOfNum>1, 2, 3, 4.1, 5.2</ArrayOfNum>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":{\"address\":{\"street\":\"Baker street 5\","+
|
||||||
|
"\"name\":\"[CDATA[Joe & T > e < s \\\" t \\\' er]]\","+
|
||||||
|
"\"ArrayOfNum\":\"1, 2, 3, 4.1, 5.2\"\n"+
|
||||||
|
"},\"xsi:noNamespaceSchemaLocation\":"+
|
||||||
|
"\"test.xsd\",\"xmlns:xsi\":\"http://www.w3.org/2001/"+
|
||||||
|
"XMLSchema-instance\"}}";
|
||||||
|
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr,
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
String xmlToStr = XML.toString(jsonObject, null,
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
JSONObject finalJsonObject = XML.toJSONObject(xmlToStr,
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(finalJsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converting a JSON doc containing '>' content to JSONObject, then
|
||||||
|
* XML.toString() should result in valid XML.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleContentNoArraytoString() {
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":{\"altContent\":\">\"}}";
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
XMLParserConfiguration config = new XMLParserConfiguration("altContent");
|
||||||
|
String finalStr = XML.toString(expectedJsonObject, null, config);
|
||||||
|
String expectedFinalStr = "<addresses>></addresses>";
|
||||||
|
assertTrue("Should handle expectedFinal: ["+expectedStr+"] final: ["+
|
||||||
|
finalStr+"]", expectedFinalStr.equals(finalStr));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converting a JSON doc containing a 'content' array to JSONObject, then
|
||||||
|
* XML.toString() should result in valid XML.
|
||||||
|
* TODO: This is probably an error in how the 'content' keyword is used.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleContentArraytoString() {
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":{\"altContent\":[1, 2, 3]}}";
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
XMLParserConfiguration config = new XMLParserConfiguration("altContent");
|
||||||
|
String finalStr = XML.toString(expectedJsonObject, null, config);
|
||||||
|
String expectedFinalStr = "<addresses>"+
|
||||||
|
"1\n2\n3"+
|
||||||
|
"</addresses>";
|
||||||
|
assertTrue("Should handle expectedFinal: ["+expectedStr+"] final: ["+
|
||||||
|
finalStr+"]", expectedFinalStr.equals(finalStr));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converting a JSON doc containing a named array to JSONObject, then
|
||||||
|
* XML.toString() should result in valid XML.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleArraytoString() {
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":{"+
|
||||||
|
"\"something\":[1, 2, 3]}}";
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
String finalStr = XML.toString(expectedJsonObject, null,
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
String expectedFinalStr = "<addresses>"+
|
||||||
|
"<something>1</something><something>2</something><something>3</something>"+
|
||||||
|
"</addresses>";
|
||||||
|
assertTrue("Should handle expectedFinal: ["+expectedStr+"] final: ["+
|
||||||
|
finalStr+"]", expectedFinalStr.equals(finalStr));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests that the XML output for empty arrays is consistent.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleEmptyArray(){
|
||||||
|
final JSONObject jo1 = new JSONObject();
|
||||||
|
jo1.put("array",new Object[]{});
|
||||||
|
final JSONObject jo2 = new JSONObject();
|
||||||
|
jo2.put("array",new JSONArray());
|
||||||
|
|
||||||
|
final String expected = "<jo></jo>";
|
||||||
|
String output1 = XML.toString(jo1, "jo",
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
assertEquals("Expected an empty root tag", expected, output1);
|
||||||
|
String output2 = XML.toString(jo2, "jo",
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
assertEquals("Expected an empty root tag", expected, output2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests that the XML output for arrays is consistent when an internal array is empty.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleEmptyMultiArray(){
|
||||||
|
final JSONObject jo1 = new JSONObject();
|
||||||
|
jo1.put("arr",new Object[]{"One", new String[]{}, "Four"});
|
||||||
|
final JSONObject jo2 = new JSONObject();
|
||||||
|
jo2.put("arr",new JSONArray(new Object[]{"One", new JSONArray(new String[]{}), "Four"}));
|
||||||
|
|
||||||
|
final String expected = "<jo><arr>One</arr><arr></arr><arr>Four</arr></jo>";
|
||||||
|
String output1 = XML.toString(jo1, "jo",
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
assertEquals("Expected a matching array", expected, output1);
|
||||||
|
String output2 = XML.toString(jo2, "jo",
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
|
||||||
|
assertEquals("Expected a matching array", expected, output2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests that the XML output for arrays is consistent when arrays are not empty.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNonEmptyArray(){
|
||||||
|
final JSONObject jo1 = new JSONObject();
|
||||||
|
jo1.put("arr",new String[]{"One", "Two", "Three"});
|
||||||
|
final JSONObject jo2 = new JSONObject();
|
||||||
|
jo2.put("arr",new JSONArray(new String[]{"One", "Two", "Three"}));
|
||||||
|
|
||||||
|
final String expected = "<jo><arr>One</arr><arr>Two</arr><arr>Three</arr></jo>";
|
||||||
|
String output1 = XML.toString(jo1, "jo",
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
assertEquals("Expected a non empty root tag", expected, output1);
|
||||||
|
String output2 = XML.toString(jo2, "jo",
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
assertEquals("Expected a non empty root tag", expected, output2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests that the XML output for arrays is consistent when arrays are not empty and contain internal arrays.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleMultiArray(){
|
||||||
|
final JSONObject jo1 = new JSONObject();
|
||||||
|
jo1.put("arr",new Object[]{"One", new String[]{"Two", "Three"}, "Four"});
|
||||||
|
final JSONObject jo2 = new JSONObject();
|
||||||
|
jo2.put("arr",new JSONArray(new Object[]{"One", new JSONArray(new String[]{"Two", "Three"}), "Four"}));
|
||||||
|
|
||||||
|
final String expected = "<jo><arr>One</arr><arr><array>Two</array><array>Three</array></arr><arr>Four</arr></jo>";
|
||||||
|
String output1 = XML.toString(jo1, "jo",
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
assertEquals("Expected a matching array", expected, output1);
|
||||||
|
String output2 = XML.toString(jo2, "jo",
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
assertEquals("Expected a matching array", expected, output2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converting a JSON doc containing a named array of nested arrays to
|
||||||
|
* JSONObject, then XML.toString() should result in valid XML.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNestedArraytoString() {
|
||||||
|
String xmlStr =
|
||||||
|
"{\"addresses\":{\"address\":{\"name\":\"\",\"nocontent\":\"\","+
|
||||||
|
"\"outer\":[[1], [2], [3]]},\"xsi:noNamespaceSchemaLocation\":\"test.xsd\",\""+
|
||||||
|
"xmlns:xsi\":\"http://www.w3.org/2001/XMLSchema-instance\"}}";
|
||||||
|
JSONObject jsonObject = new JSONObject(xmlStr);
|
||||||
|
String finalStr = XML.toString(jsonObject, null,
|
||||||
|
XMLParserConfiguration.ORIGINAL);
|
||||||
|
JSONObject finalJsonObject = XML.toJSONObject(finalStr);
|
||||||
|
String expectedStr = "<addresses><address><name/><nocontent/>"+
|
||||||
|
"<outer><array>1</array></outer><outer><array>2</array>"+
|
||||||
|
"</outer><outer><array>3</array></outer>"+
|
||||||
|
"</address><xsi:noNamespaceSchemaLocation>test.xsd</xsi:noName"+
|
||||||
|
"spaceSchemaLocation><xmlns:xsi>http://www.w3.org/2001/XMLSche"+
|
||||||
|
"ma-instance</xmlns:xsi></addresses>";
|
||||||
|
JSONObject expectedJsonObject = XML.toJSONObject(expectedStr,
|
||||||
|
XMLParserConfiguration.ORIGINAL);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(finalJsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Possible bug:
|
||||||
|
* Illegal node-names must be converted to legal XML-node-names.
|
||||||
|
* The given example shows 2 nodes which are valid for JSON, but not for XML.
|
||||||
|
* Therefore illegal arguments should be converted to e.g. an underscore (_).
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleIllegalJSONNodeNames()
|
||||||
|
{
|
||||||
|
JSONObject inputJSON = new JSONObject();
|
||||||
|
inputJSON.append("123IllegalNode", "someValue1");
|
||||||
|
inputJSON.append("Illegal@node", "someValue2");
|
||||||
|
|
||||||
|
String result = XML.toString(inputJSON, null,
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This is invalid XML. Names should not begin with digits or contain
|
||||||
|
* certain values, including '@'. One possible solution is to replace
|
||||||
|
* illegal chars with '_', in which case the expected output would be:
|
||||||
|
* <___IllegalNode>someValue1</___IllegalNode><Illegal_node>someValue2</Illegal_node>
|
||||||
|
*/
|
||||||
|
String expected = "<123IllegalNode>someValue1</123IllegalNode><Illegal@node>someValue2</Illegal@node>";
|
||||||
|
|
||||||
|
assertEquals("Length", expected.length(), result.length());
|
||||||
|
assertTrue("123IllegalNode", result.contains("<123IllegalNode>someValue1</123IllegalNode>"));
|
||||||
|
assertTrue("Illegal@node", result.contains("<Illegal@node>someValue2</Illegal@node>"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSONObject with NULL value, to XML.toString()
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNullNodeValue()
|
||||||
|
{
|
||||||
|
JSONObject inputJSON = new JSONObject();
|
||||||
|
inputJSON.put("nullValue", JSONObject.NULL);
|
||||||
|
// This is a possible preferred result
|
||||||
|
// String expectedXML = "<nullValue/>";
|
||||||
|
/**
|
||||||
|
* This is the current behavior. JSONObject.NULL is emitted as
|
||||||
|
* the string, "null".
|
||||||
|
*/
|
||||||
|
String actualXML = "<nullValue>null</nullValue>";
|
||||||
|
String resultXML = XML.toString(inputJSON, null,
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
assertEquals(actualXML, resultXML);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Investigate exactly how the "content" keyword works
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void contentOperations() {
|
||||||
|
/*
|
||||||
|
* When a standalone <!CDATA[...]] structure is found while parsing XML into a
|
||||||
|
* JSONObject, the contents are placed in a string value with key="content".
|
||||||
|
*/
|
||||||
|
String xmlStr = "<tag1></tag1><![CDATA[if (a < b && a > 0) then return]]><tag2></tag2>";
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr,
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
assertTrue("1. 3 items", 3 == jsonObject.length());
|
||||||
|
assertTrue("1. empty tag1", "".equals(jsonObject.get("tag1")));
|
||||||
|
assertTrue("1. empty tag2", "".equals(jsonObject.get("tag2")));
|
||||||
|
assertTrue("1. content found", "if (a < b && a > 0) then return".equals(jsonObject.get("content")));
|
||||||
|
|
||||||
|
// multiple consecutive standalone cdatas are accumulated into an array
|
||||||
|
xmlStr = "<tag1></tag1><![CDATA[if (a < b && a > 0) then return]]><tag2></tag2><![CDATA[here is another cdata]]>";
|
||||||
|
jsonObject = XML.toJSONObject(xmlStr,
|
||||||
|
new XMLParserConfiguration(true, "altContent"));
|
||||||
|
assertTrue("2. 3 items", 3 == jsonObject.length());
|
||||||
|
assertTrue("2. empty tag1", "".equals(jsonObject.get("tag1")));
|
||||||
|
assertTrue("2. empty tag2", "".equals(jsonObject.get("tag2")));
|
||||||
|
assertTrue("2. content array found", jsonObject.get("altContent") instanceof JSONArray);
|
||||||
|
JSONArray jsonArray = jsonObject.getJSONArray("altContent");
|
||||||
|
assertTrue("2. array size", jsonArray.length() == 2);
|
||||||
|
assertTrue("2. content array entry 0", "if (a < b && a > 0) then return".equals(jsonArray.get(0)));
|
||||||
|
assertTrue("2. content array entry 1", "here is another cdata".equals(jsonArray.get(1)));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* text content is accumulated in a "content" inside a local JSONObject.
|
||||||
|
* If there is only one instance, it is saved in the context (a different JSONObject
|
||||||
|
* from the calling code. and the content element is discarded.
|
||||||
|
*/
|
||||||
|
xmlStr = "<tag1>value 1</tag1>";
|
||||||
|
jsonObject = XML.toJSONObject(xmlStr,
|
||||||
|
new XMLParserConfiguration(true, "altContent"));
|
||||||
|
assertTrue("3. 2 items", 1 == jsonObject.length());
|
||||||
|
assertTrue("3. value tag1", "value 1".equals(jsonObject.get("tag1")));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* array-style text content (multiple tags with the same name) is
|
||||||
|
* accumulated in a local JSONObject with key="content" and value=JSONArray,
|
||||||
|
* saved in the context, and then the local JSONObject is discarded.
|
||||||
|
*/
|
||||||
|
xmlStr = "<tag1>value 1</tag1><tag1>2</tag1><tag1>true</tag1>";
|
||||||
|
jsonObject = XML.toJSONObject(xmlStr,
|
||||||
|
new XMLParserConfiguration(true, "altContent"));
|
||||||
|
assertTrue("4. 1 item", 1 == jsonObject.length());
|
||||||
|
assertTrue("4. content array found", jsonObject.get("tag1") instanceof JSONArray);
|
||||||
|
jsonArray = jsonObject.getJSONArray("tag1");
|
||||||
|
assertTrue("4. array size", jsonArray.length() == 3);
|
||||||
|
assertTrue("4. content array entry 0", "value 1".equals(jsonArray.get(0)));
|
||||||
|
assertTrue("4. content array entry 1", jsonArray.getInt(1) == 2);
|
||||||
|
assertTrue("4. content array entry 2", jsonArray.getBoolean(2) == true);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Complex content is accumulated in a "content" field. For example, an element
|
||||||
|
* may contain a mix of child elements and text. Each text segment is
|
||||||
|
* accumulated to content.
|
||||||
|
*/
|
||||||
|
xmlStr = "<tag1>val1<tag2/>val2</tag1>";
|
||||||
|
jsonObject = XML.toJSONObject(xmlStr,
|
||||||
|
new XMLParserConfiguration(true, "altContent"));
|
||||||
|
assertTrue("5. 1 item", 1 == jsonObject.length());
|
||||||
|
assertTrue("5. jsonObject found", jsonObject.get("tag1")
|
||||||
|
instanceof JSONObject);
|
||||||
|
jsonObject = jsonObject.getJSONObject("tag1");
|
||||||
|
assertTrue("5. 2 contained items", 2 == jsonObject.length());
|
||||||
|
assertTrue("5. contained tag", "".equals(jsonObject.get("tag2")));
|
||||||
|
assertTrue("5. contained content jsonArray found",
|
||||||
|
jsonObject.get("altContent") instanceof JSONArray);
|
||||||
|
jsonArray = jsonObject.getJSONArray("altContent");
|
||||||
|
assertTrue("5. array size", jsonArray.length() == 2);
|
||||||
|
assertTrue("5. content array entry 0", "val1".equals(jsonArray.get(0)));
|
||||||
|
assertTrue("5. content array entry 1", "val2".equals(jsonArray.get(1)));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If there is only 1 complex text content, then it is accumulated in a
|
||||||
|
* "content" field as a string.
|
||||||
|
*/
|
||||||
|
xmlStr = "<tag1>val1<tag2/></tag1>";
|
||||||
|
jsonObject = XML.toJSONObject(xmlStr,
|
||||||
|
new XMLParserConfiguration(true, "altContent"));
|
||||||
|
assertTrue("6. 1 item", 1 == jsonObject.length());
|
||||||
|
assertTrue("6. jsonObject found", jsonObject.get("tag1") instanceof JSONObject);
|
||||||
|
jsonObject = jsonObject.getJSONObject("tag1");
|
||||||
|
assertTrue("6. contained content found",
|
||||||
|
"val1".equals(jsonObject.get("altContent")));
|
||||||
|
assertTrue("6. contained tag2", "".equals(jsonObject.get("tag2")));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* In this corner case, the content sibling happens to have key=content
|
||||||
|
* We end up with an array within an array, and no content element.
|
||||||
|
* This is probably a bug.
|
||||||
|
*/
|
||||||
|
xmlStr = "<tag1>val1<altContent/></tag1>";
|
||||||
|
jsonObject = XML.toJSONObject(xmlStr,
|
||||||
|
new XMLParserConfiguration(true, "altContent"));
|
||||||
|
assertTrue("7. 1 item", 1 == jsonObject.length());
|
||||||
|
assertTrue("7. jsonArray found",
|
||||||
|
jsonObject.get("tag1") instanceof JSONArray);
|
||||||
|
jsonArray = jsonObject.getJSONArray("tag1");
|
||||||
|
assertTrue("array size 1", jsonArray.length() == 1);
|
||||||
|
assertTrue("7. contained array found", jsonArray.get(0)
|
||||||
|
instanceof JSONArray);
|
||||||
|
jsonArray = jsonArray.getJSONArray(0);
|
||||||
|
assertTrue("7. inner array size 2", jsonArray.length() == 2);
|
||||||
|
assertTrue("7. inner array item 0", "val1".equals(jsonArray.get(0)));
|
||||||
|
assertTrue("7. inner array item 1", "".equals(jsonArray.get(1)));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Confirm behavior of original issue
|
||||||
|
*/
|
||||||
|
String jsonStr =
|
||||||
|
"{"+
|
||||||
|
"\"Profile\": {"+
|
||||||
|
"\"list\": {"+
|
||||||
|
"\"history\": {"+
|
||||||
|
"\"entries\": ["+
|
||||||
|
"{"+
|
||||||
|
"\"deviceId\": \"id\","+
|
||||||
|
"\"altContent\": {"+
|
||||||
|
"\"material\": ["+
|
||||||
|
"{"+
|
||||||
|
"\"stuff\": false"+
|
||||||
|
"}"+
|
||||||
|
"]"+
|
||||||
|
"}"+
|
||||||
|
"}"+
|
||||||
|
"]"+
|
||||||
|
"}"+
|
||||||
|
"}"+
|
||||||
|
"}"+
|
||||||
|
"}";
|
||||||
|
jsonObject = new JSONObject(jsonStr);
|
||||||
|
xmlStr = XML.toString(jsonObject, null,
|
||||||
|
new XMLParserConfiguration(true, "altContent"));
|
||||||
|
/*
|
||||||
|
* This is the created XML. Looks like content was mistaken for
|
||||||
|
* complex (child node + text) XML.
|
||||||
|
* <Profile>
|
||||||
|
* <list>
|
||||||
|
* <history>
|
||||||
|
* <entries>
|
||||||
|
* <deviceId>id</deviceId>
|
||||||
|
* {"material":[{"stuff":false}]}
|
||||||
|
* </entries>
|
||||||
|
* </history>
|
||||||
|
* </list>
|
||||||
|
* </Profile>
|
||||||
|
*/
|
||||||
|
assertTrue("nothing to test here, see comment on created XML, above", true);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSON string lost leading zero and converted "True" to true.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testToJSONArray_jsonOutput() {
|
||||||
|
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
||||||
|
final JSONObject expected = new JSONObject("{\"root\":{\"item\":{\"id\":\"01\"},\"id\":[\"01\",1,\"00\",0],\"title\":true}}");
|
||||||
|
final JSONObject actualJsonOutput = XML.toJSONObject(originalXml,
|
||||||
|
new XMLParserConfiguration(false));
|
||||||
|
Util.compareActualVsExpectedJsonObjects(actualJsonOutput,expected);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSON string cannot be reverted to original xml.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testToJSONArray_reversibility() {
|
||||||
|
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
||||||
|
XMLParserConfiguration config = new XMLParserConfiguration(false);
|
||||||
|
final String revertedXml =
|
||||||
|
XML.toString(XML.toJSONObject(originalXml, config),
|
||||||
|
null, config);
|
||||||
|
assertNotEquals(revertedXml, originalXml);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test passes when using the new method toJsonArray.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testToJsonXML() {
|
||||||
|
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
||||||
|
final JSONObject expected = new JSONObject("{\"root\":{\"item\":{\"id\":\"01\"},\"id\":[\"01\",\"1\",\"00\",\"0\"],\"title\":\"True\"}}");
|
||||||
|
|
||||||
|
final JSONObject json = XML.toJSONObject(originalXml,
|
||||||
|
new XMLParserConfiguration(true));
|
||||||
|
Util.compareActualVsExpectedJsonObjects(json, expected);
|
||||||
|
|
||||||
|
final String reverseXml = XML.toString(json);
|
||||||
|
// this reversal isn't exactly the same. use JSONML for an exact reversal
|
||||||
|
final String expectedReverseXml = "<root><item><id>01</id></item><id>01</id><id>1</id><id>00</id><id>0</id><title>True</title></root>";
|
||||||
|
|
||||||
|
assertEquals("length",expectedReverseXml.length(), reverseXml.length());
|
||||||
|
assertTrue("array contents", reverseXml.contains("<id>01</id><id>1</id><id>00</id><id>0</id>"));
|
||||||
|
assertTrue("item contents", reverseXml.contains("<item><id>01</id></item>"));
|
||||||
|
assertTrue("title contents", reverseXml.contains("<title>True</title>"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test to validate certain conditions of XML unescaping.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testUnescape() {
|
||||||
|
assertEquals("{\"xml\":\"Can cope <;\"}",
|
||||||
|
XML.toJSONObject("<xml>Can cope <; </xml>",
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS).toString());
|
||||||
|
assertEquals("Can cope <; ", XML.unescape("Can cope <; "));
|
||||||
|
|
||||||
|
assertEquals("{\"xml\":\"Can cope & ;\"}",
|
||||||
|
XML.toJSONObject("<xml>Can cope & ; </xml>",
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS).toString());
|
||||||
|
assertEquals("Can cope & ; ", XML.unescape("Can cope & ; "));
|
||||||
|
|
||||||
|
assertEquals("{\"xml\":\"Can cope &;\"}",
|
||||||
|
XML.toJSONObject("<xml>Can cope &; </xml>",
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS).toString());
|
||||||
|
assertEquals("Can cope &; ", XML.unescape("Can cope &; "));
|
||||||
|
|
||||||
|
// unicode entity
|
||||||
|
assertEquals("{\"xml\":\"Can cope 4;\"}",
|
||||||
|
XML.toJSONObject("<xml>Can cope 4; </xml>",
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS).toString());
|
||||||
|
assertEquals("Can cope 4; ", XML.unescape("Can cope 4; "));
|
||||||
|
|
||||||
|
// double escaped
|
||||||
|
assertEquals("{\"xml\":\"Can cope <\"}",
|
||||||
|
XML.toJSONObject("<xml>Can cope &lt; </xml>",
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS).toString());
|
||||||
|
assertEquals("Can cope < ", XML.unescape("Can cope &lt; "));
|
||||||
|
|
||||||
|
assertEquals("{\"xml\":\"Can cope 4\"}",
|
||||||
|
XML.toJSONObject("<xml>Can cope &#x34; </xml>",
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS).toString());
|
||||||
|
assertEquals("Can cope 4 ", XML.unescape("Can cope &#x34; "));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Confirm XMLParserConfiguration functionality
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testConfig() {
|
||||||
|
/**
|
||||||
|
* 1st param is whether to keep the raw string, or call
|
||||||
|
* XML.stringToValue(), which may convert the token to
|
||||||
|
* boolean, null, or number.
|
||||||
|
* 2nd param is what JSON name to use for strings that are
|
||||||
|
* evaluated as xml content data in complex objects, e.g.
|
||||||
|
* <parent>
|
||||||
|
* <child>value</child>
|
||||||
|
* content data
|
||||||
|
* </tag>
|
||||||
|
*/
|
||||||
|
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" content 1\n"+
|
||||||
|
" <name>Sherlock Holmes</name>\n"+
|
||||||
|
" content 2\n"+
|
||||||
|
" <street>Baker street 5</street>\n"+
|
||||||
|
" content 3\n"+
|
||||||
|
" <num>1</num>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
|
||||||
|
// keep strings, use the altContent tag
|
||||||
|
XMLParserConfiguration config =
|
||||||
|
new XMLParserConfiguration(true, "altContent");
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr, config);
|
||||||
|
// num is parsed as a string
|
||||||
|
assertEquals(jsonObject.getJSONObject("addresses").
|
||||||
|
getJSONObject("address").getString("num"), "1");
|
||||||
|
// complex content is collected in an 'altContent' array
|
||||||
|
JSONArray jsonArray = jsonObject.getJSONObject("addresses").
|
||||||
|
getJSONObject("address").getJSONArray("altContent");
|
||||||
|
String expectedStr = "[\"content 1\", \"content 2\", \"content 3\"]";
|
||||||
|
JSONArray expectedJsonArray = new JSONArray(expectedStr);
|
||||||
|
Util.compareActualVsExpectedJsonArrays(jsonArray, expectedJsonArray);
|
||||||
|
|
||||||
|
// keepstrings only
|
||||||
|
jsonObject = XML.toJSONObject(xmlStr,
|
||||||
|
XMLParserConfiguration.KEEP_STRINGS);
|
||||||
|
// num is parsed as a string
|
||||||
|
assertEquals(jsonObject.getJSONObject("addresses").
|
||||||
|
getJSONObject("address").getString("num"), "1");
|
||||||
|
// complex content is collected in an 'content' array
|
||||||
|
jsonArray = jsonObject.getJSONObject("addresses").
|
||||||
|
getJSONObject("address").getJSONArray("content");
|
||||||
|
expectedJsonArray = new JSONArray(expectedStr);
|
||||||
|
Util.compareActualVsExpectedJsonArrays(jsonArray, expectedJsonArray);
|
||||||
|
|
||||||
|
// use alternate content name
|
||||||
|
config = new XMLParserConfiguration("altContent");
|
||||||
|
jsonObject = XML.toJSONObject(xmlStr, config);
|
||||||
|
// num is parsed as a number
|
||||||
|
assertEquals(jsonObject.getJSONObject("addresses").
|
||||||
|
getJSONObject("address").getInt("num"), 1);
|
||||||
|
// complex content is collected in an 'altContent' array
|
||||||
|
jsonArray = jsonObject.getJSONObject("addresses").
|
||||||
|
getJSONObject("address").getJSONArray("altContent");
|
||||||
|
expectedJsonArray = new JSONArray(expectedStr);
|
||||||
|
Util.compareActualVsExpectedJsonArrays(jsonArray, expectedJsonArray);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convenience method, given an input string and expected result,
|
||||||
|
* convert to JSONObject and compare actual to expected result.
|
||||||
|
* @param xmlStr the string to parse
|
||||||
|
* @param expectedStr the expected JSON string
|
||||||
|
* @param config provides more flexible XML parsing
|
||||||
|
* flexible XML parsing.
|
||||||
|
*/
|
||||||
|
private void compareStringToJSONObject(String xmlStr, String expectedStr,
|
||||||
|
XMLParserConfiguration config) {
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr, config);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convenience method, given an input string and expected result,
|
||||||
|
* convert to JSONObject via reader and compare actual to expected result.
|
||||||
|
* @param xmlStr the string to parse
|
||||||
|
* @param expectedStr the expected JSON string
|
||||||
|
* @param config provides more flexible XML parsing
|
||||||
|
*/
|
||||||
|
private void compareReaderToJSONObject(String xmlStr, String expectedStr,
|
||||||
|
XMLParserConfiguration config) {
|
||||||
|
/*
|
||||||
|
* Commenting out this method until the JSON-java code is updated
|
||||||
|
* to support XML.toJSONObject(reader)
|
||||||
|
*/
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
try(Reader reader = new StringReader(xmlStr);) {
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(reader, config);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
} catch (IOException e) {
|
||||||
|
assertTrue("IO Reader error: " +e.getMessage(), false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convenience method, given an input string and expected result, convert to
|
||||||
|
* JSONObject via file and compare actual to expected result.
|
||||||
|
*
|
||||||
|
* @param xmlStr
|
||||||
|
* the string to parse
|
||||||
|
* @param expectedStr
|
||||||
|
* the expected JSON string
|
||||||
|
* @throws IOException
|
||||||
|
*/
|
||||||
|
private void compareFileToJSONObject(String xmlStr, String expectedStr) {
|
||||||
|
/*
|
||||||
|
* Commenting out this method until the JSON-java code is updated
|
||||||
|
* to support XML.toJSONObject(reader)
|
||||||
|
*/
|
||||||
|
try {
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
File tempFile = this.testFolder.newFile("fileToJSONObject.xml");
|
||||||
|
try(FileWriter fileWriter = new FileWriter(tempFile);){
|
||||||
|
fileWriter.write(xmlStr);
|
||||||
|
}
|
||||||
|
try(Reader reader = new FileReader(tempFile);){
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(reader);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
assertTrue("file writer error: " +e.getMessage(), false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
901
src/test/java/org/json/junit/XMLTest.java
Normal file
901
src/test/java/org/json/junit/XMLTest.java
Normal file
|
@ -0,0 +1,901 @@
|
||||||
|
package org.json.junit;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright (c) 2020 JSON.org
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
The Software shall be used for Good, not Evil.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
|
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.FileReader;
|
||||||
|
import java.io.FileWriter;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.Reader;
|
||||||
|
import java.io.StringReader;
|
||||||
|
|
||||||
|
import org.json.JSONArray;
|
||||||
|
import org.json.JSONException;
|
||||||
|
import org.json.JSONObject;
|
||||||
|
import org.json.XML;
|
||||||
|
import org.json.XMLParserConfiguration;
|
||||||
|
import org.junit.Rule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.rules.TemporaryFolder;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for JSON-Java XML.java
|
||||||
|
* Note: noSpace() will be tested by JSONMLTest
|
||||||
|
*/
|
||||||
|
public class XMLTest {
|
||||||
|
/**
|
||||||
|
* JUnit supports temporary files and folders that are cleaned up after the test.
|
||||||
|
* https://garygregory.wordpress.com/2010/01/20/junit-tip-use-rules-to-manage-temporary-files-and-folders/
|
||||||
|
*/
|
||||||
|
@Rule
|
||||||
|
public TemporaryFolder testFolder = new TemporaryFolder();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSONObject from a null XML string.
|
||||||
|
* Expects a NullPointerException
|
||||||
|
*/
|
||||||
|
@Test(expected=NullPointerException.class)
|
||||||
|
public void shouldHandleNullXML() {
|
||||||
|
String xmlStr = null;
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr);
|
||||||
|
assertTrue("jsonObject should be empty", jsonObject.isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Empty JSONObject from an empty XML string.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleEmptyXML() {
|
||||||
|
|
||||||
|
String xmlStr = "";
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr);
|
||||||
|
assertTrue("jsonObject should be empty", jsonObject.isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Empty JSONObject from a non-XML string.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNonXML() {
|
||||||
|
String xmlStr = "{ \"this is\": \"not xml\"}";
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr);
|
||||||
|
assertTrue("xml string should be empty", jsonObject.isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Invalid XML string (tag contains a frontslash).
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleInvalidSlashInTag() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/x>\n"+
|
||||||
|
" <street>abc street</street>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
XML.toJSONObject(xmlStr);
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Misshaped tag at 176 [character 14 line 4]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Invalid XML string ('!' char in tag)
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleInvalidBangInTag() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/>\n"+
|
||||||
|
" <!>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
XML.toJSONObject(xmlStr);
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Misshaped meta tag at 214 [character 12 line 7]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Invalid XML string ('!' char and no closing tag brace)
|
||||||
|
* Expects a JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleInvalidBangNoCloseInTag() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/>\n"+
|
||||||
|
" <!\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
XML.toJSONObject(xmlStr);
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Misshaped meta tag at 213 [character 12 line 7]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Invalid XML string (no end brace for tag)
|
||||||
|
* Expects JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNoCloseStartTag() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/>\n"+
|
||||||
|
" <abc\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
XML.toJSONObject(xmlStr);
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Misplaced '<' at 193 [character 4 line 6]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Invalid XML string (partial CDATA chars in tag name)
|
||||||
|
* Expects JSONException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleInvalidCDATABangInTag() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name>Joe Tester</name>\n"+
|
||||||
|
" <![[]>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
try {
|
||||||
|
XML.toJSONObject(xmlStr);
|
||||||
|
fail("Expecting a JSONException");
|
||||||
|
} catch (JSONException e) {
|
||||||
|
assertEquals("Expecting an exception message",
|
||||||
|
"Expected 'CDATA[' at 204 [character 11 line 5]",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Null JSONObject in XML.toString()
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNullJSONXML() {
|
||||||
|
JSONObject jsonObject= null;
|
||||||
|
String actualXml=XML.toString(jsonObject);
|
||||||
|
assertEquals("generated XML does not equal expected XML","\"null\"",actualXml);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Empty JSONObject in XML.toString()
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleEmptyJSONXML() {
|
||||||
|
JSONObject jsonObject= new JSONObject();
|
||||||
|
String xmlStr = XML.toString(jsonObject);
|
||||||
|
assertTrue("xml string should be empty", xmlStr.isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* No SML start tag. The ending tag ends up being treated as content.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNoStartTag() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name/>\n"+
|
||||||
|
" <nocontent/>>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":{\"address\":{\"name\":\"\",\"nocontent\":\"\",\""+
|
||||||
|
"content\":\">\"},\"xsi:noNamespaceSchemaLocation\":\"test.xsd\",\""+
|
||||||
|
"xmlns:xsi\":\"http://www.w3.org/2001/XMLSchema-instance\"}}";
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Valid XML to JSONObject
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleSimpleXML() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name>Joe Tester</name>\n"+
|
||||||
|
" <street>[CDATA[Baker street 5]</street>\n"+
|
||||||
|
" <NothingHere/>\n"+
|
||||||
|
" <TrueValue>true</TrueValue>\n"+
|
||||||
|
" <FalseValue>false</FalseValue>\n"+
|
||||||
|
" <NullValue>null</NullValue>\n"+
|
||||||
|
" <PositiveValue>42</PositiveValue>\n"+
|
||||||
|
" <NegativeValue>-23</NegativeValue>\n"+
|
||||||
|
" <DoubleValue>-23.45</DoubleValue>\n"+
|
||||||
|
" <Nan>-23x.45</Nan>\n"+
|
||||||
|
" <ArrayOfNum>1, 2, 3, 4.1, 5.2</ArrayOfNum>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":{\"address\":{\"street\":\"[CDATA[Baker street 5]\","+
|
||||||
|
"\"name\":\"Joe Tester\",\"NothingHere\":\"\",TrueValue:true,\n"+
|
||||||
|
"\"FalseValue\":false,\"NullValue\":null,\"PositiveValue\":42,\n"+
|
||||||
|
"\"NegativeValue\":-23,\"DoubleValue\":-23.45,\"Nan\":-23x.45,\n"+
|
||||||
|
"\"ArrayOfNum\":\"1, 2, 3, 4.1, 5.2\"\n"+
|
||||||
|
"},\"xsi:noNamespaceSchemaLocation\":"+
|
||||||
|
"\"test.xsd\",\"xmlns:xsi\":\"http://www.w3.org/2001/"+
|
||||||
|
"XMLSchema-instance\"}}";
|
||||||
|
|
||||||
|
compareStringToJSONObject(xmlStr, expectedStr);
|
||||||
|
compareReaderToJSONObject(xmlStr, expectedStr);
|
||||||
|
compareFileToJSONObject(xmlStr, expectedStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests to verify that supported escapes in XML are converted to actual values.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testXmlEscapeToJson(){
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<root>"+
|
||||||
|
"<rawQuote>\"</rawQuote>"+
|
||||||
|
"<euro>A €33</euro>"+
|
||||||
|
"<euroX>A €22€</euroX>"+
|
||||||
|
"<unknown>some text ©</unknown>"+
|
||||||
|
"<known>" " & ' < ></known>"+
|
||||||
|
"<high>𝄢 𐅥</high>" +
|
||||||
|
"</root>";
|
||||||
|
String expectedStr =
|
||||||
|
"{\"root\":{" +
|
||||||
|
"\"rawQuote\":\"\\\"\"," +
|
||||||
|
"\"euro\":\"A €33\"," +
|
||||||
|
"\"euroX\":\"A €22€\"," +
|
||||||
|
"\"unknown\":\"some text ©\"," +
|
||||||
|
"\"known\":\"\\\" \\\" & ' < >\"," +
|
||||||
|
"\"high\":\"𝄢 𐅥\""+
|
||||||
|
"}}";
|
||||||
|
|
||||||
|
compareStringToJSONObject(xmlStr, expectedStr);
|
||||||
|
compareReaderToJSONObject(xmlStr, expectedStr);
|
||||||
|
compareFileToJSONObject(xmlStr, expectedStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests that control characters are escaped.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testJsonToXmlEscape(){
|
||||||
|
final String jsonSrc = "{\"amount\":\"10,00 €\","
|
||||||
|
+ "\"description\":\"Ação Válida\u0085\","
|
||||||
|
+ "\"xmlEntities\":\"\\\" ' & < >\""
|
||||||
|
+ "}";
|
||||||
|
JSONObject json = new JSONObject(jsonSrc);
|
||||||
|
String xml = XML.toString(json);
|
||||||
|
//test control character not existing
|
||||||
|
assertFalse("Escaping \u0085 failed. Found in XML output.", xml.contains("\u0085"));
|
||||||
|
assertTrue("Escaping \u0085 failed. Entity not found in XML output.", xml.contains("…"));
|
||||||
|
// test normal unicode existing
|
||||||
|
assertTrue("Escaping € failed. Not found in XML output.", xml.contains("€"));
|
||||||
|
assertTrue("Escaping ç failed. Not found in XML output.", xml.contains("ç"));
|
||||||
|
assertTrue("Escaping ã failed. Not found in XML output.", xml.contains("ã"));
|
||||||
|
assertTrue("Escaping á failed. Not found in XML output.", xml.contains("á"));
|
||||||
|
// test XML Entities converted
|
||||||
|
assertTrue("Escaping \" failed. Not found in XML output.", xml.contains("""));
|
||||||
|
assertTrue("Escaping ' failed. Not found in XML output.", xml.contains("'"));
|
||||||
|
assertTrue("Escaping & failed. Not found in XML output.", xml.contains("&"));
|
||||||
|
assertTrue("Escaping < failed. Not found in XML output.", xml.contains("<"));
|
||||||
|
assertTrue("Escaping > failed. Not found in XML output.", xml.contains(">"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Valid XML with comments to JSONObject
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleCommentsInXML() {
|
||||||
|
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<!-- this is a comment -->\n"+
|
||||||
|
"<addresses>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <![CDATA[ this is -- <another> comment ]]>\n"+
|
||||||
|
" <name>Joe Tester</name>\n"+
|
||||||
|
" <!-- this is a - multi line \n"+
|
||||||
|
" comment -->\n"+
|
||||||
|
" <street>Baker street 5</street>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr);
|
||||||
|
String expectedStr = "{\"addresses\":{\"address\":{\"street\":\"Baker "+
|
||||||
|
"street 5\",\"name\":\"Joe Tester\",\"content\":\" this is -- "+
|
||||||
|
"<another> comment \"}}}";
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Valid XML to XML.toString()
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleToString() {
|
||||||
|
String xmlStr =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"+
|
||||||
|
"<addresses xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""+
|
||||||
|
" xsi:noNamespaceSchemaLocation='test.xsd'>\n"+
|
||||||
|
" <address>\n"+
|
||||||
|
" <name>[CDATA[Joe & T > e < s " t ' er]]</name>\n"+
|
||||||
|
" <street>Baker street 5</street>\n"+
|
||||||
|
" <ArrayOfNum>1, 2, 3, 4.1, 5.2</ArrayOfNum>\n"+
|
||||||
|
" </address>\n"+
|
||||||
|
"</addresses>";
|
||||||
|
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":{\"address\":{\"street\":\"Baker street 5\","+
|
||||||
|
"\"name\":\"[CDATA[Joe & T > e < s \\\" t \\\' er]]\","+
|
||||||
|
"\"ArrayOfNum\":\"1, 2, 3, 4.1, 5.2\"\n"+
|
||||||
|
"},\"xsi:noNamespaceSchemaLocation\":"+
|
||||||
|
"\"test.xsd\",\"xmlns:xsi\":\"http://www.w3.org/2001/"+
|
||||||
|
"XMLSchema-instance\"}}";
|
||||||
|
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr);
|
||||||
|
String xmlToStr = XML.toString(jsonObject);
|
||||||
|
JSONObject finalJsonObject = XML.toJSONObject(xmlToStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(finalJsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converting a JSON doc containing '>' content to JSONObject, then
|
||||||
|
* XML.toString() should result in valid XML.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleContentNoArraytoString() {
|
||||||
|
String expectedStr = "{\"addresses\":{\"content\":\">\"}}";
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
String finalStr = XML.toString(expectedJsonObject);
|
||||||
|
String expectedFinalStr = "<addresses>></addresses>";
|
||||||
|
assertEquals("Should handle expectedFinal: ["+expectedStr+"] final: ["+
|
||||||
|
finalStr+"]", expectedFinalStr, finalStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converting a JSON doc containing a 'content' array to JSONObject, then
|
||||||
|
* XML.toString() should result in valid XML.
|
||||||
|
* TODO: This is probably an error in how the 'content' keyword is used.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleContentArraytoString() {
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":{" +
|
||||||
|
"\"content\":[1, 2, 3]}}";
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
String finalStr = XML.toString(expectedJsonObject);
|
||||||
|
String expectedFinalStr = "<addresses>"+
|
||||||
|
"1\n2\n3</addresses>";
|
||||||
|
assertEquals("Should handle expectedFinal: ["+expectedStr+"] final: ["+
|
||||||
|
finalStr+"]", expectedFinalStr, finalStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converting a JSON doc containing a named array to JSONObject, then
|
||||||
|
* XML.toString() should result in valid XML.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleArraytoString() {
|
||||||
|
String expectedStr =
|
||||||
|
"{\"addresses\":{"+
|
||||||
|
"\"something\":[1, 2, 3]}}";
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
String finalStr = XML.toString(expectedJsonObject);
|
||||||
|
String expectedFinalStr = "<addresses>"+
|
||||||
|
"<something>1</something><something>2</something><something>3</something>"+
|
||||||
|
"</addresses>";
|
||||||
|
assertEquals("Should handle expectedFinal: ["+expectedStr+"] final: ["+
|
||||||
|
finalStr+"]", expectedFinalStr, finalStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests that the XML output for empty arrays is consistent.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleEmptyArray(){
|
||||||
|
final JSONObject jo1 = new JSONObject();
|
||||||
|
jo1.put("array",new Object[]{});
|
||||||
|
final JSONObject jo2 = new JSONObject();
|
||||||
|
jo2.put("array",new JSONArray());
|
||||||
|
|
||||||
|
final String expected = "<jo></jo>";
|
||||||
|
String output1 = XML.toString(jo1,"jo");
|
||||||
|
assertEquals("Expected an empty root tag", expected, output1);
|
||||||
|
String output2 = XML.toString(jo2,"jo");
|
||||||
|
assertEquals("Expected an empty root tag", expected, output2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests that the XML output for arrays is consistent when an internal array is empty.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleEmptyMultiArray(){
|
||||||
|
final JSONObject jo1 = new JSONObject();
|
||||||
|
jo1.put("arr",new Object[]{"One", new String[]{}, "Four"});
|
||||||
|
final JSONObject jo2 = new JSONObject();
|
||||||
|
jo2.put("arr",new JSONArray(new Object[]{"One", new JSONArray(new String[]{}), "Four"}));
|
||||||
|
|
||||||
|
final String expected = "<jo><arr>One</arr><arr></arr><arr>Four</arr></jo>";
|
||||||
|
String output1 = XML.toString(jo1,"jo");
|
||||||
|
assertEquals("Expected a matching array", expected, output1);
|
||||||
|
String output2 = XML.toString(jo2,"jo");
|
||||||
|
assertEquals("Expected a matching array", expected, output2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests that the XML output for arrays is consistent when arrays are not empty.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNonEmptyArray(){
|
||||||
|
final JSONObject jo1 = new JSONObject();
|
||||||
|
jo1.put("arr",new String[]{"One", "Two", "Three"});
|
||||||
|
final JSONObject jo2 = new JSONObject();
|
||||||
|
jo2.put("arr",new JSONArray(new String[]{"One", "Two", "Three"}));
|
||||||
|
|
||||||
|
final String expected = "<jo><arr>One</arr><arr>Two</arr><arr>Three</arr></jo>";
|
||||||
|
String output1 = XML.toString(jo1,"jo");
|
||||||
|
assertEquals("Expected a non empty root tag", expected, output1);
|
||||||
|
String output2 = XML.toString(jo2,"jo");
|
||||||
|
assertEquals("Expected a non empty root tag", expected, output2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests that the XML output for arrays is consistent when arrays are not empty and contain internal arrays.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleMultiArray(){
|
||||||
|
final JSONObject jo1 = new JSONObject();
|
||||||
|
jo1.put("arr",new Object[]{"One", new String[]{"Two", "Three"}, "Four"});
|
||||||
|
final JSONObject jo2 = new JSONObject();
|
||||||
|
jo2.put("arr",new JSONArray(new Object[]{"One", new JSONArray(new String[]{"Two", "Three"}), "Four"}));
|
||||||
|
|
||||||
|
final String expected = "<jo><arr>One</arr><arr><array>Two</array><array>Three</array></arr><arr>Four</arr></jo>";
|
||||||
|
String output1 = XML.toString(jo1,"jo");
|
||||||
|
assertEquals("Expected a matching array", expected, output1);
|
||||||
|
String output2 = XML.toString(jo2,"jo");
|
||||||
|
assertEquals("Expected a matching array", expected, output2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converting a JSON doc containing a named array of nested arrays to
|
||||||
|
* JSONObject, then XML.toString() should result in valid XML.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNestedArraytoString() {
|
||||||
|
String xmlStr =
|
||||||
|
"{\"addresses\":{\"address\":{\"name\":\"\",\"nocontent\":\"\","+
|
||||||
|
"\"outer\":[[1], [2], [3]]},\"xsi:noNamespaceSchemaLocation\":\"test.xsd\",\""+
|
||||||
|
"xmlns:xsi\":\"http://www.w3.org/2001/XMLSchema-instance\"}}";
|
||||||
|
JSONObject jsonObject = new JSONObject(xmlStr);
|
||||||
|
String finalStr = XML.toString(jsonObject);
|
||||||
|
JSONObject finalJsonObject = XML.toJSONObject(finalStr);
|
||||||
|
String expectedStr = "<addresses><address><name/><nocontent/>"+
|
||||||
|
"<outer><array>1</array></outer><outer><array>2</array>"+
|
||||||
|
"</outer><outer><array>3</array></outer>"+
|
||||||
|
"</address><xsi:noNamespaceSchemaLocation>test.xsd</xsi:noName"+
|
||||||
|
"spaceSchemaLocation><xmlns:xsi>http://www.w3.org/2001/XMLSche"+
|
||||||
|
"ma-instance</xmlns:xsi></addresses>";
|
||||||
|
JSONObject expectedJsonObject = XML.toJSONObject(expectedStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(finalJsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Possible bug:
|
||||||
|
* Illegal node-names must be converted to legal XML-node-names.
|
||||||
|
* The given example shows 2 nodes which are valid for JSON, but not for XML.
|
||||||
|
* Therefore illegal arguments should be converted to e.g. an underscore (_).
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleIllegalJSONNodeNames()
|
||||||
|
{
|
||||||
|
JSONObject inputJSON = new JSONObject();
|
||||||
|
inputJSON.append("123IllegalNode", "someValue1");
|
||||||
|
inputJSON.append("Illegal@node", "someValue2");
|
||||||
|
|
||||||
|
String result = XML.toString(inputJSON);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This is invalid XML. Names should not begin with digits or contain
|
||||||
|
* certain values, including '@'. One possible solution is to replace
|
||||||
|
* illegal chars with '_', in which case the expected output would be:
|
||||||
|
* <___IllegalNode>someValue1</___IllegalNode><Illegal_node>someValue2</Illegal_node>
|
||||||
|
*/
|
||||||
|
String expected = "<123IllegalNode>someValue1</123IllegalNode><Illegal@node>someValue2</Illegal@node>";
|
||||||
|
|
||||||
|
assertEquals("length",expected.length(), result.length());
|
||||||
|
assertTrue("123IllegalNode",result.contains("<123IllegalNode>someValue1</123IllegalNode>"));
|
||||||
|
assertTrue("Illegal@node",result.contains("<Illegal@node>someValue2</Illegal@node>"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSONObject with NULL value, to XML.toString()
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldHandleNullNodeValue()
|
||||||
|
{
|
||||||
|
JSONObject inputJSON = new JSONObject();
|
||||||
|
inputJSON.put("nullValue", JSONObject.NULL);
|
||||||
|
// This is a possible preferred result
|
||||||
|
// String expectedXML = "<nullValue/>";
|
||||||
|
/**
|
||||||
|
* This is the current behavior. JSONObject.NULL is emitted as
|
||||||
|
* the string, "null".
|
||||||
|
*/
|
||||||
|
String actualXML = "<nullValue>null</nullValue>";
|
||||||
|
String resultXML = XML.toString(inputJSON);
|
||||||
|
assertEquals(actualXML, resultXML);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Investigate exactly how the "content" keyword works
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void contentOperations() {
|
||||||
|
/*
|
||||||
|
* When a standalone <!CDATA[...]] structure is found while parsing XML into a
|
||||||
|
* JSONObject, the contents are placed in a string value with key="content".
|
||||||
|
*/
|
||||||
|
String xmlStr = "<tag1></tag1><![CDATA[if (a < b && a > 0) then return]]><tag2></tag2>";
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr);
|
||||||
|
assertTrue("1. 3 items", 3 == jsonObject.length());
|
||||||
|
assertTrue("1. empty tag1", "".equals(jsonObject.get("tag1")));
|
||||||
|
assertTrue("1. empty tag2", "".equals(jsonObject.get("tag2")));
|
||||||
|
assertTrue("1. content found", "if (a < b && a > 0) then return".equals(jsonObject.get("content")));
|
||||||
|
|
||||||
|
// multiple consecutive standalone cdatas are accumulated into an array
|
||||||
|
xmlStr = "<tag1></tag1><![CDATA[if (a < b && a > 0) then return]]><tag2></tag2><![CDATA[here is another cdata]]>";
|
||||||
|
jsonObject = XML.toJSONObject(xmlStr);
|
||||||
|
assertTrue("2. 3 items", 3 == jsonObject.length());
|
||||||
|
assertTrue("2. empty tag1", "".equals(jsonObject.get("tag1")));
|
||||||
|
assertTrue("2. empty tag2", "".equals(jsonObject.get("tag2")));
|
||||||
|
assertTrue("2. content array found", jsonObject.get("content") instanceof JSONArray);
|
||||||
|
JSONArray jsonArray = jsonObject.getJSONArray("content");
|
||||||
|
assertTrue("2. array size", jsonArray.length() == 2);
|
||||||
|
assertTrue("2. content array entry 0", "if (a < b && a > 0) then return".equals(jsonArray.get(0)));
|
||||||
|
assertTrue("2. content array entry 1", "here is another cdata".equals(jsonArray.get(1)));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* text content is accumulated in a "content" inside a local JSONObject.
|
||||||
|
* If there is only one instance, it is saved in the context (a different JSONObject
|
||||||
|
* from the calling code. and the content element is discarded.
|
||||||
|
*/
|
||||||
|
xmlStr = "<tag1>value 1</tag1>";
|
||||||
|
jsonObject = XML.toJSONObject(xmlStr);
|
||||||
|
assertTrue("3. 2 items", 1 == jsonObject.length());
|
||||||
|
assertTrue("3. value tag1", "value 1".equals(jsonObject.get("tag1")));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* array-style text content (multiple tags with the same name) is
|
||||||
|
* accumulated in a local JSONObject with key="content" and value=JSONArray,
|
||||||
|
* saved in the context, and then the local JSONObject is discarded.
|
||||||
|
*/
|
||||||
|
xmlStr = "<tag1>value 1</tag1><tag1>2</tag1><tag1>true</tag1>";
|
||||||
|
jsonObject = XML.toJSONObject(xmlStr);
|
||||||
|
assertTrue("4. 1 item", 1 == jsonObject.length());
|
||||||
|
assertTrue("4. content array found", jsonObject.get("tag1") instanceof JSONArray);
|
||||||
|
jsonArray = jsonObject.getJSONArray("tag1");
|
||||||
|
assertTrue("4. array size", jsonArray.length() == 3);
|
||||||
|
assertTrue("4. content array entry 0", "value 1".equals(jsonArray.get(0)));
|
||||||
|
assertTrue("4. content array entry 1", jsonArray.getInt(1) == 2);
|
||||||
|
assertTrue("4. content array entry 2", jsonArray.getBoolean(2) == true);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Complex content is accumulated in a "content" field. For example, an element
|
||||||
|
* may contain a mix of child elements and text. Each text segment is
|
||||||
|
* accumulated to content.
|
||||||
|
*/
|
||||||
|
xmlStr = "<tag1>val1<tag2/>val2</tag1>";
|
||||||
|
jsonObject = XML.toJSONObject(xmlStr);
|
||||||
|
assertTrue("5. 1 item", 1 == jsonObject.length());
|
||||||
|
assertTrue("5. jsonObject found", jsonObject.get("tag1") instanceof JSONObject);
|
||||||
|
jsonObject = jsonObject.getJSONObject("tag1");
|
||||||
|
assertTrue("5. 2 contained items", 2 == jsonObject.length());
|
||||||
|
assertTrue("5. contained tag", "".equals(jsonObject.get("tag2")));
|
||||||
|
assertTrue("5. contained content jsonArray found", jsonObject.get("content") instanceof JSONArray);
|
||||||
|
jsonArray = jsonObject.getJSONArray("content");
|
||||||
|
assertTrue("5. array size", jsonArray.length() == 2);
|
||||||
|
assertTrue("5. content array entry 0", "val1".equals(jsonArray.get(0)));
|
||||||
|
assertTrue("5. content array entry 1", "val2".equals(jsonArray.get(1)));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If there is only 1 complex text content, then it is accumulated in a
|
||||||
|
* "content" field as a string.
|
||||||
|
*/
|
||||||
|
xmlStr = "<tag1>val1<tag2/></tag1>";
|
||||||
|
jsonObject = XML.toJSONObject(xmlStr);
|
||||||
|
assertTrue("6. 1 item", 1 == jsonObject.length());
|
||||||
|
assertTrue("6. jsonObject found", jsonObject.get("tag1") instanceof JSONObject);
|
||||||
|
jsonObject = jsonObject.getJSONObject("tag1");
|
||||||
|
assertTrue("6. contained content found", "val1".equals(jsonObject.get("content")));
|
||||||
|
assertTrue("6. contained tag2", "".equals(jsonObject.get("tag2")));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* In this corner case, the content sibling happens to have key=content
|
||||||
|
* We end up with an array within an array, and no content element.
|
||||||
|
* This is probably a bug.
|
||||||
|
*/
|
||||||
|
xmlStr = "<tag1>val1<content/></tag1>";
|
||||||
|
jsonObject = XML.toJSONObject(xmlStr);
|
||||||
|
assertTrue("7. 1 item", 1 == jsonObject.length());
|
||||||
|
assertTrue("7. jsonArray found", jsonObject.get("tag1") instanceof JSONArray);
|
||||||
|
jsonArray = jsonObject.getJSONArray("tag1");
|
||||||
|
assertTrue("array size 1", jsonArray.length() == 1);
|
||||||
|
assertTrue("7. contained array found", jsonArray.get(0) instanceof JSONArray);
|
||||||
|
jsonArray = jsonArray.getJSONArray(0);
|
||||||
|
assertTrue("7. inner array size 2", jsonArray.length() == 2);
|
||||||
|
assertTrue("7. inner array item 0", "val1".equals(jsonArray.get(0)));
|
||||||
|
assertTrue("7. inner array item 1", "".equals(jsonArray.get(1)));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Confirm behavior of original issue
|
||||||
|
*/
|
||||||
|
String jsonStr =
|
||||||
|
"{"+
|
||||||
|
"\"Profile\": {"+
|
||||||
|
"\"list\": {"+
|
||||||
|
"\"history\": {"+
|
||||||
|
"\"entries\": ["+
|
||||||
|
"{"+
|
||||||
|
"\"deviceId\": \"id\","+
|
||||||
|
"\"content\": {"+
|
||||||
|
"\"material\": ["+
|
||||||
|
"{"+
|
||||||
|
"\"stuff\": false"+
|
||||||
|
"}"+
|
||||||
|
"]"+
|
||||||
|
"}"+
|
||||||
|
"}"+
|
||||||
|
"]"+
|
||||||
|
"}"+
|
||||||
|
"}"+
|
||||||
|
"}"+
|
||||||
|
"}";
|
||||||
|
jsonObject = new JSONObject(jsonStr);
|
||||||
|
xmlStr = XML.toString(jsonObject);
|
||||||
|
/*
|
||||||
|
* This is the created XML. Looks like content was mistaken for
|
||||||
|
* complex (child node + text) XML.
|
||||||
|
* <Profile>
|
||||||
|
* <list>
|
||||||
|
* <history>
|
||||||
|
* <entries>
|
||||||
|
* <deviceId>id</deviceId>
|
||||||
|
* {"material":[{"stuff":false}]}
|
||||||
|
* </entries>
|
||||||
|
* </history>
|
||||||
|
* </list>
|
||||||
|
* </Profile>
|
||||||
|
*/
|
||||||
|
assertTrue("nothing to test here, see comment on created XML, above", true);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convenience method, given an input string and expected result,
|
||||||
|
* convert to JSONObject and compare actual to expected result.
|
||||||
|
* @param xmlStr the string to parse
|
||||||
|
* @param expectedStr the expected JSON string
|
||||||
|
*/
|
||||||
|
private void compareStringToJSONObject(String xmlStr, String expectedStr) {
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(xmlStr);
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convenience method, given an input string and expected result,
|
||||||
|
* convert to JSONObject via reader and compare actual to expected result.
|
||||||
|
* @param xmlStr the string to parse
|
||||||
|
* @param expectedStr the expected JSON string
|
||||||
|
*/
|
||||||
|
private void compareReaderToJSONObject(String xmlStr, String expectedStr) {
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
Reader reader = new StringReader(xmlStr);
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(reader);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convenience method, given an input string and expected result, convert to
|
||||||
|
* JSONObject via file and compare actual to expected result.
|
||||||
|
*
|
||||||
|
* @param xmlStr
|
||||||
|
* the string to parse
|
||||||
|
* @param expectedStr
|
||||||
|
* the expected JSON string
|
||||||
|
* @throws IOException
|
||||||
|
*/
|
||||||
|
private void compareFileToJSONObject(String xmlStr, String expectedStr) {
|
||||||
|
try {
|
||||||
|
JSONObject expectedJsonObject = new JSONObject(expectedStr);
|
||||||
|
File tempFile = this.testFolder.newFile("fileToJSONObject.xml");
|
||||||
|
try(FileWriter fileWriter = new FileWriter(tempFile);){
|
||||||
|
fileWriter.write(xmlStr);
|
||||||
|
}
|
||||||
|
try(Reader reader = new FileReader(tempFile);){
|
||||||
|
JSONObject jsonObject = XML.toJSONObject(reader);
|
||||||
|
Util.compareActualVsExpectedJsonObjects(jsonObject,expectedJsonObject);
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
fail("file writer error: " +e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSON string lost leading zero and converted "True" to true.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testToJSONArray_jsonOutput() {
|
||||||
|
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
||||||
|
final JSONObject expectedJson = new JSONObject("{\"root\":{\"item\":{\"id\":\"01\"},\"id\":[\"01\",1,\"00\",0],\"title\":true}}");
|
||||||
|
final JSONObject actualJsonOutput = XML.toJSONObject(originalXml, false);
|
||||||
|
|
||||||
|
Util.compareActualVsExpectedJsonObjects(actualJsonOutput,expectedJson);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSON string cannot be reverted to original xml.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testToJSONArray_reversibility() {
|
||||||
|
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
||||||
|
final String revertedXml = XML.toString(XML.toJSONObject(originalXml, false));
|
||||||
|
|
||||||
|
assertNotEquals(revertedXml, originalXml);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test passes when using the new method toJsonArray.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testToJsonXML() {
|
||||||
|
final String originalXml = "<root><id>01</id><id>1</id><id>00</id><id>0</id><item id=\"01\"/><title>True</title></root>";
|
||||||
|
final JSONObject expected = new JSONObject("{\"root\":{\"item\":{\"id\":\"01\"},\"id\":[\"01\",\"1\",\"00\",\"0\"],\"title\":\"True\"}}");
|
||||||
|
|
||||||
|
final JSONObject actual = XML.toJSONObject(originalXml,true);
|
||||||
|
|
||||||
|
Util.compareActualVsExpectedJsonObjects(actual, expected);
|
||||||
|
|
||||||
|
final String reverseXml = XML.toString(actual);
|
||||||
|
// this reversal isn't exactly the same. use JSONML for an exact reversal
|
||||||
|
// the order of the elements may be differnet as well.
|
||||||
|
final String expectedReverseXml = "<root><item><id>01</id></item><id>01</id><id>1</id><id>00</id><id>0</id><title>True</title></root>";
|
||||||
|
|
||||||
|
assertEquals("length",expectedReverseXml.length(), reverseXml.length());
|
||||||
|
assertTrue("array contents", reverseXml.contains("<id>01</id><id>1</id><id>00</id><id>0</id>"));
|
||||||
|
assertTrue("item contents", reverseXml.contains("<item><id>01</id></item>"));
|
||||||
|
assertTrue("title contents", reverseXml.contains("<title>True</title>"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test to validate certain conditions of XML unescaping.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testUnescape() {
|
||||||
|
assertEquals("{\"xml\":\"Can cope <;\"}",
|
||||||
|
XML.toJSONObject("<xml>Can cope <; </xml>").toString());
|
||||||
|
assertEquals("Can cope <; ", XML.unescape("Can cope <; "));
|
||||||
|
|
||||||
|
assertEquals("{\"xml\":\"Can cope & ;\"}",
|
||||||
|
XML.toJSONObject("<xml>Can cope & ; </xml>").toString());
|
||||||
|
assertEquals("Can cope & ; ", XML.unescape("Can cope & ; "));
|
||||||
|
|
||||||
|
assertEquals("{\"xml\":\"Can cope &;\"}",
|
||||||
|
XML.toJSONObject("<xml>Can cope &; </xml>").toString());
|
||||||
|
assertEquals("Can cope &; ", XML.unescape("Can cope &; "));
|
||||||
|
|
||||||
|
// unicode entity
|
||||||
|
assertEquals("{\"xml\":\"Can cope 4;\"}",
|
||||||
|
XML.toJSONObject("<xml>Can cope 4; </xml>").toString());
|
||||||
|
assertEquals("Can cope 4; ", XML.unescape("Can cope 4; "));
|
||||||
|
|
||||||
|
// double escaped
|
||||||
|
assertEquals("{\"xml\":\"Can cope <\"}",
|
||||||
|
XML.toJSONObject("<xml>Can cope &lt; </xml>").toString());
|
||||||
|
assertEquals("Can cope < ", XML.unescape("Can cope &lt; "));
|
||||||
|
|
||||||
|
assertEquals("{\"xml\":\"Can cope 4\"}",
|
||||||
|
XML.toJSONObject("<xml>Can cope &#x34; </xml>").toString());
|
||||||
|
assertEquals("Can cope 4 ", XML.unescape("Can cope &#x34; "));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test passes when xsi:nil="true" converting to null (JSON specification-like nil conversion enabled)
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testToJsonWithNullWhenNilConversionEnabled() {
|
||||||
|
final String originalXml = "<root><id xsi:nil=\"true\"/></root>";
|
||||||
|
final String expectedJsonString = "{\"root\":{\"id\":null}}";
|
||||||
|
|
||||||
|
final JSONObject json = XML.toJSONObject(originalXml, new XMLParserConfiguration(false, "content", true));
|
||||||
|
assertEquals(expectedJsonString, json.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test passes when xsi:nil="true" not converting to null (JSON specification-like nil conversion disabled)
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testToJsonWithNullWhenNilConversionDisabled() {
|
||||||
|
final String originalXml = "<root><id xsi:nil=\"true\"/></root>";
|
||||||
|
final String expectedJsonString = "{\"root\":{\"id\":{\"xsi:nil\":true}}}";
|
||||||
|
|
||||||
|
final JSONObject json = XML.toJSONObject(originalXml, new XMLParserConfiguration());
|
||||||
|
assertEquals(expectedJsonString, json.toString());
|
||||||
|
}
|
||||||
|
}
|
13
src/test/java/org/json/junit/data/BrokenToString.java
Normal file
13
src/test/java/org/json/junit/data/BrokenToString.java
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test class for verifying write errors.
|
||||||
|
* @author John Aylward
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class BrokenToString {
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
throw new IllegalStateException("Something went horribly wrong!");
|
||||||
|
}
|
||||||
|
}
|
66
src/test/java/org/json/junit/data/ExceptionalBean.java
Normal file
66
src/test/java/org/json/junit/data/ExceptionalBean.java
Normal file
|
@ -0,0 +1,66 @@
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
import java.io.Closeable;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.lang.reflect.InvocationTargetException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Object for testing the exception handling in {@link JSONObject#populateMap}.
|
||||||
|
*
|
||||||
|
* @author John Aylward
|
||||||
|
*/
|
||||||
|
public class ExceptionalBean {
|
||||||
|
/**
|
||||||
|
* @return a closeable.
|
||||||
|
*/
|
||||||
|
public Closeable getCloseable() {
|
||||||
|
// anonymous inner class did not work...
|
||||||
|
return new MyCloseable();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return Nothing really. Just can't be void.
|
||||||
|
* @throws IllegalAccessException
|
||||||
|
* always thrown
|
||||||
|
*/
|
||||||
|
public int getIllegalAccessException() throws IllegalAccessException {
|
||||||
|
throw new IllegalAccessException("Yup, it's illegal");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return Nothing really. Just can't be void.
|
||||||
|
* @throws IllegalArgumentException
|
||||||
|
* always thrown
|
||||||
|
*/
|
||||||
|
public int getIllegalArgumentException() throws IllegalArgumentException {
|
||||||
|
throw new IllegalArgumentException("Yup, it's illegal");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return Nothing really. Just can't be void.
|
||||||
|
* @throws InvocationTargetException
|
||||||
|
* always thrown
|
||||||
|
*/
|
||||||
|
public int getInvocationTargetException() throws InvocationTargetException {
|
||||||
|
throw new InvocationTargetException(new Exception("Yup, it's illegal"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/** My closeable class. */
|
||||||
|
public static final class MyCloseable implements Closeable {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return a string
|
||||||
|
*/
|
||||||
|
public String getString() {
|
||||||
|
return "Yup, it's closeable";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void close() throws IOException {
|
||||||
|
throw new IOException("Closing is too hard!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
180
src/test/java/org/json/junit/data/Fraction.java
Normal file
180
src/test/java/org/json/junit/data/Fraction.java
Normal file
|
@ -0,0 +1,180 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
import java.math.BigDecimal;
|
||||||
|
import java.math.BigInteger;
|
||||||
|
import java.math.RoundingMode;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* basic fraction class, no frills.
|
||||||
|
* @author John Aylward
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class Fraction extends Number implements Comparable<Fraction> {
|
||||||
|
/**
|
||||||
|
* serial id.
|
||||||
|
*/
|
||||||
|
private static final long serialVersionUID = 1L;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* value as a big decimal.
|
||||||
|
*/
|
||||||
|
private final BigDecimal bigDecimal;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* value of the denominator.
|
||||||
|
*/
|
||||||
|
private final BigInteger denominator;
|
||||||
|
/**
|
||||||
|
* value of the numerator.
|
||||||
|
*/
|
||||||
|
private final BigInteger numerator;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param numerator
|
||||||
|
* numerator
|
||||||
|
* @param denominator
|
||||||
|
* denominator
|
||||||
|
*/
|
||||||
|
public Fraction(final BigInteger numerator, final BigInteger denominator) {
|
||||||
|
super();
|
||||||
|
if (numerator == null || denominator == null) {
|
||||||
|
throw new IllegalArgumentException("All values must be non-null");
|
||||||
|
}
|
||||||
|
if (denominator.compareTo(BigInteger.ZERO)==0) {
|
||||||
|
throw new IllegalArgumentException("Divide by zero");
|
||||||
|
}
|
||||||
|
|
||||||
|
final BigInteger n;
|
||||||
|
final BigInteger d;
|
||||||
|
// normalize fraction
|
||||||
|
if (denominator.signum()<0) {
|
||||||
|
n = numerator.negate();
|
||||||
|
d = denominator.negate();
|
||||||
|
} else {
|
||||||
|
n = numerator;
|
||||||
|
d = denominator;
|
||||||
|
}
|
||||||
|
this.numerator = n;
|
||||||
|
this.denominator = d;
|
||||||
|
if (n.compareTo(BigInteger.ZERO)==0) {
|
||||||
|
this.bigDecimal = BigDecimal.ZERO;
|
||||||
|
} else if (n.compareTo(d)==0) {// i.e. 4/4, 10/10
|
||||||
|
this.bigDecimal = BigDecimal.ONE;
|
||||||
|
} else {
|
||||||
|
this.bigDecimal = new BigDecimal(this.numerator).divide(new BigDecimal(this.denominator),
|
||||||
|
RoundingMode.HALF_EVEN);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param numerator
|
||||||
|
* numerator
|
||||||
|
* @param denominator
|
||||||
|
* denominator
|
||||||
|
*/
|
||||||
|
public Fraction(final long numerator, final long denominator) {
|
||||||
|
this(BigInteger.valueOf(numerator),BigInteger.valueOf(denominator));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the decimal
|
||||||
|
*/
|
||||||
|
public BigDecimal bigDecimalValue() {
|
||||||
|
return this.bigDecimal;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int compareTo(final Fraction o) {
|
||||||
|
// .equals call this, so no .equals compare allowed
|
||||||
|
|
||||||
|
// if they are the same reference, just return equals
|
||||||
|
if (this == o) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// if my denominators are already equal, just compare the numerators
|
||||||
|
if (this.denominator.compareTo(o.denominator)==0) {
|
||||||
|
return this.numerator.compareTo(o.numerator);
|
||||||
|
}
|
||||||
|
|
||||||
|
// get numerators of common denominators
|
||||||
|
// a x ay xb
|
||||||
|
// --- --- = ---- ----
|
||||||
|
// b y by yb
|
||||||
|
final BigInteger thisN = this.numerator.multiply(o.denominator);
|
||||||
|
final BigInteger otherN = o.numerator.multiply(this.denominator);
|
||||||
|
|
||||||
|
return thisN.compareTo(otherN);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public double doubleValue() {
|
||||||
|
return this.bigDecimal.doubleValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see java.lang.Object#equals(java.lang.Object)
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public boolean equals(final Object obj) {
|
||||||
|
if (this == obj) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (obj == null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (this.getClass() != obj.getClass()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
final Fraction other = (Fraction) obj;
|
||||||
|
return this.compareTo(other) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public float floatValue() {
|
||||||
|
return this.bigDecimal.floatValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the denominator
|
||||||
|
*/
|
||||||
|
public BigInteger getDenominator() {
|
||||||
|
return this.denominator;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the numerator
|
||||||
|
*/
|
||||||
|
public BigInteger getNumerator() {
|
||||||
|
return this.numerator;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see java.lang.Object#hashCode()
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
final int prime = 31;
|
||||||
|
int result = 1;
|
||||||
|
result = prime * result + (this.bigDecimal == null ? 0 : this.bigDecimal.hashCode());
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int intValue() {
|
||||||
|
return this.bigDecimal.intValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public long longValue() {
|
||||||
|
return this.bigDecimal.longValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see java.lang.Object#toString()
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return this.numerator + "/" + this.denominator;
|
||||||
|
}
|
||||||
|
}
|
79
src/test/java/org/json/junit/data/GenericBean.java
Normal file
79
src/test/java/org/json/junit/data/GenericBean.java
Normal file
|
@ -0,0 +1,79 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
import java.io.StringReader;
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author John Aylward
|
||||||
|
*
|
||||||
|
* @param <T>
|
||||||
|
* generic number value
|
||||||
|
*/
|
||||||
|
public class GenericBean<T extends Number & Comparable<T>> implements MyBean {
|
||||||
|
/**
|
||||||
|
* @param genericValue
|
||||||
|
* value to initiate with
|
||||||
|
*/
|
||||||
|
public GenericBean(T genericValue) {
|
||||||
|
super();
|
||||||
|
this.genericValue = genericValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** */
|
||||||
|
protected T genericValue;
|
||||||
|
/** to be used by the calling test to see how often the getter is called */
|
||||||
|
public int genericGetCounter;
|
||||||
|
/** to be used by the calling test to see how often the setter is called */
|
||||||
|
public int genericSetCounter;
|
||||||
|
|
||||||
|
/** @return the genericValue */
|
||||||
|
public T getGenericValue() {
|
||||||
|
this.genericGetCounter++;
|
||||||
|
return this.genericValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param genericValue
|
||||||
|
* generic value to set
|
||||||
|
*/
|
||||||
|
public void setGenericValue(T genericValue) {
|
||||||
|
this.genericSetCounter++;
|
||||||
|
this.genericValue = genericValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Integer getIntKey() {
|
||||||
|
return Integer.valueOf(42);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Double getDoubleKey() {
|
||||||
|
return Double.valueOf(4.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getStringKey() {
|
||||||
|
return "MyString Key";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getEscapeStringKey() {
|
||||||
|
return "\"My String with \"s";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Boolean isTrueKey() {
|
||||||
|
return Boolean.TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Boolean isFalseKey() {
|
||||||
|
return Boolean.FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public StringReader getStringReaderKey() {
|
||||||
|
return new StringReader("Some String Value in a reader");
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
69
src/test/java/org/json/junit/data/GenericBeanInt.java
Normal file
69
src/test/java/org/json/junit/data/GenericBeanInt.java
Normal file
|
@ -0,0 +1,69 @@
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author john
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class GenericBeanInt extends GenericBean<Integer> {
|
||||||
|
/** */
|
||||||
|
final char a = 'A';
|
||||||
|
|
||||||
|
/** @return the a */
|
||||||
|
public char getA() {
|
||||||
|
return this.a;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Should not be beanable
|
||||||
|
*
|
||||||
|
* @return false
|
||||||
|
*/
|
||||||
|
public boolean getable() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Should not be beanable
|
||||||
|
*
|
||||||
|
* @return false
|
||||||
|
*/
|
||||||
|
public boolean get() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Should not be beanable
|
||||||
|
*
|
||||||
|
* @return false
|
||||||
|
*/
|
||||||
|
public boolean is() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Should be beanable
|
||||||
|
*
|
||||||
|
* @return false
|
||||||
|
*/
|
||||||
|
public boolean isB() {
|
||||||
|
return this.genericValue.equals((Integer.valueOf(this.a+1)));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param genericValue
|
||||||
|
* the value to initiate with.
|
||||||
|
*/
|
||||||
|
public GenericBeanInt(Integer genericValue) {
|
||||||
|
super(genericValue);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** override to generate a bridge method */
|
||||||
|
@Override
|
||||||
|
public Integer getGenericValue() {
|
||||||
|
return super.getGenericValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
16
src/test/java/org/json/junit/data/MyBean.java
Normal file
16
src/test/java/org/json/junit/data/MyBean.java
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
import java.io.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Used in testing when Bean behavior is needed
|
||||||
|
*/
|
||||||
|
public interface MyBean {
|
||||||
|
public Integer getIntKey();
|
||||||
|
public Double getDoubleKey();
|
||||||
|
public String getStringKey();
|
||||||
|
public String getEscapeStringKey();
|
||||||
|
public Boolean isTrueKey();
|
||||||
|
public Boolean isFalseKey();
|
||||||
|
public StringReader getStringReaderKey();
|
||||||
|
}
|
20
src/test/java/org/json/junit/data/MyBeanCustomName.java
Normal file
20
src/test/java/org/json/junit/data/MyBeanCustomName.java
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
import org.json.JSONPropertyName;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test bean for the {@link JSONPropertyName} annotation.
|
||||||
|
*/
|
||||||
|
public class MyBeanCustomName implements MyBeanCustomNameInterface {
|
||||||
|
public int getSomeInt() { return 42; }
|
||||||
|
@JSONPropertyName("")
|
||||||
|
public long getSomeLong() { return 42L; }
|
||||||
|
@JSONPropertyName("myStringField")
|
||||||
|
public String getSomeString() { return "someStringValue"; }
|
||||||
|
@JSONPropertyName("Some Weird NAme that Normally Wouldn't be possible!")
|
||||||
|
public double getMyDouble() { return 0.0d; }
|
||||||
|
@Override
|
||||||
|
public float getSomeFloat() { return 2.0f; }
|
||||||
|
@Override
|
||||||
|
public int getIgnoredInt() { return 40; }
|
||||||
|
}
|
|
@ -0,0 +1,11 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
import org.json.JSONPropertyIgnore;
|
||||||
|
import org.json.JSONPropertyName;
|
||||||
|
|
||||||
|
public interface MyBeanCustomNameInterface {
|
||||||
|
@JSONPropertyName("InterfaceField")
|
||||||
|
float getSomeFloat();
|
||||||
|
@JSONPropertyIgnore
|
||||||
|
int getIgnoredInt();
|
||||||
|
}
|
|
@ -0,0 +1,32 @@
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
import org.json.JSONPropertyIgnore;
|
||||||
|
import org.json.JSONPropertyName;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test bean to verify that the {@link org.json.JSONPropertyName} annotation
|
||||||
|
* is inherited.
|
||||||
|
*/
|
||||||
|
public class MyBeanCustomNameSubClass extends MyBeanCustomName {
|
||||||
|
@Override
|
||||||
|
@JSONPropertyName("forcedInt")
|
||||||
|
public int getIgnoredInt() { return 42*42; }
|
||||||
|
@Override
|
||||||
|
@JSONPropertyName("newIntFieldName")
|
||||||
|
public int getSomeInt() { return 43; }
|
||||||
|
@Override
|
||||||
|
public String getSomeString() { return "subClassString"; }
|
||||||
|
@Override
|
||||||
|
@JSONPropertyName("AMoreNormalName")
|
||||||
|
public double getMyDouble() { return 1.0d; }
|
||||||
|
@Override
|
||||||
|
public float getSomeFloat() { return 3.0f; }
|
||||||
|
@JSONPropertyIgnore
|
||||||
|
@JSONPropertyName("ShouldBeIgnored")
|
||||||
|
public boolean getShouldNotBeJSON() { return true; }
|
||||||
|
@JSONPropertyName("Getable")
|
||||||
|
public boolean getable() { return true; }
|
||||||
|
}
|
11
src/test/java/org/json/junit/data/MyBigNumberBean.java
Normal file
11
src/test/java/org/json/junit/data/MyBigNumberBean.java
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
import java.math.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Used in testing when a Bean containing big numbers is needed
|
||||||
|
*/
|
||||||
|
public interface MyBigNumberBean {
|
||||||
|
public BigInteger getBigInteger();
|
||||||
|
public BigDecimal getBigDecimal();
|
||||||
|
}
|
10
src/test/java/org/json/junit/data/MyEnum.java
Normal file
10
src/test/java/org/json/junit/data/MyEnum.java
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An enum with no methods or data
|
||||||
|
*/
|
||||||
|
public enum MyEnum {
|
||||||
|
VAL1,
|
||||||
|
VAL2,
|
||||||
|
VAL3;
|
||||||
|
}
|
22
src/test/java/org/json/junit/data/MyEnumClass.java
Normal file
22
src/test/java/org/json/junit/data/MyEnumClass.java
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* this is simply a class that contains some enum instances
|
||||||
|
*/
|
||||||
|
public class MyEnumClass {
|
||||||
|
private MyEnum myEnum;
|
||||||
|
private MyEnumField myEnumField;
|
||||||
|
|
||||||
|
public MyEnum getMyEnum() {
|
||||||
|
return this.myEnum;
|
||||||
|
}
|
||||||
|
public void setMyEnum(MyEnum myEnum) {
|
||||||
|
this.myEnum = myEnum;
|
||||||
|
}
|
||||||
|
public MyEnumField getMyEnumField() {
|
||||||
|
return this.myEnumField;
|
||||||
|
}
|
||||||
|
public void setMyEnumField(MyEnumField myEnumField) {
|
||||||
|
this.myEnumField = myEnumField;
|
||||||
|
}
|
||||||
|
}
|
28
src/test/java/org/json/junit/data/MyEnumField.java
Normal file
28
src/test/java/org/json/junit/data/MyEnumField.java
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An enum that contains getters and some internal fields
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("boxing")
|
||||||
|
public enum MyEnumField {
|
||||||
|
VAL1(1, "val 1"),
|
||||||
|
VAL2(2, "val 2"),
|
||||||
|
VAL3(3, "val 3");
|
||||||
|
|
||||||
|
private String value;
|
||||||
|
private Integer intVal;
|
||||||
|
private MyEnumField(Integer intVal, String value) {
|
||||||
|
this.value = value;
|
||||||
|
this.intVal = intVal;
|
||||||
|
}
|
||||||
|
public String getValue() {
|
||||||
|
return this.value;
|
||||||
|
}
|
||||||
|
public Integer getIntVal() {
|
||||||
|
return this.intVal;
|
||||||
|
}
|
||||||
|
@Override
|
||||||
|
public String toString(){
|
||||||
|
return this.value;
|
||||||
|
}
|
||||||
|
}
|
14
src/test/java/org/json/junit/data/MyJsonString.java
Normal file
14
src/test/java/org/json/junit/data/MyJsonString.java
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
import org.json.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Used in testing when a JSONString is needed
|
||||||
|
*/
|
||||||
|
public class MyJsonString implements JSONString {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toJSONString() {
|
||||||
|
return "my string";
|
||||||
|
}
|
||||||
|
}
|
12
src/test/java/org/json/junit/data/MyLocaleBean.java
Executable file
12
src/test/java/org/json/junit/data/MyLocaleBean.java
Executable file
|
@ -0,0 +1,12 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
public class MyLocaleBean {
|
||||||
|
private final String id = "beanId";
|
||||||
|
private final String i = "beanI";
|
||||||
|
public String getId() {
|
||||||
|
return this.id;
|
||||||
|
}
|
||||||
|
public String getI() {
|
||||||
|
return this.i;
|
||||||
|
}
|
||||||
|
}
|
97
src/test/java/org/json/junit/data/MyNumber.java
Normal file
97
src/test/java/org/json/junit/data/MyNumber.java
Normal file
|
@ -0,0 +1,97 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
import java.math.BigDecimal;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Number override for testing. Number overrides should always override
|
||||||
|
* toString, hashCode, and Equals.
|
||||||
|
*
|
||||||
|
* @see <a
|
||||||
|
* href="https://docs.oracle.com/javase/tutorial/java/data/numberclasses.html">The
|
||||||
|
* Numbers Classes</a>
|
||||||
|
* @see <a
|
||||||
|
* href="https://docs.oracle.com/javase/tutorial/java/data/numberformat.html">Formatting
|
||||||
|
* Numeric Print Output</a>
|
||||||
|
*
|
||||||
|
* @author John Aylward
|
||||||
|
*/
|
||||||
|
public class MyNumber extends Number {
|
||||||
|
private Number number = BigDecimal.valueOf(42);
|
||||||
|
/**
|
||||||
|
*/
|
||||||
|
private static final long serialVersionUID = 1L;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return number!
|
||||||
|
*/
|
||||||
|
public Number getNumber() {
|
||||||
|
return this.number;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int intValue() {
|
||||||
|
return getNumber().intValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public long longValue() {
|
||||||
|
return getNumber().longValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public float floatValue() {
|
||||||
|
return getNumber().floatValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public double doubleValue() {
|
||||||
|
return getNumber().doubleValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (non-Javadoc)
|
||||||
|
* @see java.lang.Object#toString()
|
||||||
|
*
|
||||||
|
* Number overrides should in general always override the toString method.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return getNumber().toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (non-Javadoc)
|
||||||
|
* @see java.lang.Object#hashCode()
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
final int prime = 31;
|
||||||
|
int result = 1;
|
||||||
|
result = prime * result + ((this.number == null) ? 0 : this.number.hashCode());
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (non-Javadoc)
|
||||||
|
* @see java.lang.Object#equals(java.lang.Object)
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object obj) {
|
||||||
|
if (this == obj) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (obj == null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!(obj instanceof MyNumber)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
MyNumber other = (MyNumber) obj;
|
||||||
|
if (this.number == null) {
|
||||||
|
if (other.number != null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} else if (!this.number.equals(other.number)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
13
src/test/java/org/json/junit/data/MyNumberContainer.java
Normal file
13
src/test/java/org/json/junit/data/MyNumberContainer.java
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class that holds our MyNumber override as a property.
|
||||||
|
* @author John Aylward
|
||||||
|
*/
|
||||||
|
public class MyNumberContainer {
|
||||||
|
private MyNumber myNumber = new MyNumber();
|
||||||
|
/**
|
||||||
|
* @return a MyNumber.
|
||||||
|
*/
|
||||||
|
public Number getMyNumber() {return this.myNumber;}
|
||||||
|
}
|
10
src/test/java/org/json/junit/data/MyPublicClass.java
Normal file
10
src/test/java/org/json/junit/data/MyPublicClass.java
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Need a class with some public data members for testing
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("boxing")
|
||||||
|
public class MyPublicClass {
|
||||||
|
public Integer publicInt = 42;
|
||||||
|
public String publicString = "abc";
|
||||||
|
}
|
91
src/test/java/org/json/junit/data/Singleton.java
Normal file
91
src/test/java/org/json/junit/data/Singleton.java
Normal file
|
@ -0,0 +1,91 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sample singleton for use with bean testing.
|
||||||
|
*
|
||||||
|
* @author John Aylward
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public final class Singleton {
|
||||||
|
/** */
|
||||||
|
private int someInt;
|
||||||
|
/** */
|
||||||
|
private String someString;
|
||||||
|
/** single instance. */
|
||||||
|
private static final Singleton INSTANCE = new Singleton();
|
||||||
|
|
||||||
|
/** @return the singleton instance. */
|
||||||
|
public static final Singleton getInstance() {
|
||||||
|
return INSTANCE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** */
|
||||||
|
private Singleton() {
|
||||||
|
if (INSTANCE != null) {
|
||||||
|
throw new IllegalStateException("Already instantiated");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Object clone() throws CloneNotSupportedException {
|
||||||
|
return INSTANCE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return someInt */
|
||||||
|
public int getSomeInt() {
|
||||||
|
return this.someInt;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* sets someInt.
|
||||||
|
*
|
||||||
|
* @param someInt
|
||||||
|
* the someInt to set
|
||||||
|
*/
|
||||||
|
public void setSomeInt(int someInt) {
|
||||||
|
this.someInt = someInt;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return someString */
|
||||||
|
public String getSomeString() {
|
||||||
|
return this.someString;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* sets someString.
|
||||||
|
*
|
||||||
|
* @param someString
|
||||||
|
* the someString to set
|
||||||
|
*/
|
||||||
|
public void setSomeString(String someString) {
|
||||||
|
this.someString = someString;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
final int prime = 31;
|
||||||
|
int result = 1;
|
||||||
|
result = prime * result + this.someInt;
|
||||||
|
result = prime * result + ((this.someString == null) ? 0 : this.someString.hashCode());
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object obj) {
|
||||||
|
if (this == obj)
|
||||||
|
return true;
|
||||||
|
if (obj == null)
|
||||||
|
return false;
|
||||||
|
if (getClass() != obj.getClass())
|
||||||
|
return false;
|
||||||
|
Singleton other = (Singleton) obj;
|
||||||
|
if (this.someInt != other.someInt)
|
||||||
|
return false;
|
||||||
|
if (this.someString == null) {
|
||||||
|
if (other.someString != null)
|
||||||
|
return false;
|
||||||
|
} else if (!this.someString.equals(other.someString))
|
||||||
|
return false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
62
src/test/java/org/json/junit/data/SingletonEnum.java
Normal file
62
src/test/java/org/json/junit/data/SingletonEnum.java
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sample singleton done as an Enum for use with bean testing.
|
||||||
|
*
|
||||||
|
* @author John Aylward
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public enum SingletonEnum {
|
||||||
|
/**
|
||||||
|
* the singleton instance.
|
||||||
|
*/
|
||||||
|
INSTANCE;
|
||||||
|
/** */
|
||||||
|
private int someInt;
|
||||||
|
/** */
|
||||||
|
private String someString;
|
||||||
|
|
||||||
|
/** single instance. */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the singleton instance. In a real application, I'd hope no one did
|
||||||
|
* this to an enum singleton.
|
||||||
|
*/
|
||||||
|
public static final SingletonEnum getInstance() {
|
||||||
|
return INSTANCE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** */
|
||||||
|
private SingletonEnum() {
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return someInt */
|
||||||
|
public int getSomeInt() {
|
||||||
|
return this.someInt;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* sets someInt.
|
||||||
|
*
|
||||||
|
* @param someInt
|
||||||
|
* the someInt to set
|
||||||
|
*/
|
||||||
|
public void setSomeInt(int someInt) {
|
||||||
|
this.someInt = someInt;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return someString */
|
||||||
|
public String getSomeString() {
|
||||||
|
return this.someString;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* sets someString.
|
||||||
|
*
|
||||||
|
* @param someString
|
||||||
|
* the someString to set
|
||||||
|
*/
|
||||||
|
public void setSomeString(String someString) {
|
||||||
|
this.someString = someString;
|
||||||
|
}
|
||||||
|
}
|
19
src/test/java/org/json/junit/data/StringsResourceBundle.java
Normal file
19
src/test/java/org/json/junit/data/StringsResourceBundle.java
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A resource bundle class
|
||||||
|
*/
|
||||||
|
public class StringsResourceBundle extends ListResourceBundle {
|
||||||
|
@Override
|
||||||
|
public Object[][] getContents() {
|
||||||
|
return contents;
|
||||||
|
}
|
||||||
|
static final Object[][] contents = {
|
||||||
|
{"greetings.hello", "Hello, "},
|
||||||
|
{"greetings.world", "World!"},
|
||||||
|
{"farewells.later", "Later, "},
|
||||||
|
{"farewells.gator", "Alligator!"}
|
||||||
|
};
|
||||||
|
}
|
68
src/test/java/org/json/junit/data/WeirdList.java
Normal file
68
src/test/java/org/json/junit/data/WeirdList.java
Normal file
|
@ -0,0 +1,68 @@
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package org.json.junit.data;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author John Aylward
|
||||||
|
*/
|
||||||
|
public class WeirdList {
|
||||||
|
/** */
|
||||||
|
private final List<Integer> list = new ArrayList<>();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param vals
|
||||||
|
*/
|
||||||
|
public WeirdList(Integer... vals) {
|
||||||
|
this.list.addAll(Arrays.asList(vals));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return a copy of the list
|
||||||
|
*/
|
||||||
|
public List<Integer> get() {
|
||||||
|
return new ArrayList<>(this.list);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return a copy of the list
|
||||||
|
*/
|
||||||
|
public List<Integer> getALL() {
|
||||||
|
return new ArrayList<>(this.list);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get a value at an index.
|
||||||
|
*
|
||||||
|
* @param i
|
||||||
|
* index to get
|
||||||
|
* @return the value at the index
|
||||||
|
*/
|
||||||
|
public Integer get(int i) {
|
||||||
|
return this.list.get(i);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get a value at an index.
|
||||||
|
*
|
||||||
|
* @param i
|
||||||
|
* index to get
|
||||||
|
* @return the value at the index
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("boxing")
|
||||||
|
public int getInt(int i) {
|
||||||
|
return this.list.get(i);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param value
|
||||||
|
* new value to add to the end of the list
|
||||||
|
*/
|
||||||
|
public void add(Integer value) {
|
||||||
|
this.list.add(value);
|
||||||
|
}
|
||||||
|
}
|
28
src/test/resources/jsonpointer-testdoc.json
Normal file
28
src/test/resources/jsonpointer-testdoc.json
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
{
|
||||||
|
"foo":
|
||||||
|
[
|
||||||
|
"bar",
|
||||||
|
"baz"
|
||||||
|
],
|
||||||
|
"": 0,
|
||||||
|
"a/b": 1,
|
||||||
|
"c%d": 2,
|
||||||
|
"e^f": 3,
|
||||||
|
"g|h": 4,
|
||||||
|
"i\\j": 5,
|
||||||
|
"k\"l": 6,
|
||||||
|
" ": 7,
|
||||||
|
"m~n": 8,
|
||||||
|
"obj" : {
|
||||||
|
"key" : "value",
|
||||||
|
"other~key" : {
|
||||||
|
"another/key" : [
|
||||||
|
"val"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"" : {
|
||||||
|
"" : "empty key of an object with an empty key",
|
||||||
|
"subKey" : "Some other value"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue