2

该问题的摘要已在 Github 问题中进行了描述:

https://github.com/apollographql/apollo-server/issues/2794


简而言之,有没有办法实现 Apollo Federation,让网关本身有自己的模式?

联合 Apollo 服务器,其中网关本身具有模式

@apollo/gateway@0.6.5 @apollo/federation@0.6.2

预期行为

当我实例化我的 Apollo 网关服务器时,我希望我应该能够合并来自联合服务的模式,以及来自网关本身的模式。

实际行为

网关服务器无法挂载/graphql路由,因为它期望所有服务在它这样做之前当前正在运行。

在运行时,控制台会打印以下错误:

POST /graphql 404 11.251 ms - 147
Encountered error when loading gateway-app at http://localhost:8000/graphql: invalid json response body at http://localhost:8000/graphql reason: Unexpected token < in JSON at position 0
[DEBUG] Fri Jun 07 2019 12:11:07 GMT-0400 (Eastern Daylight Time) apollo-gateway: Configuration loaded for Gateway
[DEBUG] Fri Jun 07 2019 12:11:07 GMT-0400 (Eastern Daylight Time) apollo-gateway: Composing schema from service list:
  remove-svc
TypeError: schema.toConfig is not a function
    at Object.composeServices (/gateway-app/node_modules/@apollo/federation/dist/composition/compose.js:191:67)
    at Object.composeAndValidate (/gateway-app/node_modules/@apollo/federation/dist/composition/composeAndValidate.js:13:41)
    at ApolloGateway.createSchema (/gateway-app/node_modules/@apollo/gateway/dist/index.js:90:47)
    at ApolloGateway.<anonymous> (/gateway-app/node_modules/@apollo/gateway/dist/index.js:81:22)
    at Generator.next (<anonymous>)
    at fulfilled (/gateway-app/node_modules/@apollo/gateway/dist/index.js:4:58)
    at process._tickCallback (internal/process/next_tick.js:68:7)

源代码

const url = new URL(`redis://${REDIS_URL}:${REDIS_PORT}/${REDIS_DATABASE}`).toString()

const cache = new RedisCache({ url })

const context = ({ req }) => {
  if (!(req.user || req.headers.authorization === ROUTE_AUTH)) {
    throw new AuthenticationError('Not authenticated')
  }
  return { user: req.user, req: req }
}

try {
  const loadGateway = async () => {
    try {
      const { schema, executor } = await gateway.load()
      return { schema, executor }
    } catch (err) {
      console.error(err)
      return null
    }
  }

  const gateway = new ApolloGateway({
    debug: process.env.ENV !== 'prod',
    serviceList: [
      { name: 'gateway', url: `${GATEWAY_HOSTNAME}/graphql` },
      { name: 'remote-svc', url: `${REMOTE_SERVICE_HOSTNAME}/graphql` },
    ],
  })

  const { schema, executor } = loadGateway()
  const server = new ApolloServer({
    schema,
    executor,
    cache,
    dataSources,
    engine: { apiKey: ENGINE_API_KEY },
    tracing: true,
    context,
  })
  server.applyMiddleware({ app })
} catch (err) {
  console.error(err)
}
4

1 回答 1

3

根据我的问题报告的原始错误,有两个错误。

  1. Apollo 网关服务器不能也不应该是具有自己模式的联合服务(参见Github 问题 #2792)。

从阿波罗出发:

我将根据您的服务的命名在这里得出一个结论:您是否尝试在端口 8000 上运行网关并同时将其视为联合服务?如果是这样,那是行不通的;网关必须先解析所有模式,然后才能组成最终模式。

  1. GraphQL 核心依赖项必须至少是版本14.2.0(请参阅Github 问题 #2825)。

来自 GraphQL JS 变更日志:

toConfig 旨在通过最小化您需要编写和维护的代码量来帮助模式转换,例如来自 graphql-tools 的 recreateType。

当我在 graphql-voyager 工作时,我创建了自己的中间格式,只是为了进行模式转换。

此外,toConfig 将防止添加新字段时出现的错误。例如,lexicographicSortSchema 不尊重新添加的assumeValid 和allowedLegacyNames 字段。

在将模式从网关迁移到它自己的联合服务并将我的核心 GraphQL 依赖项更新到适当的版本之后,我成功地实现了网关/联合模式。

于 2019-06-11T20:46:23.390 回答