/* * 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 { import scala.concurrent.{Await, Future} import scala.concurrent.duration._ import scala.concurrent.ExecutionContext.Implicits.global val largeJsonSource = scala.io.Source.fromInputStream(getClass.getResourceAsStream("/test.json")).mkString val list = Await.result( Future.traverse(List.fill(20)(largeJsonSource))(src => Future(JsonParser(src))), 5.seconds ) list.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") } } } }