0

I have a Publication object that includes an imageId which I want to use to create a schema delegation and load the whole object.

Publication Type is defined as:

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
}
})
});

Media Type is defined as :

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

where MetadataType is another type defined in the code.

the 2 object are exposed via theese queries:

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)
}
}),
});

And

 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)
                  }
              }),
          });

Now Im Trying to merge all the schemas that are defined into one schema and adding a delegation to load the type Media instead of ImageID so the next class represent the main schema that I will expose:

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}

the main schema will be exposed to the end user using graphqlExpress Server However, I could see the extended type in the schema definition.

Publication Type Schema Description

But When I try to Query the field image of type media using a query defined as

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

the query will hit directly the getGroupPublications publications without taking in consideration the mergeSchema and I get this error:

{ 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 } ] }

But If I run my query without field image it works fine

query {
getGroupPublications(groupId:1, from: 1531180800,to: 1532276998) {
id
imageId
}
}
Adel Bachene
  • 974
  • 1
  • 14
  • 34

1 Answers1

0

The issue that I faced was due to passing the full Query to the remote schema, MergeSchema Resolver will execute only after the remote schema returns the results.

So In getGroupPublications resolver PublicationResolver.groupPublications(context) I replaced the fields added by extending the type.

            let queryString = context.req.body.query;
            queryString = queryString.replace(/image[\s\r\n]*?{[^}]*}/gm, "");
            queryString = queryString.replace(/thumbnail[\s\r\n]*?{[^}]*}/gm, "");
            return graphql(schema, queryString,
                null,
                context,
                context.req.body.variables)
Adel Bachene
  • 974
  • 1
  • 14
  • 34