GraphQL-如果父级返回类型是对象列表,则无法进行关于模式缝合的模式委托

时间:2018-07-16 15:13:34

标签: node.js graphql delegation express-graphql graphql-tools

我有一个Publication对象,其中包含一个imageId,我想使用它来创建模式委托并加载整个对象。

出版物类型定义为:

const PublicationType = new GraphQLObjectType({
name: 'Publication',
description: 'Publication Type',
fields: () => ({
id: {
type: GraphQLID
},
callToAction: {
type: CallToActionType
},
userId: {
type: GraphQLID
},
groupId: {
type: GraphQLID
},
instruction: {
type: InstructionType
},
customizedTitle: {
type: GraphQLString
},
content: {
type: GraphQLString
},
activityType: {
type: ActivityType
},
creationTool: {
type: CreationTool
},
gift: {
type: GiftType
},
imageId: {
type: GraphQLInt
},
thumbnailId: {
type: GraphQLInt
},
createdAt: {
type: GraphQLInt
},
updatedAt: {
type: GraphQLInt
}
})
});

媒体类型定义为:

const MediaType = new GraphQLObjectType({
name: 'Media',
description: 'Media Object',
fields: () => ({
id: {
type: GraphQLInt
},
resource: {
type: GraphQLString
},
metadata: {
type: MetadataType
},
discr: {
type: GraphQLString
}
})
});

其中MetadataType是代码中定义的另一种类型。

这两个对象通过这些查询公开:

this._schema = new GraphQLObjectType({
name: 'Query',
description: 'Wall Query',
fields: () => ({
getGroupPublications: {
type: new GraphQLList(PublicationType),
args: {
groupId: {type: new GraphQLNonNull(GraphQLInt)},
from: {type: new GraphQLNonNull(GraphQLInt)},
to: {type: new GraphQLNonNull(GraphQLInt)},
},
resolve: (root, args, context) => PublicationResolver.groupPublications(context)
}
}),
});

 this._schema = new GraphQLObjectType({
              name: 'Query',
              description: 'Media Query',
              fields: () => ({
                  getMedia: {
                      type: MediaType,
                      args: {
                          mediaId: {type: new GraphQLNonNull(GraphQLInt)}
                      },
                      resolve: (root, args, context) => MediaResolver.mediaLoader(context).load(args.mediaId)
                  }
              }),
          });

现在我正在尝试将所有定义的架构合并到一个架构中,并添加一个委托以加载Media类型而不是ImageID,以便下一个类代表我将公开的主要架构:

import {mergeSchemas} from 'graphql-tools';
// other imports ...

class Services {

    private static _instance: Services;
    private _wall: PublicationService;
    private _media: MediaService;

    private constructor() {
        this._media = new MediaService();
        this._wall = new PublicationService();
    }

    public static Instance() {
        return this._instance || (this._instance = new this())
    }

    public MainSchema = async () => Promise.all([this._publication.schema, this._media.schema])
            .then(args => {
                const publicationSchema = args[0];
                const mediaSchema = args[1];

                const linkSchemaDefs = `
                                extend type Publication {
                                    image: Media
                                    thumbnail: Media
                                }`;

                return mergeSchemas({
                    schemas: [publicationSchema, mediaSchema, linkSchemaDefs],
                    resolvers: 
                        {
                            Publication: {
                                image: {
                                    fragment: `... on Publication { imageId }`,
                                    resolve(parent, args, context, info) {
                                        let mediaId = (parent.imageId === parseInt(parent.imageId, 10)) ? parent.imageId : 0;
                                        return info.mergeInfo.delegateToSchema({
                                            schema: mediaSchema,
                                            operation: 'query',
                                            fieldName: 'getMedia',
                                            args: {
                                                mediaId: mediaId,
                                            },
                                            context,
                                            info
                                        })
                                    }
                                },
                                thumbnail: {
                                    fragment: 'fragment PublicationThumbnailFragment on Publication { thumbnailId }',
                                    resolve(parent, args, context, info) {
                                        let mediaId = (parent.thumbnailId === parseInt(parent.thumbnailId, 10)) ? parent.thumbnailId : 0;
                                        return info.mergeInfo.delegateToSchema({
                                            schema: mediaSchema,
                                            operation: 'query',
                                            fieldName: 'getMedia',
                                            args: {
                                                mediaId: mediaId,
                                            },
                                            context,
                                            info
                                        })
                                    }
                                },
                            },
                        },
                });
            })
            .catch(error => Logger.Instance().error({
                    errorName: 'SchemaStitchingError',
                    logName: 'GraphQL Schema Stitching Error',
                    cause: error,
                    message: [`Services GraphQl Schema stitching rejected`],
                })
            );
}

export {Services}

将使用graphqlExpress Server向最终用户公开主要架构 但是,我可以在模式定义中看到扩展类型。

Publication Type Schema Description

但是当我尝试使用定义为的查询来查询媒体类型的字段图像时

query {
getGroupPublications(groupId:2, from: 1531180800,to: 1532276998) {
id
imageId
image {
resource
}
createdAt
}
}

该查询将直接命中getGroupPublications出版物,而无需考虑mergeSchema,并且出现此错误:

{ errors:
[ { Cannot query field "image" on type "Publication". Did you mean "imageId"?

GraphQL request (5:3)
4: imageId
5: image {
^
6: resource

at Object.Field (/home/docker/workspace/api-gateway/node_modules/graphql/validation/rules/FieldsOnCorrectType.js:65:31)
at Object.enter (/home/docker/workspace/api-gateway/node_modules/graphql/language/visitor.js:324:29)
at Object.enter (/home/docker/workspace/api-gateway/node_modules/graphql/language/visitor.js:366:25)
at visit (/home/docker/workspace/api-gateway/node_modules/graphql/language/visitor.js:254:26)
at visitUsingRules (/home/docker/workspace/api-gateway/node_modules/graphql/validation/validate.js:74:22)
at validate (/home/docker/workspace/api-gateway/node_modules/graphql/validation/validate.js:59:10)
at graphqlImpl (/home/docker/workspace/api-gateway/node_modules/graphql/graphql.js:106:50)
at /home/docker/workspace/api-gateway/node_modules/graphql/graphql.js:66:223
at new Promise ()
at Object.graphql (/home/docker/workspace/api-gateway/node_modules/graphql/graphql.js:63:10)
message: 'Cannot query field "image" on type "Publication". Did you mean "imageId"?',
locations: [Array],
path: undefined } ] }

但是如果我在没有字段图像的情况下运行查询,效果很好

query {
getGroupPublications(groupId:1, from: 1531180800,to: 1532276998) {
id
imageId
}
}

1 个答案:

答案 0 :(得分:0)

我遇到的问题是由于将完整的查询传递给远程模式,因此MergeSchema Resolver仅在远程模式返回结果后才执行。

因此,在getGroupPublications解析器myswitch.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() { @Override public void onCheckedChanged(CompoundButton compoundButton, boolean isChecked) { if (isChecked){ PreferenceManager .getDefaultSharedPreferences(DisplaySettingActivity.this) .edit() .putInt("ActivityTheme", R.style.darktheme) .commit(); DisplaySettingActivity.this.recreate(); } 中,我通过扩展类型替换了添加的字段。

PublicationResolver.groupPublications(context)