4

我需要完全改变字段类型。

情况:我有一个结合了多种服务的网关。Query: { ServiceName: { serviceQuery } }它通过内省获取它们的模式,通过向类型添加名称空间(使用 apollo graphql-tools)来转换它们,并通过构建具有ServiceNameQuery类型和设置delegate解析器的新模式将它们放入嵌套查询(例如)中。之后,我需要创建跨服务链接。

我有ServiceA一个带有属性的类型related,其中包含来自TargetTypefrom 的ID ServiceB。我需要将属性类型relatedIDto更改为TargetType并添加一个解析器,它将从ServiceB.

这是一个真实的例子: 在此处输入图像描述

为此,我在合并模式上运行visitSchema(从转换),找到我需要的字段,从模式中获取并更改字段类型并设置如下解析器:graphql-toolsTargetType

const targetType = schema.getType('TargetType');
field.type = targetType;
field.astNode.type = targetType.astNode;
field.resolve = (parent, args, ctx, info) => {
    console.log(parent);
    return null;
  };

当我在将结果模式传递给 graphql 服务器之前运行 console.log 类型时 - 我看到它的类型和 astNode 确实发生了变化并且它们是正确的。

当我运行自省时——我发现这种类型的字段是正确的。

但是当我运行一个查询时,我query { ServiceA { serviceQuery { id related { id } } } }在请求解析阶段遇到错误:

Error: Field "related" must not have a selection since type "ID" has no subfields.

GraphQL request (5:19)
4:       id
5:       related {
                     ^
6:         id

    at asErrorInstance (/.../node_modules/graphql/execution/execute.js:555:43)
    at process._tickCallback (internal/process/next_tick.js:68:7)

它事件不会尝试调用serviceQuery解析器。

当我更改类型定义时,它从哪里获取该字段的 ID 类型?我还需要在哪里更改它的类型?

UPD 即使我创建这样的新类型:

schema = return visitSchema(schema, {
    [VisitSchemaKind.OBJECT_TYPE](type: GraphQLObjectType) {
      if (type.getFields().relatedNews) {
        // console.log(type.getFields());
        return new GraphQLObjectType({
          name: type.name,
          fields: {
            ...fromPairs(Object.values(type.getFields()).map(({name, type, args, resolve, description}) => [name, {type, args: {}, resolve, description}])),
            relatedNews: {
              type: schema.getType('NewsArticle'),
              resolve(obj) {
                console.log('foo');
              }
            }
          }
        });
      }
      return type;
    },
  });

我不断收到此错误,并且未触发解析器。如果我以同样的方式添加一个具有不同名称的新字段 - 就可以了。但如果名称相同 - 错误

4

1 回答 1

1

好的,我想通了。缝合模式跟踪底层缝合模式。在顶级架构中更改类型不会影响它,因此会导致内部类型冲突。所以类型应该在合并之前改变

于 2018-06-27T13:19:55.817 回答