1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
|
/*
* Copyright (C) 2011 Mathias Doenitz
*
* 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
*
* http://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.
*/
package spray.json
import org.specs2.mutable._
class JsonParserSpec extends Specification {
"The JsonParser" should {
"parse 'null' to JsNull" in {
JsonParser("null") === JsNull
}
"parse 'true' to JsTrue" in {
JsonParser("true") === JsTrue
}
"parse 'false' to JsFalse" in {
JsonParser("false") === JsFalse
}
"parse '0' to JsNumber" in {
JsonParser("0") === JsNumber(0)
}
"parse '1.23' to JsNumber" in {
JsonParser("1.23") === JsNumber(1.23)
}
"parse '-1E10' to JsNumber" in {
JsonParser("-1E10") === JsNumber("-1E+10")
}
"parse '12.34e-10' to JsNumber" in {
JsonParser("12.34e-10") === JsNumber("1.234E-9")
}
"parse \"xyz\" to JsString" in {
JsonParser("\"xyz\"") === JsString("xyz")
}
"parse escapes in a JsString" in {
JsonParser(""""\"\\/\b\f\n\r\t"""") === JsString("\"\\/\b\f\n\r\t")
JsonParser("\"L\\" + "u00e4nder\"") === JsString("Länder")
}
"parse all representations of the slash (SOLIDUS) character in a JsString" in {
JsonParser( "\"" + "/\\/\\u002f" + "\"") === JsString("///")
}
"parse a simple JsObject" in (
JsonParser(""" { "key" :42, "key2": "value" }""") ===
JsObject("key" -> JsNumber(42), "key2" -> JsString("value"))
)
"parse a simple JsArray" in (
JsonParser("""[null, 1.23 ,{"key":true } ] """) ===
JsArray(JsNull, JsNumber(1.23), JsObject("key" -> JsTrue))
)
"parse directly from UTF-8 encoded bytes" in {
val json = JsObject(
"7-bit" -> JsString("This is regular 7-bit ASCII text."),
"2-bytes" -> JsString("2-byte UTF-8 chars like £, æ or Ö"),
"3-bytes" -> JsString("3-byte UTF-8 chars like ヨ, ᄅ or ᐁ."),
"4-bytes" -> JsString("4-byte UTF-8 chars like \uD801\uDC37, \uD852\uDF62 or \uD83D\uDE01."))
JsonParser(json.prettyPrint.getBytes("UTF-8")) === json
}
"parse directly from UTF-8 encoded bytes when string starts with a multi-byte character" in {
val json = JsString("£0.99")
JsonParser(json.prettyPrint.getBytes("UTF-8")) === json
}
"be reentrant" in {
val largeJsonSource = scala.io.Source.fromInputStream(getClass.getResourceAsStream("/test.json")).mkString
import scala.collection.parallel.immutable.ParSeq
ParSeq.fill(20)(largeJsonSource).map(JsonParser(_)).toList.map {
_.asInstanceOf[JsObject].fields("questions").asInstanceOf[JsArray].elements.size
} === List.fill(20)(100)
}
"produce proper error messages" in {
def errorMessage(input: String) =
try JsonParser(input) catch { case e: JsonParser.ParsingException => e.getMessage }
errorMessage("""[null, 1.23 {"key":true } ]""") ===
"""Unexpected character '{' at input index 12 (line 1, position 13), expected ']':
|[null, 1.23 {"key":true } ]
| ^
|""".stripMargin
errorMessage("""[null, 1.23, { key":true } ]""") ===
"""Unexpected character 'k' at input index 16 (line 1, position 17), expected '"':
|[null, 1.23, { key":true } ]
| ^
|""".stripMargin
errorMessage("""{"a}""") ===
"""Unexpected end-of-input at input index 4 (line 1, position 5), expected '"':
|{"a}
| ^
|""".stripMargin
errorMessage("""{}x""") ===
"""Unexpected character 'x' at input index 2 (line 1, position 3), expected end-of-input:
|{}x
| ^
|""".stripMargin
}
"parse multiple values when allowTrailingInput" in {
val parser = new JsonParser("""{"key":1}{"key":2}""")
parser.parseJsValue(true) === JsObject("key" -> JsNumber(1))
parser.parseJsValue(true) === JsObject("key" -> JsNumber(2))
}
"reject trailing input when !allowTrailingInput" in {
def parser = JsonParser("""{"key":1}x""")
parser must throwA[JsonParser.ParsingException].like {
case e: JsonParser.ParsingException => e.getMessage must contain("expected end-of-input")
}
}
}
}
|