😀
ExpressのGraphQLで相互のRelation関係にあるTypeをimportする方法
ExpressでGraphQLを実装している時に、TypeでRelationの関係を作ろうとして
Error: Expected undefined to be a GraphQL type
というエラーに出会いました。
原因はType同士でimportしていたために、一方のTypeがundefinedになってしまうことです。
この問題に対応してみました。
GraphQLのType同士で循環参照する場合
まず今回エラーが発生したコードは書きのようなものです。
それぞれのTypeからお互いをimportしています。
import UserType from './UserType';
const PostType = new GraphQLObjectType({
name: 'Post',
fields: {
title: { type: GraphQLString },
user: { type: UserType }
}
});
export default PostType;
import PostType from './PostType';
const UserType = new GraphQLObjectType({
name: 'User',
fields: {
name: { type: GraphQLString },
posts: { type: new GraphQLList(PostType) }
}
});
export default UserType;
お互いをimportしている循環参照の関係にあるため、当然ですがこのコードでは先に読み込まれる方が、読み込み先のTypeをundefinedとして認識します。
ただGraphQLでTypeを定義しているとこういうコードは避けられないと思う一方で、どのような風に対策するのがよいのか。
fieldsではオブジェクトではなく関数で返す
fieldsはオブジェクトではなく、オブジェクトを関数を指定できます。
関数で返すことをfieldsの部分が遅延評価されるようになり、先に書いたように読み込んだ時点ではundefinedでもエラーが発生しなくなります。
コードとしては下記のような形です。
const PostType = new GraphQLObjectType({
name: 'Post',
fields: {
title: { type: GraphQLString },
user: { type: UserType }
}
});
export default PostType;
const UserType = new GraphQLObjectType({
name: 'User',
fields: {
name: { type: GraphQLString },
posts: { type: new GraphQLList(PostType) }
}
});
export default UserType;
わかってしまえばなんてことないけど、少しハマってしまいました・・。
Discussion