spray.json
by Miguel Lopez
由Miguel Lopez
In the previous post, we added JSON support to our Akka HTTP API using circe.
在上一篇文章中 ,我们使用circe将JSON支持添加到了Akka HTTP API中。
This time we’ll do the same but using spray-json. Akka HTTP supports it by providing an official library — we don’t need a third-party party one like we did with circe.
这次我们将执行相同的操作,但使用spray-json。 Akka HTTP通过提供一个官方库来支持它-我们不需要像circe那样的第三方。
We’ll go through the same steps as the previous tutorial to set up the project.
我们将按照与上一教程相同的步骤来设置项目。
Clone the repo, and check out the branch 3.3-repository-implementation
.
克隆回购 ,并检查了分支3.3-repository-implementation
。
We will also do the changes we did in the previous tutorial.
我们还将进行上一教程中所做的更改。
First, we will replace the circe dependencies with the spray-json dependency since we won’t be needing it for this tutorial. Update the build.sbt
file with the following contents:
首先,我们将用circle-json依赖关系替换circe依赖关系,因为本教程将不需要它。 使用以下内容更新build.sbt
文件:
name := "akkahttp-quickstart"version := "0.1"scalaVersion := "2.12.6"val akkaVersion = "2.5.13"val akkaHttpVersion = "10.1.3"libraryDependencies ++= Seq( "com.typesafe.akka" %% "akka-actor" % akkaVersion, "com.typesafe.akka" %% "akka-testkit" % akkaVersion % Test, "com.typesafe.akka" %% "akka-stream" % akkaVersion, "com.typesafe.akka" %% "akka-stream-testkit" % akkaVersion % Test, "com.typesafe.akka" %% "akka-http" % akkaHttpVersion, "com.typesafe.akka" %% "akka-http-testkit" % akkaHttpVersion % Test, "com.typesafe.akka" %% "akka-http-spray-json" % akkaHttpVersion, "org.scalatest" %% "scalatest" % "3.0.5" % Test)
Next, we will add a save
function to the TodoRepository
and its implementation:
接下来,我们将save
函数添加到TodoRepository
及其实现中:
import scala.concurrent.{ExecutionContext, Future}trait TodoRepository { def all(): Future[Seq[Todo]] def done(): Future[Seq[Todo]] def pending(): Future[Seq[Todo]] def save(todo: Todo): Future[Todo]}class InMemoryTodoRepository(initialTodos: Seq[Todo] = Seq.empty)(implicit ec: ExecutionContext) extends TodoRepository { private var todos: Vector[Todo] = initialTodos.toVector override def all(): Future[Seq[Todo]] = Future.successful(todos) override def done(): Future[Seq[Todo]] = Future.successful(todos.filter(_.done)) override def pending(): Future[Seq[Todo]] = Future.successful(todos.filterNot(_.done)) override def save(todo: Todo): Future[Todo] = Future.successful { todos = todos :+ todo todo }}
This will allow us to create a POST request to create new todos.
这将允许我们创建一个POST请求来创建新的待办事项。
And finally, update the Main
object to create a list of todos for testing purposes, and with the appropriate routes:
最后,更新Main
对象以创建用于测试目的的待办事项列表,并使用适当的路由:
import akka.actor.ActorSystemimport akka.http.scaladsl.Httpimport akka.stream.ActorMaterializerimport scala.concurrent.Awaitimport scala.util.{Failure, Success}object Main extends App { val host = "0.0.0.0" val port = 9000 implicit val system: ActorSystem = ActorSystem(name = "todoapi") implicit val materializer: ActorMaterializer = ActorMaterializer() import system.dispatcher val todos = Seq( Todo("1", "Record amazing gifs for the tutorials", "", done = false), Todo("2", "Finish the spray-json tutorial", "", done = true), ) val todoRepository = new InMemoryTodoRepository(todos) import akka.http.scaladsl.server.Directives._ def route = path("todos") { get { complete(todoRepository.all()) } ~ post { entity(as[Todo]) { todo => complete(todoRepository.save(todo)) } } } val binding = Http().bindAndHandle(route, host, port) binding.onComplete { case Success(_) => println("Success!") case Failure(error) => println(s"Failed: ${error.getMessage}") } import scala.concurrent.duration._ Await.result(binding, 3.seconds)}
With this in place, we can now move to support JSON parsing.
有了这个,我们现在可以支持JSON解析了。
The project shouldn’t be compiling right now because Akka HTTP doesn’t know how to convert JSON to our models and vice versa.
该项目不应立即编译,因为Akka HTTP不知道如何将JSON转换为我们的模型,反之亦然。
Adding JSON support with circe was quite simple. It only involved adding a couple of import statements.
使用circe添加JSON支持非常简单。 它只涉及添加几个导入语句。
Sadly, with spray-json that isn’t the case. The effort isn’t that great either.
可悲的是,使用Spray-json并非如此。 努力也不是很好。
So, let’s start.
所以,让我们开始吧。
Because we want to use spray-json with Akka HTTP, we can look at the Akka HTTP’s official docs on how to accomplish what we want.
因为我们要在Akka HTTP上使用spray-json,所以我们可以查看Akka HTTP的官方文档 ,了解如何完成我们想要的。
We need to extend the trait SprayJsonSupport
to let Akka HTTP know how to parse our models to and from JSON (via the FromEntityUnmarshaller
and ToEntityMarshaller
provided by the trait).
我们需要扩展特征SprayJsonSupport
以使Akka HTTP知道如何与JSON解析模型(通过ToEntityMarshaller
提供的FromEntityUnmarshaller
和ToEntityMarshaller
)。
And to create the actual format, we will use the trait DefaultJsonProtocol
from spray-json.
为了创建实际的格式 ,我们将使用spray-json中的特征DefaultJsonProtocol
。
Add the following object below the Todo
model:
在Todo
模型下面添加以下对象:
object TodoFormat extends SprayJsonSupport with DefaultJsonProtocol { implicit val todoFormat = jsonFormat4(Todo)}
This is the extra step we need when using spray-json. It has to be done for every model we have.
这是使用spray-json时需要的额外步骤。 必须为我们拥有的每个模型完成此操作。
To get our project working, we need to import TodoFormat
in our Main
object:
为了使我们的项目正常工作,我们需要在我们的Main
对象中导入TodoFormat
:
import TodoFormat._import akka.http.scaladsl.server.Directives._def route = path("todos") { get { complete(todoRepository.all()) } ~ post { entity(as[Todo]) { todo => complete(todoRepository.save(todo)) } }}
Run the application and it should be working fine.
运行该应用程序,它应该可以正常工作。
Let’s make some tests!
让我们做一些测试!
We need to make sure our API is working as intended. So let’s query it as we did in the previous tutorial to check the functionality is the same.
我们需要确保我们的API能够按预期工作。 因此,让我们像上一教程中那样查询它,以检查功能是否相同。
Sending a GET request to localhost:9000/todos
should give us the initial todos:
发送GET请求到localhost:9000/todos
应该给我们初始的待办事项:
Great, that works the same.
太好了,工作原理是一样的。
Let’s see if sending invalid JSON gives us something similar:
让我们看看是否发送无效的JSON给我们带来了类似的效果:
It does. The error message is different but we get the same 400 Bad Request
which is the important part.
是的 错误消息是不同的,但我们得到相同的400 Bad Request
,这是重要的部分。
Let’s create a new todo with valid JSON:
让我们使用有效的JSON创建一个新的待办事项:
And to finish off, let’s query the todos again to make sure it was saved:
最后,让我们再次查询待办事项以确保已保存:
There we go. We have a working application with spray-json.
好了 我们有一个使用Spray-json的应用程序。
Cool, isn’t it?
不错,不是吗?
Even though working with spray-json involves some extra manual work, you don’t need an additional third-party dependency to get it to work with Akka HTTP.
即使使用spray-json涉及一些额外的手动工作,您也不需要额外的第三方依赖关系即可使其与Akka HTTP一起使用。
It’s a matter of preference really.
这确实是一个优先事项。
In the future, we will explore how to accomplish different use cases with both to compare them. So stay tuned!
将来,我们将探索如何完成不同的用例并将它们进行比较。 敬请期待!
If you liked this tutorial and wanted to learn how to build an API for a todo application, check out our new free course! ???
如果您喜欢本教程,并且想学习如何为待办事项应用程序构建API,请查看我们的新免费课程! ???
Akka HTTP QuickstartLearn how to create web applications and APIs with Akka HTTP in this free course!link.codemunity.io
Akka HTTP快速入门 在此免费课程中,了解如何使用Akka HTTP创建Web应用程序和API! link.codemunity.io
Originally published at www.codemunity.io.
最初在www.codemunity.io上发布。
翻译自: https://www.freecodecamp.org/news/how-to-un-marshal-json-in-akka-http-with-spray-json-1407876373a7/
spray.json