记录每个请求的Apollo服务器GraphQL查询和变量

时间:2019-01-20 02:52:30

标签: apollo-server

使用apollo-server 2.2.1或更高版本时,如何为每个请求,查询和变量记录一个日志?

这似乎是一个简单的要求和常见用例,但是文档是very vague,并且传递给query no longerformatResponse对象具有{{1} }和queryString属性。

4 个答案:

答案 0 :(得分:4)

Amit的答案(今天)有效,但是恕我直言,它有点不可靠,将来可能无法按预期工作,或者在某些情况下可能无法正常工作。

例如,当我看到它时,我想到的第一件事是:“如果查询无效,则可能不起作用”,事实证明,当查询无效时, today 确实可以工作。因为在当前的实现中,上下文是在验证查询之前评估的。但是,这是一个实现细节,将来可能会更改。例如,如果有一天阿波罗团队决定仅在查询被解析和验证之后评估上下文,这将是性能上的胜利?这正是我所期望的:-)

我要说的是,如果您只是想快速记录一些东西以便在您的dev环境中进行调试,那么Amit的解决方案无疑是解决之道。

但是,如果您想要为生产环境注册日志,那么使用context函数可能不是最好的主意。在这种情况下,我将安装graphql-extensions并将其用于日志记录,例如:

const { print } = require('graphql');

class BasicLogging {
  requestDidStart({queryString, parsedQuery, variables}) {
    const query = queryString || print(parsedQuery);
    console.log(query);
    console.log(variables);
  }

  willSendResponse({graphqlResponse}) {
    console.log(JSON.stringify(graphqlResponse, null, 2));
  }
}

const server = new ApolloServer({
  typeDefs,
  resolvers,
  extensions: [() => new BasicLogging()]
});

编辑:

如Dan所指出的,由于graphql-extensions软件包已集成在apollo-server-core软件包中,因此无需安装。

答案 1 :(得分:3)

Dan的解决方案主要解决了该问题,但是如果您想在不使用Express的情况下进行记录, 您可以在以下示例中显示的上下文中捕获它。

const server = new ApolloServer({
schema,
context: params => () => {
    console.log(params.req.body.query);
    console.log(params.req.body.variables);
}
});

答案 2 :(得分:3)

使用new plugins API,您可以使用与Josep答案非常相似的方法,除了您的代码结构略有不同。

const BASIC_LOGGING = {
    requestDidStart(requestContext) {
        console.log("request started");
        console.log(requestContext.request.query);
        console.log(requestContext.request.variables);
        return {
            didEncounterErrors(requestContext) {
                console.log("an error happened in response to query " + requestContext.request.query);
                console.log(requestContext.errors);
            }
        };
    },

    willSendResponse(requestContext) {
        console.log("response sent", requestContext.response);
    }
};


const server = new ApolloServer(
    {
        schema,
        plugins: [BASIC_LOGGING]
    }
)

server.listen(3003, '0.0.0.0').then(({ url }) => {
    console.log(`GraphQL API ready at ${url}`);
});

答案 3 :(得分:2)

如果必须记录查询和变量,我可能会使用apollo-server-express而不是apollo-server,以便可以在记录日志的graphql之前添加一个单独的快速中间件。对我来说:

const express = require('express')
const { ApolloServer } = require('apollo-server-express')
const { typeDefs, resolvers } = require('./graphql')

const server = new ApolloServer({ typeDefs, resolvers })
const app = express()

app.use(bodyParser.json())
app.use('/graphql', (req, res, next) => {
  console.log(req.body.query)
  console.log(req.body.variables)
  return next()
})

server.applyMiddleware({ app })

app.listen({ port: 4000}, () => {
  console.log(` Server ready at http://localhost:4000${server.graphqlPath}`)
})