Movatterモバイル変換


[0]ホーム

URL:


Skip to content
DEV Community
Log in Create account

DEV Community

Microsoft Azure profile imageAaron Powell
Aaron Powell forMicrosoft Azure

Posted on • Originally published ataaron-powell.com on

     

GraphQL on Azure: Part 8 - Logging

As we’ve been looking at how to run GraphQL on Azure we’ve covered several topics of importance with Azure integration, but what we haven’t looked at is how we make sure that we are getting insights into our application so that if something goes wrong, we know about it. So for this post we’re going to address that as we take a look at logging using theAzure Application Insights platform (often referred to as AppInsights).

If you’re deploying into Azure in the ways that we’ve looked at in this series, chances are you’re already using AppInsights, as it’s the cornerstone of Azure’s monitoring platform, so let’s look at how to get better insights out of our GraphQL server.

Side note: There’s a lot more you can do with AppInsights in monitoring your infrastructure, monitoring across resources, etc., but that’ll be beyond the scope of this article.

Tracing Requests

Apollo has aplugin system that allows us to tap into the life cycle of the server and requests it receives/responds to, so that we can inspect them and operate against them.

Let’s have a look at how we have create some tracing through therequest life cycle with a custom plugin.

We’ll need theapplicationinsights npm package, since this is a Node.js app and not client side (there’s different packages depending if you’re doing server or client side JavaScript).

I’m also going to use theuuid package to generate a GUID for each request, allowing us to trace the events within a single request.

Let’s get started coding:

import{ApolloServerPlugin,GraphQLSchemaContext,GraphQLServerListener}from"apollo-server-plugin-base";import{TelemetryClient}from"applicationinsights";import{v4asuuid}from"uuid";exportdefaultfunction(input:string|TelemetryClient,logName?:string):ApolloServerPlugin{letclient:TelemetryClient;if(typeofinput==="string"){client=newTelemetryClient(input);}else{client=input;}return{};}
Enter fullscreen modeExit fullscreen mode

Here’s the starting point. I’m making this a generic plugin that you can either pass in the Instrumentation Key for AppInsights, or an existingTelemetryClient (the thing you create using the npm package), which allow you create a unique client or share it with the rest of your codebase. I’ve also added an optionallogName argument, which we’ll put in each message for easy querying.

Time to hook into our life cycle:

exportdefaultfunction(input:string|TelemetryClient,logName?:string):ApolloServerPlugin{letclient:TelemetryClient;if(typeofinput==="string"){client=newTelemetryClient(input);}else{client=input;}return{requestDidStart(context){constrequestId=uuid();constheaders:{[key:string]:string|null}={};if(context.request.http?.headers){for(const[key,value]ofcontext.request.http.headers){headers[key]=value;}}client.trackEvent({name:"requestDidStart",time:newDate(),properties:{requestId,metrics:context.metrics,request:context.request,headers,isDebug:context.debug,operationName:context.operationName,operation:context.operation,logName}});}};}
Enter fullscreen modeExit fullscreen mode

TherequestDidStart method will receive aGraphQLRequestContext which has a bunch of useful information about the request as Apollo has understood it, headers, the operation, etc., so we’re going to want to log some of that, but we’ll also enrich it a little ourselves with arequestId that will be common for allow events within this request and thelogName, if provided.

You might be wondering why I’m doingheaders in the way I am, that’s becausecontext.request.http.headers is anIterable and won’t get serialized properly, so we need to convert it into a standard object if we want to capture them.

We send this off to AppInsights usingclient.trackEvent:

client.trackEvent({name:"requestDidStart",time:newDate(),properties:{requestId,metrics:context.metrics,request:context.request,headers,isDebug:context.debug,operationName:context.operationName||context.request.operationName,operation:context.operation,logName}});
Enter fullscreen modeExit fullscreen mode

Thename for the event will help us find the same event multiple times, so I’m using the life cycle method name,requestDidStart, and popping the current timestamp on there. Since I’m usingtrackEvent this will appear in thecustomEvents table within AppInsights, but you could usetrackTrace or any of the other tables for storage, depending on how you want to query and correlate your logs across services.

Example log view in AppInsights

This is an example of how that will appear in AppInsights, you can see the custom information we’ve pushed, such as the GraphQL operation and it’s name, the headers, etc.

We could then write a query against the table for all operations namedTestQuery:

customEvents| extend req = todynamic(tostring(customDimensions.["request"]))| where req.operationName == 'TestQuery'
Enter fullscreen modeExit fullscreen mode

The plugin can then be expanded out to cover each of the life cycle methods, pushing the relevant information to AppInsights, and allowing you to understand the life cycle of your server anf requests.

Conclusion

This is a really quick look at how we can integrateAzure Application Insights into the life cycle of Apollo Server and get some insights into the performance of our GraphQL server.

I’ve created aGitHub repo with this plugin, and it’savailable on npm.

GitHub logo aaronpowell / apollo-graphql-appinsights

An example of how to integrate AppInsights into Apollo GraphQL

AppInsights for Apollo GraphQL

This repo contains two integrations betweenApplication Insights (AppInsights) andApollo Server.

Refer to the readme's within the individual packages for their specific usage, and theexample folder contains a example of how to use them.






There’s another package in the repo,apollo-server-logger-appinsights, which provides ageneric logger for Apollo, so that any logging Apollo (or third-party plugins) does will be pushed to AppInsights.

Happy monitoring!

Top comments(0)

Subscribe
pic
Create template

Templates let you quickly answer FAQs or store snippets for re-use.

Dismiss

Are you sure you want to hide this comment? It will become hidden in your post, but will still be visible via the comment'spermalink.

For further actions, you may consider blocking this person and/orreporting abuse

Invent with purpose

Any language. Any platform.

More fromMicrosoft Azure

DEV Community

We're a place where coders share, stay up-to-date and grow their careers.

Log in Create account

[8]ページ先頭

©2009-2025 Movatter.jp