Overview
There's one important step that we need to take care of before proceeding with our TypeScript app: we need to generate the TypeScript types to represent all of the GraphQL types in our schema!
In this lesson, we will:
- Configure the GraphQL Code Generator
- Generate types from our GraphQL API that we can use in our frontend code
Generating types
Right now, our frontend app doesn't know anything about the schema that our backend GraphQL API is using. But because we're going to be writing queries for Track
and Author
data, we need the frontend to understand what type of data they involve. We could write out the TypeScript types manually—we know from checking our Schema reference in Sandbox that a Track
has text for a title
, and an Author
has text for a name
, and so on—but if we change our schema in the future, we have to remember to update our frontend as well; this means that our frontend's TypeScript types can easily get out of sync, if we're not careful!
Instead, we can look to the GraphQL API's schema as the "single source of truth" for all of the types we could possibly query on the frontend. An easy way to do this, and to keep our frontend's type definitions consistent with the backend, is to use a GraphQL Code Generator.
@graphql-codegen/cli
is one such tool that can read in a GraphQL schema, compare it against the queries we're asking our frontend code to run, and generate all of the types that we'll need to use on the frontend. As we work on new features, we'll benefit from the clarity TypeScript gives us about what data exists on each type and what kinds of operations can be performed on it.
Codegen: an overview
Here's how the process works.
- First, we'll install our packages and set up a
generate
command to launch the Code Generator. - Next, we'll create a configuration file that guides the codegen process. At its most basic, this specifies the path to the GraphQL schema, the files in our frontend app it should inspect for GraphQL operations, and where to output the necessary TypeScript types.
- We'll run our codegen command.
- Finally, we'll check out the types it generated for us!
Ready to get started? Let's go!
Step 1: Installing @graphql-codegen/cli
Let's get started with the GraphQL Code Generator by installing it in our project. We'll also install @graphql-codegen/client-preset
, which gives us some React-specific configuration out of the box.
We only need these packages during development, so we'll install them under devDependencies
.
npm install -D @graphql-codegen/cli @graphql-codegen/client-preset
When the installation is complete, open up package.json
. Under the scripts
key, we'll add a new command, generate
, that will call the @graphql-codegen/cli
package.
"scripts": {"test": "vitest","start": "vite","build": "vite build","generate": "graphql-codegen"},
If we run the command right away with npm run generate
, we'll see that we get an error—this is because we haven't yet created the codegen.ts
file that the Code Generator looks for.
Error: Unable to find Codegen config file!Please make sure that you have a configuration file under the current directory!
Step 2: Defining the configuration file
Let's define codegen.ts
right in the root of our project.
// TODO
This file should contain a number of instructions that tell the GraphQL Code Generator what we want it to do, such as:
- Where our GraphQL API is running, so that it can retrieve information from the GraphQL API about the types and fields that exist in the schema.
- Which of our frontend files it should scan to find all of the GraphQL operations we're using (right now we don't have any!)
- Where it should output the types it generates. The GraphQL Code Generator will use all of the information we gave it to output generated types in a folder of our choosing.
We'll start by creating a new object called config
and exporting it.
const config = {};export default config;
Next, let's import the type definition we'll use for the config
object. @graphql-codegen/cli
provides one out of the box for us.
import { CodegenConfig } from "@graphql-codegen/cli";const config: CodegenConfig = {};export default config;
Within the config
object, we'll define three properties: schema
, documents
, and generates
.
The schema
property
For the schema
property, we need to pass in our GraphQL server's endpoint, https://odyssey-lift-off-server.herokuapp.com/
. The GraphQL Code Generator will look at this address and read the types and fields in the server's schema.
const config: CodegenConfig = {schema: "https://odyssey-lift-off-server.herokuapp.com/",};
The documents
property
Next, we'll define the documents
that GraphQL Code Generator should consider when generating types for our frontend. All of our code is contained in the src
folder, and we want to be sure that it looks for files in all of the src
subfolders as well! Finally, we should set in the configuration that we want to scan only files that end with .tsx
.
const config: CodegenConfig = {schema: "https://odyssey-lift-off-server.herokuapp.com/",documents: ["src/**/*.tsx"],};
The generates
property
The last thing the Code Generator needs to know is where to output all of the code that it generates. For this, we'll ask it to create a new folder called __generated__
under src
. We set this as a key, whose value is an object with some additional configuration.
const config: CodegenConfig = {schema: "https://odyssey-lift-off-server.herokuapp.com/",documents: ["src/**/*.tsx"],generates: {"./src/__generated__/": {// TODO},},};
Here we can make use of the @graphql-codegen/client-preset
package that we installed earlier. This package configures the Code Generator to work well with React apps, particularly those using Apollo Client. It also gives us some additional plugins that we can tweak to modify how the Code Generator behaves.
We'll add the client
preset indicator to the object, as shown below.
generates: {"./src/__generated__/": {preset: "client",},},
Customizing the graphql
function
One other piece of the generated code we want to control is the name of the graphql
function the Code Generator will give us to work with our queries. This function is a tagged template literal that we use to prepare our GraphQL strings for a GraphQL server.
When produced by the GraphQL Code Generator, this utility comes with an understanding of all the types that the Code Generator produces—meaning that we don't have to type them ourselves! We can simply use the generated function, and it does all the work for us, looking up the types for the GraphQL operations we pass to it.
By default, the Code Generator exports this function as graphql
, but we have the option to rename it. We can do this by setting an additional property below preset
called presetConfig
. We'll use the name gql
instead.
generates: {"./src/__generated__/": {preset: "client",presetConfig: {// TODO},},},
This is an object where we can set a gqlTagName
property, along with our preferred name for this function, gql
.
generates: {"./src/__generated__/": {preset: "client",presetConfig: {gqlTagName: "gql",},},},
Step 3: Running GraphQL Code Generator
If we run our npm run generate
command now, we'll see an error: that's because currently our frontend code doesn't contain any GraphQL operations for the Code Generator to scan.
Unable to find any GraphQL type definitions for the following pointers:- src/**/*.tsx
To run the command anyway, and see what the Code Generator outputs by default for us, we can pass an additional flag into our config
object called ignoreNoDocuments
. By setting ignoreNoDocuments
to true
, we're telling GraphQL Code Generator not to worry if it doesn't find any GraphQL operations in our frontend code; it should proceed with outputting its default generated code anyway. That will give us a chance to take a look at what else it generates for us!
Add ignoreNoDocuments: true
to your config
object, as shown below:
import { CodegenConfig } from "@graphql-codegen/cli";const config: CodegenConfig = {schema: "https://odyssey-lift-off-server.herokuapp.com/",documents: ["src/**/*.tsx"],generates: {"./src/__generated__/": {preset: "client",presetConfig: {gqlTagName: "gql",},},},ignoreNoDocuments: true,};export default config;
Let's run our generate command again in the root folder.
npm run generate
If all goes well, we'll see some green checkmarks in the terminal, along with a new folder under src
called __generated__
!
Step 4: Inspecting the output
Let's look at the files that the GraphQL Code Generator created for us.
📂 __generated__┣ 📄 fragment-masking.ts┣ 📄 gql.ts┣ 📄 graphql.ts┗ 📄 index.ts
The first file fragment-masking.ts
contains some functions that will aid us when working with GraphQL fragments (more on that in Side Quest: Intermediate Schema Design). The second file, gql.ts
, contains the gql
function we asked GraphQL Code Generator to create to help parse GraphQL queries so they can be used by a GraphQL client, like Apollo Client.
When we jump into graphql.ts
, however, we'll see something really interesting—towards the bottom of the file, we can find the three types that we defined in our backend schema!
Query
, Track
, and Author
have successfully been introspected from our GraphQL server running on https://odyssey-lift-off-server.herokuapp.com/
, and the GraphQL Code Generator has generated all the information about the fields they contain and the types of data they are meant to return!
Even though we haven't written any GraphQL operations on the frontend yet, we can see how GraphQL Code Generator was able to pull information about the type of data our frontend will be working with.
From now on, we want to know if there aren't any GraphQL documents for the Code Generator to scan, so we can clean up our codegen.ts
file by removing the ignoreNoDocuments
flag. (Running npm run generate
will still produce an error, but we'll take care of that in the next lesson by adding our first GraphQL operation!)
import { CodegenConfig } from "@graphql-codegen/cli";const config: CodegenConfig = {schema: "https://odyssey-lift-off-server.herokuapp.com/",documents: ["src/**/*.tsx"],generates: {"./src/__generated__/": {preset: "client",presetConfig: {gqlTagName: "gql",},},},- ignoreNoDocuments: true,};export default config;
We're ready to start sending queries!
Note: At this point, you might see an error where your client application is running on http://localhost:3000
! The Variable 'documents' implicitly has an 'any[]' type
error message refers to our generated code (which didn't find any GraphQL operations to include), and we'll clear it up in the next lesson!
Key takeaways
- We can use the GraphQL Code Generator to read the types and fields from a schema, and generate the corresponding TypeScript types.
- We use the
codegen.ts
file to tell the GraphQL Code Generator where our GraphQL API is running, where we want our generated types to be stored, and which plugins to use.
Up next
Without further ado, let's give our frontend app something to query, and populate our homepage with data!
Share your questions and comments about this lesson
This course is currently in
You'll need a GitHub account to post below. Don't have one? Post in our Odyssey forum instead.