Code Monkey home page Code Monkey logo

scala-jsonapi's Introduction

Description

Build Status Coverage Status codecov.io Join the chat at https://gitter.im/zalando/scala-jsonapi Maven Central Javadocs

scala-jsonapi is a Scala library that aims to help you produce JSON output based on the JSON API specification easily and painlessly. The library is compatible with Scala version 2.11. It supports read and write for the following backends:

In addition, scala-jsonapi provides out-of-the-box (un)marshallers for:

  • Spray and Play
  • Akka Http

Current Status

This library is very much a work in progress, so expect its API to change.

Setup

To use scala-jsonapi, first add a library dependency—assuming that you have sonatype resolvers set up.

libraryDependencies += "org.zalando" %% "scala-jsonapi" % "0.6.2"

You also have to provide the used backend (e.g. spray-json).

Usage

The rich JSON API model is available via the following import:

import org.zalando.jsonapi.model._

The library provides serialization and deserialization of JSON API root objects to JSON using either Spray-JSON or Play-JSON. Please note that you need to explicitly add a dependency to either spray-json or play-json to your project.

Spray-JSON

import org.zalando.jsonapi.json.sprayjson.SprayJsonJsonapiProtocol._
import spray.json._

// Serialization
val rootObject: RootObject = ???
rootObject.toJson

// Deserialization
val json: JsValue = ???
json.convertTo[RootObject]

Play-JSON

Along with adding the scala-jsonapi library dependency, also include spray-httpx for marshalling support

libraryDependencies += "io.spray" %% "spray-httpx" % "1.3.3"

The JSON API support can then be imported using PlayJsonJsonapiSupport as follows

import org.zalando.jsonapi.json.playjson.PlayJsonJsonapiSupport._
import play.api.libs.json._

// Serialization
val rootObject: RootObject = ???
Json.toJson(rootObject)

// Deserialization
val json: JsValue = ???
Json.fromJson[RootObject](json)

Creating a JSON API Root Object

scala-jsonapi provides type class JsonapiRootObjectWriter so that you can create a JSON API representation for your resources. The following code snippet demonstrates its usage:

import org.zalando.jsonapi
import jsonapi.Jsonapi

case class Person(name: String)

implicit val personJsonapiWriter = new JsonapiRootObjectWriter[Person] {
  override def toJsonapi(person: Person) = {
    ???
  }
}

val personRootObject: RootObject = Jsonapi.asJsonapi(Person("Boris M."))

In contrast there is a type class called JsonapiRootObjectReader that supports conversion from JSON API representation to your resources. To illustrate:

import org.zalando.jsonapi
import jsonapi.Jsonapi

case class Person(name: String)

implicit val personJsonapiReader = new JsonapiRootObjectReader[Person] {
  override def fromJsonapi(rootObject: RootObject) = {
    ???
  }
}

val person: Person = Jsonapi.fromJsonapi[Person](RootObject(...))

For complete usage, see the specs example.

JSON API Links Support

There is support for string and object links.

To create a string "self" link:

Links.self("href", None)

To create an object "self" link:

Links.self("href", Some(meta))

Publishing and Releasing

Publishing and releasing is made with help of the sbt-sonatype plugin.

License

scala-jsonapi is licensed under the MIT license unless otherwise stated in the license files in higher directories (if any).

scala-jsonapi's People

Contributors

alanwguo avatar asgoncalves avatar camuthig avatar emanresusername avatar gitter-badger avatar jmhofer avatar jvexiau avatar meln1k avatar oporkka avatar palhash avatar unisay avatar wlk avatar wontheone1 avatar yvoderatskyi avatar zmeda avatar

Stargazers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

Watchers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

scala-jsonapi's Issues

json4s support

I like the idea of this project. I'm curious why json4s doesn't have an implementation. Is it just because it hasn't been contributed yet?

Fix issues described in warnings and add more tests

  • Fix issue (compile time warning) with match may not be exhaustive
  • Fix issues (compile time warnings) within Scaladocs like Could not find any member to link for "..."
  • Add more tests to cover Play-JSON support error cases

Missing PlayJsonJsonapiProtocol?

There is no PlayJsonJsonapiProtocol class in package org.zalando.jsonapi.json.playjson as the documentation suggests.

