2014-08-09 16:13:20 +00:00
|
|
|
|
Moshi
|
2014-08-09 15:58:42 +00:00
|
|
|
|
=====
|
2014-08-09 16:13:20 +00:00
|
|
|
|
|
2015-06-16 03:51:47 +00:00
|
|
|
|
Moshi is a modern JSON library for Android and Java. It makes it easy to parse JSON into Java
|
|
|
|
|
objects:
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
String json = ...;
|
|
|
|
|
|
|
|
|
|
Moshi moshi = new Moshi.Builder().build();
|
|
|
|
|
JsonAdapter<BlackjackHand> jsonAdapter = moshi.adapter(BlackjackHand.class);
|
|
|
|
|
|
|
|
|
|
BlackjackHand blackjackHand = jsonAdapter.fromJson(json);
|
|
|
|
|
System.out.println(blackjackHand);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
And it can just as easily serialize Java objects as JSON:
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
BlackjackHand blackjackHand = new BlackjackHand(
|
|
|
|
|
new Card('6', SPADES),
|
|
|
|
|
Arrays.asList(new Card('4', CLUBS), new Card('A', HEARTS)));
|
|
|
|
|
|
|
|
|
|
Moshi moshi = new Moshi.Builder().build();
|
|
|
|
|
JsonAdapter<BlackjackHand> jsonAdapter = moshi.adapter(BlackjackHand.class);
|
|
|
|
|
|
|
|
|
|
String json = jsonAdapter.toJson(blackjackHand);
|
|
|
|
|
System.out.println(json);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### Built-in Type Adapters
|
|
|
|
|
|
|
|
|
|
Moshi has built-in support for reading and writing Java’s core data types:
|
|
|
|
|
|
|
|
|
|
* Primitives (int, float, char...) and their boxed counterparts (Integer, Float, Character...).
|
|
|
|
|
* Arrays, Collections, Lists, Sets, and Maps
|
|
|
|
|
* Strings
|
|
|
|
|
* Enums
|
|
|
|
|
|
|
|
|
|
It supports your model classes by writing them out field-by-field. In the example above Moshi uses
|
|
|
|
|
these classes:
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
class BlackjackHand {
|
|
|
|
|
public final Card hidden_card;
|
|
|
|
|
public final List<Card> visible_cards;
|
|
|
|
|
...
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class Card {
|
|
|
|
|
public final char rank;
|
|
|
|
|
public final Suit suit;
|
|
|
|
|
...
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
enum Suit {
|
|
|
|
|
CLUBS, DIAMONDS, HEARTS, SPADES;
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
to read and write this JSON:
|
|
|
|
|
|
2015-06-16 18:49:49 +00:00
|
|
|
|
```json
|
2015-06-16 03:51:47 +00:00
|
|
|
|
{
|
|
|
|
|
"hidden_card": {
|
|
|
|
|
"rank": "6",
|
|
|
|
|
"suit": "SPADES"
|
|
|
|
|
},
|
|
|
|
|
"visible_cards": [
|
|
|
|
|
{
|
|
|
|
|
"rank": "4",
|
|
|
|
|
"suit": "CLUBS"
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"rank": "A",
|
|
|
|
|
"suit": "HEARTS"
|
|
|
|
|
}
|
|
|
|
|
]
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2015-06-16 16:58:09 +00:00
|
|
|
|
The [Javadoc][javadoc] catalogs the complete Moshi API, which we explore below.
|
|
|
|
|
|
2015-06-16 03:51:47 +00:00
|
|
|
|
### Custom Type Adapters
|
|
|
|
|
|
|
|
|
|
With Moshi, it’s particularly easy to customize how values are converted to and from JSON. A type
|
|
|
|
|
adapter is any class that has methods annotated `@ToJson` and `@FromJson`.
|
|
|
|
|
|
|
|
|
|
For example, Moshi’s default encoding of a playing card is verbose: the JSON defines the rank and
|
|
|
|
|
suit in separate fields: `{"rank":"A","suit":"HEARTS"}`. With a type adapter, we can change the
|
|
|
|
|
encoding to something more compact: `"4H"` for the four of hearts or `"JD"` for the jack of
|
|
|
|
|
diamonds:
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
class CardAdapter {
|
|
|
|
|
@ToJson String toJson(Card card) {
|
|
|
|
|
return card.rank + card.suit.name().substring(0, 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@FromJson Card fromJson(String card) {
|
|
|
|
|
if (card.length() != 2) throw new JsonDataException("Unknown card: " + card);
|
|
|
|
|
|
|
|
|
|
char rank = card.charAt(0);
|
|
|
|
|
switch (card.charAt(1)) {
|
|
|
|
|
case 'C': return new Card(rank, Suit.CLUBS);
|
|
|
|
|
case 'D': return new Card(rank, Suit.DIAMONDS);
|
|
|
|
|
case 'H': return new Card(rank, Suit.HEARTS);
|
|
|
|
|
case 'S': return new Card(rank, Suit.SPADES);
|
|
|
|
|
default: throw new JsonDataException("unknown suit: " + card);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Register the type adapter with the `Moshi.Builder` and we’re good to go.
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
Moshi moshi = new Moshi.Builder()
|
|
|
|
|
.add(new CardAdapter())
|
|
|
|
|
.build();
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Voila:
|
|
|
|
|
|
|
|
|
|
```json
|
|
|
|
|
{
|
|
|
|
|
"hidden_card": "6S",
|
|
|
|
|
"visible_cards": [
|
|
|
|
|
"4C",
|
|
|
|
|
"AH"
|
|
|
|
|
]
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2015-10-06 15:06:05 +00:00
|
|
|
|
#### Another example
|
|
|
|
|
|
2016-01-11 08:24:00 +00:00
|
|
|
|
Note that the method annotated with `@FromJson` does not need to take a String as an argument.
|
|
|
|
|
Rather it can take input of any type and Moshi will first parse the JSON to an object of that type
|
|
|
|
|
and then use the `@FromJson` method to produce the desired final value. Conversely, the method
|
|
|
|
|
annotated with `@ToJson` does not have to produce a String.
|
2015-10-06 15:06:05 +00:00
|
|
|
|
|
2016-01-11 08:24:00 +00:00
|
|
|
|
Assume, for example, that we have to parse a JSON in which the date and time of an event are
|
|
|
|
|
represented as two separate strings.
|
2015-10-06 15:06:05 +00:00
|
|
|
|
|
|
|
|
|
```json
|
|
|
|
|
{
|
|
|
|
|
"title": "Blackjack tournament",
|
|
|
|
|
"begin_date": "20151010",
|
|
|
|
|
"begin_time": "17:04"
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
We would like to combine these two fields into one string to facilitate the date parsing at a
|
2016-01-11 08:24:00 +00:00
|
|
|
|
later point. Also, we would like to have all variable names in CamelCase. Therefore, the `Event`
|
|
|
|
|
class we want Moshi to produce like this:
|
2015-10-06 15:06:05 +00:00
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
class Event {
|
2016-01-11 08:24:00 +00:00
|
|
|
|
String title;
|
|
|
|
|
String beginDateAndTime;
|
2015-10-06 15:06:05 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2016-01-11 08:24:00 +00:00
|
|
|
|
Instead of manually parsing the JSON line per line (which we could also do) we can have Moshi do the
|
|
|
|
|
transformation automatically. We simply define another class `EventJson` that directly corresponds
|
|
|
|
|
to the JSON structure:
|
2015-10-06 15:06:05 +00:00
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
class EventJson {
|
2016-01-11 08:24:00 +00:00
|
|
|
|
String title;
|
|
|
|
|
String begin_date;
|
|
|
|
|
String begin_time;
|
2015-10-06 15:06:05 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2016-01-11 08:24:00 +00:00
|
|
|
|
And another class with the appropriate `@FromJson` and `@ToJson` methods that are telling Moshi how
|
|
|
|
|
to convert an `EventJson` to an `Event` and back. Now, whenever we are asking Moshi to parse a JSON
|
|
|
|
|
to an `Event` it will first parse it to an `EventJson` as an intermediate step. Conversely, to
|
|
|
|
|
serialize an `Event` Moshi will first create an `EventJson` object and then serialize that object as
|
|
|
|
|
usual.
|
2015-10-06 15:06:05 +00:00
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
class EventJsonAdapter {
|
2016-01-11 08:24:00 +00:00
|
|
|
|
@FromJson Event eventFromJson(EventJson eventJson) {
|
|
|
|
|
Event event = new Event();
|
|
|
|
|
event.title = eventJson.title;
|
|
|
|
|
event.beginDateAndTime = eventJson.begin_date + " " + eventJson.begin_time;
|
|
|
|
|
return event;
|
|
|
|
|
}
|
2015-10-06 15:06:05 +00:00
|
|
|
|
|
2016-01-11 08:24:00 +00:00
|
|
|
|
@ToJson EventJson eventToJson(Event event) {
|
|
|
|
|
EventJson json = new EventJson();
|
|
|
|
|
json.title = event.title;
|
|
|
|
|
json.begin_date = event.beginDateAndTime.substring(0, 8);
|
|
|
|
|
json.begin_time = event.beginDateAndTime.substring(9, 14);
|
|
|
|
|
return json;
|
|
|
|
|
}
|
2015-10-06 15:06:05 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Again we register the adapter with Moshi.
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
Moshi moshi = new Moshi.Builder()
|
|
|
|
|
.add(new EventJsonAdapter())
|
|
|
|
|
.build();
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
We can now use Moshi to parse the JSON directly to an `Event`.
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
JsonAdapter<Event> jsonAdapter = moshi.adapter(Event.class);
|
|
|
|
|
Event event = jsonAdapter.fromJson(json);
|
|
|
|
|
```
|
|
|
|
|
|
2015-06-16 03:51:47 +00:00
|
|
|
|
### Fails Gracefully
|
|
|
|
|
|
|
|
|
|
Automatic databinding almost feels like magic. But unlike the black magic that typically accompanies
|
|
|
|
|
reflection, Moshi is designed to help you out when things go wrong.
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
|
JsonDataException: Expected one of [CLUBS, DIAMONDS, HEARTS, SPADES] but was ANCHOR at path $.visible_cards[2].suit
|
|
|
|
|
at com.squareup.moshi.JsonAdapters$11.fromJson(JsonAdapters.java:188)
|
|
|
|
|
at com.squareup.moshi.JsonAdapters$11.fromJson(JsonAdapters.java:180)
|
|
|
|
|
...
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Moshi always throws a standard `java.io.IOException` if there is an error reading the JSON document,
|
|
|
|
|
or if it is malformed. It throws a `JsonDataException` if the JSON document is well-formed, but
|
|
|
|
|
doesn’t match the expected format.
|
|
|
|
|
|
|
|
|
|
### Built on Okio
|
|
|
|
|
|
|
|
|
|
Moshi uses [Okio][okio] for simple and powerful I/O. It’s a fine complement to [OkHttp][okhttp],
|
|
|
|
|
which can share buffer segments for maximum efficiency.
|
|
|
|
|
|
|
|
|
|
### Borrows from Gson
|
|
|
|
|
|
|
|
|
|
Moshi uses the same streaming and binding mechanisms as [Gson][gson]. If you’re a Gson user you’ll
|
|
|
|
|
find Moshi works similarly. If you try Moshi and don’t love it, you can even migrate to Gson without
|
|
|
|
|
much violence!
|
|
|
|
|
|
|
|
|
|
But the two libraries have a few important differences:
|
|
|
|
|
|
|
|
|
|
* **Moshi has fewer built-in type adapters.** For example, you need to configure your own date
|
|
|
|
|
adapter. Most binding libraries will encode whatever you throw at them. Moshi refuses to
|
|
|
|
|
serialize platform types (`java.*`, `javax.*`, and `android.*`) without a user-provided type
|
|
|
|
|
adapter. This is intended to prevent you from accidentally locking yourself to a specific JDK or
|
|
|
|
|
Android release.
|
|
|
|
|
* **Moshi is less configurable.** There’s no field naming strategy, versioning, instance creators,
|
|
|
|
|
or long serialization policy. Instead of naming a field `visibleCards` and using a policy class
|
|
|
|
|
to convert that to `visible_cards`, Moshi wants you to just name the field `visible_cards` as it
|
|
|
|
|
appears in the JSON.
|
|
|
|
|
* **Moshi doesn’t have a `JsonElement` model.** Instead it just uses built-in types like `List` and
|
|
|
|
|
`Map`.
|
|
|
|
|
* **No HTML-safe escaping.** Gson encodes `=` as `\u003d` by default so that it can be safely
|
|
|
|
|
encoded in HTML without additional escaping. Moshi encodes it naturally (as `=`) and assumes that
|
|
|
|
|
the HTML encoder – if there is one – will do its job.
|
2014-08-09 16:13:20 +00:00
|
|
|
|
|
2016-01-11 08:24:00 +00:00
|
|
|
|
### Custom field names with @Json
|
|
|
|
|
|
|
|
|
|
Moshi works best when your JSON objects and Java objects have the same structure. But when they
|
|
|
|
|
don't, Moshi has annotations to customize data binding.
|
|
|
|
|
|
|
|
|
|
Use `@Json` to specify how Java fields map to JSON names. This is necessary when the JSON name
|
|
|
|
|
contains spaces or other characters that aren’t permitted in Java field names. For example, this
|
|
|
|
|
JSON has a field name containing a space:
|
|
|
|
|
|
|
|
|
|
```json
|
|
|
|
|
{
|
|
|
|
|
"username": "jesse",
|
|
|
|
|
"lucky number": 32
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
With `@Json` its corresponding Java class is easy:
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
class Player {
|
|
|
|
|
String username;
|
|
|
|
|
@Json(name = "lucky number") int luckyNumber;
|
|
|
|
|
|
|
|
|
|
...
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Because JSON field names are always defined with their Java fields, Moshi makes it easy to find
|
|
|
|
|
fields when navigating between Java and JSON.
|
|
|
|
|
|
|
|
|
|
### Alternate type adapters with @JsonQualifier
|
|
|
|
|
|
|
|
|
|
Use `@JsonQualifier` to customize how a type is encoded for some fields without changing its
|
|
|
|
|
encoding everywhere. This works similarly to the qualifier annotations in dependency injection
|
|
|
|
|
tools like Dagger and Guice.
|
|
|
|
|
|
|
|
|
|
Here’s a JSON message with two integers and a color:
|
|
|
|
|
|
|
|
|
|
```json
|
|
|
|
|
{
|
|
|
|
|
"width": 1024,
|
|
|
|
|
"height": 768,
|
|
|
|
|
"color": "#ff0000"
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
By convention, Android programs also use `int` for colors:
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
class Rectangle {
|
|
|
|
|
int width;
|
|
|
|
|
int height;
|
|
|
|
|
int color;
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
But if we encoded the above Java class as JSON, the color isn't encoded properly!
|
|
|
|
|
|
|
|
|
|
```json
|
|
|
|
|
{
|
|
|
|
|
"width": 1024,
|
|
|
|
|
"height": 768,
|
|
|
|
|
"color": 16711680
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
The fix is to define a qualifier annotation, itself annotated `@JsonQualifier`:
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
@Retention(RUNTIME)
|
|
|
|
|
@JsonQualifier
|
|
|
|
|
public @interface HexColor {
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Next apply this `@HexColor` annotation to the appropriate field:
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
|
class Rectangle {
|
|
|
|
|
int width;
|
|
|
|
|
int height;
|
|
|
|
|
@HexColor int color;
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
And finally define a type adapter to handle it:
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
|
/** Converts strings like #ff0000 to the corresponding color ints. */
|
|
|
|
|
class ColorAdapter {
|
|
|
|
|
@ToJson String toJson(@HexColor int rgb) {
|
|
|
|
|
return String.format("#%06x", rgb);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@FromJson @HexColor int fromJson(String rgb) {
|
|
|
|
|
return Integer.parseInt(rgb.substring(1), 16);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Use `@JsonQualifier` when you need different JSON encodings for the same type. Most programs
|
|
|
|
|
shouldn’t need this `@JsonQualifier`, but it’s very handy for those that do.
|
2015-03-27 04:29:13 +00:00
|
|
|
|
|
|
|
|
|
Download
|
|
|
|
|
--------
|
|
|
|
|
|
2015-10-25 12:38:26 +00:00
|
|
|
|
Download [the latest JAR][dl] or depend via Maven:
|
2015-06-16 16:58:09 +00:00
|
|
|
|
```xml
|
|
|
|
|
<dependency>
|
|
|
|
|
<groupId>com.squareup.moshi</groupId>
|
|
|
|
|
<artifactId>moshi</artifactId>
|
2016-01-18 22:16:20 +00:00
|
|
|
|
<version>1.1.0</version>
|
2015-06-16 16:58:09 +00:00
|
|
|
|
</dependency>
|
|
|
|
|
```
|
|
|
|
|
or Gradle:
|
|
|
|
|
```groovy
|
2016-01-18 22:16:20 +00:00
|
|
|
|
compile 'com.squareup.moshi:moshi:1.1.0'
|
2015-06-16 16:58:09 +00:00
|
|
|
|
```
|
2015-03-27 04:29:13 +00:00
|
|
|
|
|
2015-06-16 16:58:09 +00:00
|
|
|
|
Snapshots of the development version are available in [Sonatype's `snapshots` repository][snap].
|
2015-03-27 04:29:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
License
|
|
|
|
|
--------
|
|
|
|
|
|
|
|
|
|
Copyright 2015 Square, Inc.
|
|
|
|
|
|
|
|
|
|
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.
|
2015-03-29 06:40:42 +00:00
|
|
|
|
|
|
|
|
|
|
2015-06-16 16:58:09 +00:00
|
|
|
|
[dl]: https://search.maven.org/remote_content?g=com.squareup.moshi&a=moshi&v=LATEST
|
|
|
|
|
[snap]: https://oss.sonatype.org/content/repositories/snapshots/com/squareup/moshi/
|
2015-06-16 03:51:47 +00:00
|
|
|
|
[okio]: https://github.com/square/okio/
|
2015-10-25 12:38:26 +00:00
|
|
|
|
[okhttp]: https://github.com/square/okhttp/
|
|
|
|
|
[gson]: https://github.com/google/gson/
|
2015-10-26 15:01:29 +00:00
|
|
|
|
[javadoc]: http://square.github.io/moshi/1.x/moshi/
|