Skip to content

Khan/genqlient

Repository files navigation

generated graphql client ⇒ genqlient

genqlient: a truly type-safe Go GraphQL client

This is a proof-of-concept of using code-generation to create a truly type-safe GraphQL client in Go. It is certainly not ready for production use nor or contributions (see below).

Why another GraphQL client?

To understand the issue, consider the example from the documentation of shurcooL/graphql:

// to send a query `{ me { name } }`:
var query struct {
	Me struct {
		Name graphql.String
	}
}
// error handling omitted for brevity
client.Query(context.Background(), &query, nil)
fmt.Println(query.Me.Name)
// Output: Luke Skywalker

While this code may seem type-safe, and at the Go level it is, there's nothing to check that the schema looks like you expect it to. In fact, perhaps here we're querying the GitHub API, in which the field is called viewer, not me, so this query will fail. More common than misusing the name of the field is mis-capitalizing it, since Go and GraphQL have somewhat different conventions there. And even if you get it right, it adds up to a lot of handwritten boilerplate! And that's the best case; other clients, such as machinebox/graphql, have even fewer guardrails to help you make the right query and use the result correctly. This isn't a big deal in a small application, but for serious production-grade tools it's not ideal.

These problems should be entirely avoidable: GraphQL and Go are both typed languages; and GraphQL servers expose their schema in a standard, machine-readable format. We should be able to simply write a query { viewer { name } }, have that automatically validated against the schema and turned into a Go struct which we can use in our code. In fact, there's already good prior art to do this sort of thing: 99designs/gqlgen is a popular server library that generates types, and Apollo has a codegen tool to generate similar client-types for several other languages. (See DESIGN.md for more prior art.)

This is a GraphQL client that does the same sort of thing: you specify the query, and it generates type-safe helpers that make your query.

Usage

Example

To use genqlient, put your GraphQL schema (in SDL format) in a file schema.graphql, and put a query like the following in queries.graphql:

# queries.graphql
query getViewer {
  viewer {
    name
  }
}

Then run genqlient (go run github.com/Khan/genqlient), and it will generate:

// generated.go (auto-generated):
type getViewerResponse struct { ... }
func getViewer(ctx context.Context, client *graphql.Client) (*getViewerResponse, error) { ... }

Finally, write your code to call genqlient, like so:

// your code (error handling omitted for brevity)
graphqlClient := graphql.NewClient("https://example.com/graphql", nil)
viewerResp, _ := getViewer(context.Background(), graphqlClient)
fmt.Println("you are", viewerResp.Viewer.MyName)

//go:generate go run github.com/Khan/genqlient

For a complete working example, see example/. For configuration options, see go doc github.com/Khan/genqlient/generate.Config.

Documentation for generated code

For each GraphQL operation (query or mutation), genqlient generates a Go function with the exact same name, which accepts:

  • a context.Context (unless configured otherwise)
  • a genqlient/graphql.Client (you might have a global one, or init it inline)
  • arguments corresponding to the query arguments

It returns a pointer to a struct representing the query-result, and an error. The struct will always be initialized (never nil), even on error. The error may be a github.com/vektah/gqlparser/v2/gqlerror.List, if it was a GraphQL-level error (in this case the returned struct may still contain useful data, if the API returns data even on error), or may be another error if, for example, the whole HTTP request failed (in which case the struct is unlikely to contain useful data). If the GraphQL operation has a comment immediately above it, that comment text will be used as the GoDoc for the generated function.

The generated code may be customized using a directive-like syntax, # @genqlient(...). For full documentation of options, see go doc github.com/Khan/genqlient/generate.GenqlientDirective.

TODO: consider inlining the direct stuff; and document generated types further.

Development

genqlient is not yet accepting contributions. The library is still in a very rough state, so while we intend to accept contributions in the future, we're not ready for them just yet. Please contact us (via an issue or email) if you are interested in helping out, but please be aware the answer may be that we aren't ready for your help yet, even if such help will be greatly useful once we are! We'll be ready for the rest of the world soon.

Khan Academy is a non-profit organization with a mission to provide a free, world-class education to anyone, anywhere. If you're looking for other ways to help us, You can help us in that mission by donating or looking at career opportunities.

Tests

go test ./... tests code generation. (This is run by GitHub Actions.) Most of the tests are snapshot-based; see generate/generate_test.go. If GITHUB_TOKEN is available in the environment, it also checks that the example returns the expected output when run against the real API. This is configured automatically in GitHub Actions, but you can also use a personal access token with no scopes.

Design

See DESIGN.md for documentation of major design decisions in this library.

Major TODOs

(+) denotes things we further need before recommending anyone else use this in prod

Basic features:

  • redo support for interfaces, unions, fragments (see DESIGN)

Fancy features:

  • allow __all to request all fields (probably a terrible idea though)
  • plugin to make _entities queries for apollo federation easy (basically you would just write the fragment)
  • auto-caching -- if you @genqlient(cache: true) we cache (you can pass a cache into the client or something)

Generated code customization:

  • add flag(s) to make a field use a pointer (for optionality or perf; see DESIGN)
  • collapsing -- should be able to have mutation { myMutation { error { code } } } just return (code string, err error)
  • map a field to a particular Go type (if you want to use a named type for some string, say)
  • specify a particular name for a Go type, or for the helper function, or whatever else
  • include full query in generated godoc

Config options:

  • proper config/arguments setup (e.g. with viper)
  • (+) improve client_getter to be more usable (and document it), or flag it out for now.
  • get schema via GraphQL introspection (although honestly, it seems like SDL is becoming popular enough there may be no need)
  • whether names should be exported
  • default handling for optional fields? (maybe generate a HasFoo, you can always ignore if you don't care)
  • generate mocks?

Runtime:

  • (+) basic tests for graphql package
  • integration tests against common servers for graphql package
  • send hash rather than full query

Other:

  • (+) API cleanup (check godoc)
  • (+) improved validation and error checking
  • (+) documentation
  • get a designer to fix my bad logo-thing