There is one in the sprayjson package but not in playjson.

The members data and errors MUST NOT coexist in the same document.

At the moment, RootObject is

  case class RootObject(
    data: Option[RootObject.Data] = None,
    links: Option[Links] = None,
    errors: Option[Errors] = None,
    meta: Option[Meta] = None,
    included: Option[Included] = None,
    jsonApi: Option[JsonApi] = None)

thus making it possible to instantiate a RootObject with both data and errors.

However, JSON-API spec clearly says:
http://jsonapi.org/format/#document-top-level

The members data and errors MUST NOT coexist in the same document.

Suggestion:

  case class RootObject(
    body: Either[Errors, RootObject.Data], // <<<< 
    links: Option[Links] = None,
    meta: Option[Meta] = None,
    included: Option[Included] = None,
    jsonApi: Option[JsonApi] = None)

(but it makes it impossible to have neither data nor errors)

Cloning and building gives me an unresolved dependency

Hi,

I just cloned the project completely fresh and started sbt.

I got this error message from sbt:

[warn]  ::::::::::::::::::::::::::::::::::::::::::::::
[warn]  ::          UNRESOLVED DEPENDENCIES         ::
[warn]  ::::::::::::::::::::::::::::::::::::::::::::::
[warn]  :: com.thoughtworks.paranamer#paranamer;2.6: configuration not found in com.thoughtworks.paranamer#paranamer;2.6: 'master(compile)'. Missing configuration: 'compile'. It was required from com.fasterxml.jackson.module#jackson-module-scala_2.10;2.5.1 compile
[warn]  ::::::::::::::::::::::::::::::::::::::::::::::
[warn] 
[warn]  Note: Unresolved dependencies path:
[warn]      com.thoughtworks.paranamer:paranamer:2.6
[warn]        +- org.json4s:json4s-core_2.10:3.2.10
[warn]        +- org.scala-sbt:serialization_2.10:0.1.2
[warn]        +- org.scala-sbt:cache:0.13.11
[warn]        +- org.scala-sbt:tracking:0.13.11
[warn]        +- org.scala-sbt:actions:0.13.11
[warn]        +- org.scala-sbt:main:0.13.11
[warn]        +- org.scala-sbt:sbt:0.13.11
[warn]        +- default:scala-jsonapi-build:0.1-SNAPSHOT (scalaVersion=2.10, sbtVersion=0.13)

Just to keep note of this for now, I'll try my hand at fixing this later.

Improve ExampleSpec.scala

"serialize accordingly with links object in data array" should be "serialize accordingly with links object in data object"

Unable to get correct Content-Type working in spray-json

I'm unable to get correct Content-Type as a response from spray-json marshaling (in akka-http).

I'm using similar code to this in my routing:

complete {
    val u: User = service.getUser(...)
    Jsonapi.asRootObject(u)
}

I have appropriate JsonapiRootObjectWriter[User] in scope and getting JSON output as expected, only issue I have is that Content-Type is application/json instead of application/vnd.api+json

I'm using latest scala-jsonapi and akka-http v 2.4.4

Support for null relationships

Currently when deserializing a json string into a RootObject using spray, I get a deserialization error if a relationship's data is null. This is supported by the json-api spec and should be allowed.

Example json object:

{
  "data": {
    "type": "note",
    "id": "1234",
    "attributes": {
      "title": "blah"
    },
    "relationships": {
      "author": {
        "data": null
      }
    }
  }
}

Call for maintainers

Hi everyone,

Zalando is not using this library anymore, and therefore it has been moved to its own Github organization. It would be good to find at least two maintainers from the current user base of the library to continue the work.

