-
Notifications
You must be signed in to change notification settings - Fork 0
ScalaJson
The recommend way of dealing with JSON is using Play’s typeclass based JSON library, located at play.api.libs.json
.
This library is built on top of Jerkson, which is a Scala wrapper around the super-fast Java based JSON library, Jackson.
The benefit of this approach is that both the Java and the Scala side of Play can share the same underlying library (Jackson), while Scala users can enjoy the extra type safety that Play’s JSON support brings to the table.
play.api.libs.json
package contains seven JSON data types:
JsObject
JsNull
JsUndefined
JsBoolean
JsNumber
JsArray
JsString
All of them inherit from the generic JSON value, JsValue
.
You can easily parse any JSON string as a JsValue
:
val json: JsValue = Json.parse(jsonString)
As soon as you have a JsValue
you can navigate into the tree. The API looks like the one provided to navigate into XML document by Scala using NodeSeq
:
val json = Json.parse(jsonString)
val maybeName = (json \ "user" \ "name").asOpt[String]
val emails = (json \ "user" \\\\ "emails").map(_.as[String])
Note that navigating using \ and \\ never fails. You must handle the error case at the end using
asOpt[T]
that will returnNone
if the value is missing. Otherwiser you can useas[T]
that we fail with an exception if the value was missing.
As soon as you have a type class able to transform the Scala type to Json, it is pretty easy to generate any Scala value to Json. For example let's create a simple Json object:
val jsonNumber = Json.toJson(4)
Or create a json array:
val jsonArray = Json.toJson(Seq(1, 2, 3, 4))
Here we have no problem to convert a Seq[Int]
into a Json array. However it is more complicated if the Seq
contains heterogeneous values:
val jsonArray = Json.toJson(Seq(1, "Bob", 3, 4))
Because there is no way to convert a Seq[Any]
to Json (Any
could be anything including something not supported by Json right?)
A simple solution is to handle it as a Seq[JsValue]
:
val jsonArray = Json.toJson(Seq(
toJson(1), toJson("Bob"), toJson(3), toJson(4)
))
Now let's see a last example of creating a more complex Json object:
val jsonObject = Json.toJson(
Map(
"users" -> Seq(
toJson(
Map(
"name" -> toJson("Bob"),
"age" -> toJson(31),
"email" -> toJson("[email protected]")
)
),
toJson(
Map(
"name" -> toJson("Kiki"),
"age" -> toJson(25),
"email" -> JsNull
)
)
)
)
)
That will generate this Json result:
{
"users":[
{
"name": "Bob",
"age": 31.0,
"email": "[email protected]"
},
{
"name": "Kiki",
"age": 25.0,
"email": null
}
]
}
The above Json object can be created in other ways too. Here is an alternative approach.
JsObject(
"users" -> JsArray(
JsObject(
"name" -> JsString("Bob") ::
"age" -> JsNumber(31) ::
"email" -> JsString("[email protected]") ::
Nil) ::
JsObject(
"name" -> JsString("Kiki") ::
"age" -> JsNumber(25) ::
"email" -> JsNull ::
Nil
) :: Nil
) :: Nil
)
Serializing a JsValue
to its json String representation is easy:
val jsonString: String = Json.stringify(jsValue)
While the typeclass based solution describe above is the on that's recommended, nothing stopping users from using any other JSON libraries if needed.
For example, here is a small snippet which demonstrates how to marshal plain scala objects into JSON and send it over the wire using the bundled, reflection based Jerkson library:
import com.codahale.jerkson.Json._
val json = generate(
Map(
"url"-> "http://nytimes.com",
"attributes" -> Map(
"name" -> "nytimes",
"country" -> "US",
"id" -> 25
),
"links" -> List(
"http://link1",
"http://link2"
)
)
)
- HTTP programming
- Asynchronous HTTP programming
- The template engine
- HTTP form submission and validation
- Working with JSON
- Working with XML
- Handling file upload
- Accessing an SQL database
- Using the Cache
- Calling WebServices
- Integrating with Akka
- Internationalization
- The application Global object
- Testing your application