如何在不创建Mongoose模型的情况下将GraphQL与Mongoose和MongoDB一起使用

时间:2017-12-05 18:54:03

标签: mongodb mongoose graphql

在Mongoose中创建模型是没有意义的,因为这些模型已经使用GraphQL和现有结构(即TypeScript接口)创建。

我们怎样才能让GraphQL在GraphQL提供的模型上使用Mongoose的操作而无需在Mongoose中重新创建模型?

此外,似乎应该有一个GraphQL包装器只与数据库通信,避免编写MyModel.findById等

如何做到这一点?

互联网上谈论GraphQL和Mongodb的每个例子都使用Mongoose。

1 个答案:

答案 0 :(得分:2)

你应该看看GraphQL-to-MongoDB, or how I learned to stop worrying and love generated query APIs。它讨论了一个中间件包,它利用GraphQL的类型生成GraphQL API并解析从客户端发送到MongoDB查询的请求。它或多或少地超过了Mongoose。

免责声明:这是我的博文。

package为您的架构字段args生成GraphQL输入类型,并包装resolve函数以将它们解析为MongoDB查询。

给出一个简单的GraphQLType:

const PersonType = new GraphQLObjectType({
    name: 'PersonType',
    fields: () => ({
        age: { type: GraphQLInt },
        name: {
            type: new GraphQLNonNull(new GraphQLObjectType({
                name: 'NameType',
                fields: () => ({
                    firstName: { type: GraphQLString },
                    lastName: { type: GraphQLString }
                })
            }))
        }
    })
}); 

对于最常见的用例,您将使用getMongoDbQueryResolvergetGraphQLQueryArgs在GraphQL架构中构建一个字段。包装器提供的filterprojectionoptions可以直接传递给find函数。

person: {
    type: new GraphQLList(PersonType),
    args: getGraphQLQueryArgs(PersonType),
    resolve: getMongoDbQueryResolver(PersonType,
        async (filter, projection, options, source, args, context) =>
            await context.db.collection('person').find(filter, projection, options).toArray()
    )
}

您可以发送到此类字段的查询示例:

{
    person (
        filter: {
            age: { GT: 18 },
            name: { 
                firstName: { EQ: "John" } 
            }
        },
        sort: { age: DESC },
        pagination: { limit: 50 }
    ) {
        name { 
            lastName
        }
        age
    }
}

还有一个用于变异字段的包装器和参数类型生成器。