circe is a JSON library for Scala (and Scala.js). The rest of this page tries to give some justification for its existence. There are also API docs.
circe's working title was jfc, which stood for "JSON for cats". The name was changed for a number of reasons.
circe is published to Maven Central and cross-built for Scala 2.10 and 2.11, so you can just add the following to your build:
libraryDependencies ++= Seq(
"io.circe" %% "circe-core" % "0.2.0",
"io.circe" %% "circe-generic" % "0.2.0",
"io.circe" %% "circe-parse" % "0.2.0"
)If you are using circe's generic derivation with Scala 2.10, you'll also need to include the Macro Paradise compiler plugin in your build:
addCompilerPlugin(
"org.scalamacros" % "paradise" % "2.1.0-M5" cross CrossVersion.full
)Then type sbt console to start a REPL and then paste the following (this will also work from the
root directory of this repository):
scala> import io.circe._, io.circe.generic.auto._, io.circe.parse._, io.circe.syntax._
import io.circe._
import io.circe.generic.auto._
import io.circe.parse._
import io.circe.syntax._
scala> sealed trait Foo
defined trait Foo
scala> case class Bar(xs: List[String]) extends Foo
defined class Bar
scala> case class Qux(i: Int, d: Option[Double]) extends Foo
defined class Qux
scala> val foo: Foo = Qux(13, Some(14.0))
foo: Foo = Qux(13,Some(14.0))
scala> foo.asJson.noSpaces
res0: String = {"Qux":{"d":14.0,"i":13}}
scala> decode[Foo](foo.asJson.spaces4)
res1: cats.data.Xor[io.circe.Error,Foo] = Right(Qux(13,Some(14.0)))No boilerplate, no runtime reflection.
Argonaut is a great library. It's by far the best JSON library for Scala, and the best JSON library on the JVM. If you're doing anything with JSON in Scala, you should be using Argonaut.
circe is a fork of Argonaut with a few important differences.
circe depends on cats instead of Scalaz, and the core project has only two
dependencies: cats-core and export-hook (a lightweight mechanism for cleaner generic
type class instance derivation).
Other subprojects bring in dependencies on Jawn (for parsing in the jawn
subproject), Shapeless (for automatic codec derivation in generic),
and Twitter Util (for tools for asynchronous parsing in async), but it would be possible
to replace the functionality provided by these subprojects with alternative implementations that use
other libraries.
circe doesn't include a JSON parser in the core project, which is focused on the JSON AST, zippers,
and codecs. The jawn subproject provides support for parsing JSON via a Jawn
facade. Jawn is fast, it offers asynchronous parsing, and best of all it lets us drop a lot of the
fussiest code in Argonaut.
circe also provides a parse subproject that provides parsing support for Scala.js,
with JVM parsing provided by io.circe.jawn and JavaScript parsing from scalajs.js.JSON.
circe doesn't use or provide lenses in the core project (or at all, for now). This is related to
the first point above, since Monocle has a Scalaz dependency, but we also feel that it
simplifies the API. We'd consider adding lenses in a subproject if Monocle (or something similar)
gets ported to cats.
circe does not use macros or provide any kind of automatic derivation in the core project. Instead
of Argonaut's limited macro-based derivation (which does not support sealed trait hierarchies, for
example), circe includes a subproject (generic) that provides generic codec derivation using
Shapeless.
This subproject is currently a simplified port of argonaut-shapeless that provides fully automatic derivation of instances for case classes and sealed trait hierarchies. It also includes derivation of "incomplete" case class instances (see my recent blog post for details).
circe aims to simplify Argonaut's API by removing all operator aliases. This is largely a matter of personal taste, and may change in the future.
The Argonaut documentation is good, but it could be better: to take just one example, it can be hard
to tell at a glance why there are three different Cursor, HCursor, and ACursor types. In this
particular case, circe introduces an abstraction over cursors that makes the relationship clearer and
allows these three types to share API documentation.
I'd like to provide more complete test coverage (in part via Discipline), but it's early days for this.
circe aims to be more focused on performance. I'm still experimenting with the right balance, but I'm open to using mutability, inheritance, and all kinds of other horrible things under the hood if they make circe faster (the public API does not and will never expose any of this, though).
My initial benchmarks suggest this is at least kind of working (higher numbers are better):
Benchmark Mode Cnt Score Error Units
DecodingBenchmark.decodeFoosA thrpt 80 1185.694 ± 1.947 ops/s
DecodingBenchmark.decodeFoosC thrpt 80 1972.754 ± 6.829 ops/s
DecodingBenchmark.decodeFoosP thrpt 80 1592.846 ± 2.842 ops/s
DecodingBenchmark.decodeIntsA thrpt 80 7229.576 ± 11.329 ops/s
DecodingBenchmark.decodeIntsC thrpt 80 11136.719 ± 84.116 ops/s
DecodingBenchmark.decodeIntsP thrpt 80 12368.444 ± 17.704 ops/s
EncodingBenchmark.encodeFoosA thrpt 80 6206.449 ± 17.351 ops/s
EncodingBenchmark.encodeFoosC thrpt 80 6401.355 ± 30.861 ops/s
EncodingBenchmark.encodeFoosP thrpt 80 2475.053 ± 8.681 ops/s
EncodingBenchmark.encodeIntsA thrpt 80 59894.725 ± 99.221 ops/s
EncodingBenchmark.encodeIntsC thrpt 80 96090.720 ± 215.812 ops/s
EncodingBenchmark.encodeIntsP thrpt 80 56837.610 ± 126.424 ops/s
ParsingBenchmark.parseFoosA thrpt 80 2485.312 ± 120.205 ops/s
ParsingBenchmark.parseFoosC thrpt 80 3189.966 ± 27.194 ops/s
ParsingBenchmark.parseFoosP thrpt 80 1961.319 ± 8.772 ops/s
ParsingBenchmark.parseIntsA thrpt 80 11356.440 ± 51.679 ops/s
ParsingBenchmark.parseIntsC thrpt 80 34158.979 ± 119.792 ops/s
ParsingBenchmark.parseIntsP thrpt 80 14032.825 ± 51.602 ops/s
PrintingBenchmark.printFoosA thrpt 80 2867.108 ± 8.117 ops/s
PrintingBenchmark.printFoosC thrpt 80 3429.558 ± 7.957 ops/s
PrintingBenchmark.printFoosP thrpt 80 7156.045 ± 15.081 ops/s
PrintingBenchmark.printIntsA thrpt 80 14836.201 ± 90.667 ops/s
PrintingBenchmark.printIntsC thrpt 80 22402.903 ± 49.486 ops/s
PrintingBenchmark.printIntsP thrpt 80 71329.444 ± 473.565 ops/s
And allocation rates (lower is better):
Benchmark Mode Cnt Score Error Units
DecodingBenchmark.decodeFoosA:gc.alloc.rate.norm thrpt 40 3732264.935 ± 23018.108 B/op
DecodingBenchmark.decodeFoosC:gc.alloc.rate.norm thrpt 40 1996228.623 ± 34534.041 B/op
DecodingBenchmark.decodeFoosP:gc.alloc.rate.norm thrpt 40 2136256.692 ± 5469.639 B/op
DecodingBenchmark.decodeIntsA:gc.alloc.rate.norm thrpt 40 599396.406 ± 13688.247 B/op
DecodingBenchmark.decodeIntsC:gc.alloc.rate.norm thrpt 40 366632.531 ± 1.560 B/op
DecodingBenchmark.decodeIntsP:gc.alloc.rate.norm thrpt 40 369120.092 ± 0.130 B/op
EncodingBenchmark.encodeFoosA:gc.alloc.rate.norm thrpt 40 522119.886 ± 638.624 B/op
EncodingBenchmark.encodeFoosC:gc.alloc.rate.norm thrpt 40 414886.029 ± 4.374 B/op
EncodingBenchmark.encodeFoosP:gc.alloc.rate.norm thrpt 40 1346197.162 ± 4.083 B/op
EncodingBenchmark.encodeIntsA:gc.alloc.rate.norm thrpt 40 80136.019 ± 0.027 B/op
EncodingBenchmark.encodeIntsC:gc.alloc.rate.norm thrpt 40 48360.012 ± 0.016 B/op
EncodingBenchmark.encodeIntsP:gc.alloc.rate.norm thrpt 40 71352.020 ± 0.028 B/op
ParsingBenchmark.parseFoosA:gc.alloc.rate.norm thrpt 40 1455168.738 ± 453.037 B/op
ParsingBenchmark.parseFoosC:gc.alloc.rate.norm thrpt 40 735320.446 ± 13.339 B/op
ParsingBenchmark.parseFoosP:gc.alloc.rate.norm thrpt 40 982880.544 ± 0.764 B/op
ParsingBenchmark.parseIntsA:gc.alloc.rate.norm thrpt 40 310280.090 ± 0.126 B/op
ParsingBenchmark.parseIntsC:gc.alloc.rate.norm thrpt 40 105220.031 ± 6.837 B/op
ParsingBenchmark.parseIntsP:gc.alloc.rate.norm thrpt 40 200464.075 ± 0.105 B/op
PrintingBenchmark.printFoosA:gc.alloc.rate.norm thrpt 40 596120.397 ± 6837.064 B/op
PrintingBenchmark.printFoosC:gc.alloc.rate.norm thrpt 40 381989.540 ± 75.447 B/op
PrintingBenchmark.printFoosP:gc.alloc.rate.norm thrpt 40 349576.152 ± 0.212 B/op
PrintingBenchmark.printIntsA:gc.alloc.rate.norm thrpt 40 239712.070 ± 0.098 B/op
PrintingBenchmark.printIntsC:gc.alloc.rate.norm thrpt 40 95408.048 ± 0.069 B/op
PrintingBenchmark.printIntsP:gc.alloc.rate.norm thrpt 40 24328.015 ± 0.021 B/op
The Foos benchmarks work with a map containing case class values, and the Ints ones are an array
of integers. C suffixes indicate circe's throughput, A is for Argonaut, and P is for
play-json.
This section needs a lot of expanding.
circe uses Encoder and Decoder type classes for encoding and decoding. An Encoder[A] instance
provides a function that will convert any A to a JSON, and a Decoder[A] takes a Json value
to either an exception or an A. circe provides implicit instances of these type classes for many
types from the Scala standard library, including Int, String, and others. It also
provides instances for List[A], Option[A], and other generic types, but only if A has an
Encoder instance.
Suppose we have the following JSON document:
import io.circe._, io.circe.generic.auto._, io.circe.jawn._, io.circe.syntax._
import cats.data.Xor
val json: String = """
{
"id": "c730433b-082c-4984-9d66-855c243266f0",
"name": "Foo",
"counts": [1, 2, 3],
"values": {
"bar": true,
"baz": 100.001,
"qux": ["a", "b"]
}
}
"""
val doc: Json = parse(json).getOrElse(Json.empty)In order to transform this document we need to create an HCursor with the focus at the document's
root:
val cursor: HCursor = doc.hcursorWe can then use various operations to move the focus of the cursor around the document and to "modify" the current focus:
val reversedNameCursor: ACursor =
cursor.downField("name").withFocus(_.mapString(_.reverse))We can then return to the root of the document and return its value with top:
val reversedName: Option[Json] = reversedNameCursor.topThe result will contain the original document with the "name" field reversed.
circe is a fork of Argonaut, and if you find it at all useful, you should thank Mark Hibberd, Tony Morris, Kenji Yoshida, and the rest of the Argonaut contributors.
circe is currently maintained by Travis Brown, Alexandre Archambault, and Vladimir Kostyukov. After the 0.3.0 release, all pull requests will require two sign-offs by a maintainer to be merged.
The circe project supports the Typelevel code of conduct and wants all of its channels (Gitter, GitHub, etc.) to be welcoming environments for everyone.
circe is licensed under the Apache License, Version 2.0 (the "License"); you may not use this software except in compliance with the License.
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.