json - 带有桑格利亚汽酒的 GraphQL 模式

标签 json scala graphql sangria

我正在查看用于在 Scala 中编写 GraphQL 服务器的 Sangria 库。然而,感觉很奇怪,同一个类型系统必须实现两次:(1)作为 GraphQL 类型声明的一部分,以及(2)也在服务器端,作为 Scala 案例类,伴随着 ObjectType、InterfaceType 等。瓦尔斯

在 Scala 中对类型系统进行硬编码尤其令人讨厌,因为我的目的是能够 CRUD 聚合任意形状,其中每个形状都被定义为一个 GraphQL 类型集合。例如,假设一个 Shape 类型的实例包含一个 GraphQL 文档作为一个字段; Entity 类型的实例具有对其 S​​hape 的引用,并且还包含在该 Shape 中定义的形状的 Json 对象。

case class Shape(id: String, name: String, doc: sangria.ast.Document)
case class Entity(id: String, name: String, shape: Shape, content: JsValue)

例如,如果形状文档是这样的:
type Person {
  firstName: String!
  lastName: String!
  age: Int
}

那么实体中的 Json 内容可能是这样的:
{
  "firstName": "John",
  "lastName": "Smith",
  "age": 30
}

(当然,一个真实的例子也会有嵌套类型等)

因此,我寻求能够定义 Entity 类型的实例,其形状在其相应的形状中定义。我不想对相应的 sangria.schema.Schema 进行硬编码,而是想直接从形状文档中导出它。

有没有现成的方法可以从包含类型声明的 GraphQL 文档以编程方式生成 GraphQL 模式?

最佳答案

对于此类动态用例,sangria 提供了一种从 GraphQL IDL 构建模式的方法。这是您的操作方法(我稍微简化了您的示例,但是当所有这些数据都来自诸如 ShapeEntity 之类的单独类时,同样可以实现):

import sangria.ast._
import sangria.schema._
import sangria.macros._
import sangria.marshalling.sprayJson._
import sangria.execution.Executor

import scala.concurrent.ExecutionContext.Implicits.global
import spray.json._

val schemaAst =
  gql"""
    type Person {
      firstName: String!
      lastName: String!
      age: Int
    }

    type Query {
      people: [Person!]
    }
  """

val schema = Schema.buildFromAst(schemaAst, builder)

val query =
  gql"""
    {
      people {
        firstName
        age
      }
    }
  """

val data =
  """
    {
      "people": [{
        "firstName": "John",
        "lastName": "Smith",
        "age": 30
      }]
    }
  """.parseJson

val result = Executor.execute(schema, query, data)

为了定义如何resolve应该生成函数,您需要创建一个自定义模式构建器,就像这个一样,只需覆盖 resolveField方法:
val builder =
  new DefaultAstSchemaBuilder[JsValue] {
    override def resolveField(typeDefinition: TypeDefinition, definition: FieldDefinition) =
      typeDefinition.name match {
        case "Query" ⇒
          c ⇒ c.ctx.asJsObject.fields get c.field.name map fromJson
        case _ ⇒
          c ⇒ fromJson(c.value.asInstanceOf[JsObject].fields(c.field.name))
      }

    def fromJson(v: JsValue) = v match {
      case JsArray(l) ⇒ l
      case JsString(s) ⇒ s
      case JsNumber(n) ⇒ n.intValue()
      case other ⇒ other
    }
  }

执行此示例时,您将看到以下 JSON result :
{
  "data": {
    "people": [{
      "firstName": "John",
      "age": 30
    }]
  }
}

如果您想查看更复杂的示例,我建议您查看 GrapohQL Toolbox "proxy" .这个项目更进一步,甚至添加了自定义指令来控制解析函数的生成。代码可以在这里找到:

https://github.com/OlegIlyenko/graphql-toolbox

关于json - 带有桑格利亚汽酒的 GraphQL 模式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43665937/

相关文章:

Scala 隐式排序

websocket - Apollo GraphQL : How to Set Up Secure Websockets?

php - PHP 中的 json_encode 输出没有给数据提供标签,以便可以访问它

scala - Mono存储库的SBT构建结构

javascript - 将 JSON 响应传递给 tinymce 模板配置初始值

scala - 访问HOCON文件中的环境变量时 `?`的作用

graphql - Apollo 客户端片段未嵌入数据

javascript - 如何根据 Drupal JSON 中的术语关系在 Gatsby 中创建动态 URL :API response?

json - 如何使用 VueJS 和 AXIOS 映射 JSON 对象

jquery - IE 中存在“JSON”未定义问题,但 chrome 中没有