I think the near future projects would be to release version 1.0 with new jar structure (see #64, pretty much ready), and find out new package names for the project. After that release it would be a lot easier to start adding support for e.g. new Play and Scala versions.

I'll tag here people from latest pull requests. Please tell what you think. @alanwguo @emanresusername @texvex @zmeda

Dependency on io.spray:httpx

Artifact brings the dependency on io.spray:httpx transitively.
Its not clear why is it needed (esp. as we're using Akka HTTP in our project)
It should be possible to make it optional.

Please clarify what JSON API is

Could you please provide some links to the specification of JSON API? As far as I understand it is not JSON-schema or is it?

Simple play-json example

I am trying to make a simple example using play-json, but am running into a run-time error.

build.sbt

name := "JsonApiTest"
version := "1.0"
scalaVersion := "2.11.8"

libraryDependencies ++= Seq(
    "org.zalando" %% "scala-jsonapi" % "0.5.2",
    "com.typesafe.play" %% "play-json" % "2.5.4",
    "io.spray" %% "spray-httpx" % "1.3.3"
)

main.scala

import org.zalando.jsonapi.{JsonapiRootObjectWriter, _}
import org.zalando.jsonapi.json.playjson.PlayJsonJsonapiSupport._
import org.zalando.jsonapi.model.JsonApiObject.StringValue
import org.zalando.jsonapi.model.RootObject.ResourceObject
import org.zalando.jsonapi.model.{Attribute, RootObject}
import play.api.libs.json._

case class Person(id: String, name: String)
object Person {
    implicit val personJsonapiRootObjectWriter: JsonapiRootObjectWriter[Person] = new JsonapiRootObjectWriter[Person] {
        override def toJsonapi(person: Person) = {
            RootObject(data = Some(ResourceObject(
                `type` = "person",
                id = Some(person.id.toString),
                attributes = Some(List(
                    Attribute("name", StringValue(person.name))
                )), links = None)
            ))
        }
    }
}

object main {
    def main(args: Array[String]) {
        val p = Person("asd3nbs32g5", "Tyler")
        val j = Json.toJson(p.rootObject)
    }
}

Error:

java.lang.NoSuchMethodError: play.api.libs.functional.syntax.package$.functionalCanBuildApplicative(Lplay/api/libs/functional/Applicative;)Lplay/api/libs/functional/FunctionalCanBuild;
    at org.zalando.jsonapi.json.playjson.PlayJsonJsonapiFormat$class.rootObjectFormat(PlayJsonJsonapiFormat.scala:18)
    at org.zalando.jsonapi.json.playjson.PlayJsonJsonapiSupport$.rootObjectFormat$lzycompute(PlayJsonJsonapiSupport.scala:19)
    at org.zalando.jsonapi.json.playjson.PlayJsonJsonapiSupport$.rootObjectFormat(PlayJsonJsonapiSupport.scala:19)
    at main$.main(main.scala:26)
    at main.main(main.scala)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)

It will work if I use spray instead of play.

Inconsistency handling error links

Error objects are allowed to have links only with about member: http://jsonapi.org/format/#error-objects
Right now scala-jsonapi allows to create links with any type of allowed link.

At the same time, RootObject can't have about member as part of the links

The top-level links object MAY contain the following members:

self: the link that generated the current response document.
related: a related resource link when the primary data represents a resource relationship.
pagination links for the primary data.

Update project structure to use submodules for each backend

As a suggestion from issue #55 we should change project structure accordingly:

  • scala-jsonapi-core - Includes all core functionalities and domain model
  • scala-jsonapi-spray - Spray specific implementation
  • scala-jsonapi-play - Play specific implementation
  • scala-jsonapi-akkahttp - AkkaHttp specific implementation
  • scala-jsonapi-circe - Circe specific implementation

outcome of that should be 4 different artifacts which would all include core as a basis.

Support for Play-JSON

The library should support Play-JSON too. Maybe consider using rapture as abstract layer instead of providing support for all different JSON libraries.

Support Play 2.5

The current documented play-json version is causing errors with Play 2.5 applications.

java.lang.NoSuchMethodError: play.api.libs.functional.syntax.package$.functionalCanBuildApplicative(Lplay/api/libs/functional/Applicative;)Lplay/api/libs/functional/FunctionalCanBuild;
at org.zalando.jsonapi.json.playjson.PlayJsonJsonapiFormat$class.rootObjectFormat(PlayJsonJsonapiFormat.scala:18)
at org.zalando.jsonapi.json.playjson.PlayJsonJsonapiSupport$.rootObjectFormat$lzycompute(PlayJsonJsonapiSupport.scala:19)
at org.zalando.jsonapi.json.playjson.PlayJsonJsonapiSupport$.rootObjectFormat(PlayJsonJsonapiSupport.scala:19)

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.