Queries Generator
The queries generator scans your project for *.edgeql files and generates functions that allow you to execute these queries with fully defined input parameters and return types.
Installation
To get started, install the following packages.
Install the gel package.
$
npm install gel # npm users$
yarn add gel # yarn users$
bun add gel # bun users$
deno add npm:gel # deno usersThen install @gel/generate as a dev dependency.
$
npm install @gel/generate --save-dev # npm users$
yarn add @gel/generate --dev # yarn users$
bun add --dev @gel/generate # bun users$
deno add --dev npm:@gel/generate # deno usersGeneration
Consider the following file tree.
.
├── package.json
├── gel.toml
├── index.ts
├── dbschema
└── queries
└── getUser.edgeqlThe following command will run the queries generator.
$
npx @gel/generate queries$
deno run --allow-all npm:@gel/generate queries$
bunx @gel/generate queriesThe generator will detect the project root by looking for an gel.toml, then scan the directory for *.edgeql files. In this case, there's only one: queries/getUser.edgeql.
select User { name, email } filter .id = <uuid>$user_id;For each .edgeql file, the generator will read the contents and send the query to the database, which returns type information about its parameters and return type. The generator uses this information to create a new file getUser.query.ts alongside the original getUser.edgeql file.
.
├── package.json
├── gel.toml
├── index.ts
├── dbschema
└── queries
└── getUser.edgeql
└── getUser.query.ts <-- generated fileThis example assumes you are using TypeScript. The generator tries to auto-detect the language you're using; you can also specify the language with the --target flag. See the Targets section for more information.
The generated file will look something like this:
import type { Client } from "gel";
export type GetUserArgs = {
user_id: string;
};
export type GetUserReturns = {
name: string;
email: string;
} | null;
export async function getUser(
client: Client,
args: GetUserArgs
): Promise<GetUserReturns> {
return await client.querySingle(
`select User { name, email } filter .id = <uuid>$user_id;`,
args
);
}Some things to note:
-
The first argument is a
Clientinstance. This is the same client you would use to execute a query manually. You can use the same client for both manual and generated queries. -
The second argument is a parameter object. The keys of this object are the names of the parameters in the query.
-
The code uses the
querySinglemethod, since the query is only expected to return a single result. -
We export the type of the parameter object and the return value unwrapped from the promise.
We can now use this function in our code.
import { getUser } from "./queries/getUser.query";
import {
createClient,
type GetUserArgs,
type GetUserReturns,
} from "gel";
const client = await createClient();
const newUser: GetUserArgs = {
user_id: "00000000-0000-0000-0000-000000000000"
};
const user = await getUser(client, newUser); // GetUserReturns
if (user) {
user.name; // string
user.email; // string
}Generators work by connecting to the database to get information about the current state of the schema. Make sure you run the generators again any time the schema changes so that the generated code is in-sync with the current state of the schema. The easiest way to do this is to add the generator command to the schema.update.after hook (added in 6) in your gel.toml.
Single-file mode
Pass the --file flag to generate a single file that contains functions for all detected .edgeql files. This lets you import all your queries from a single file.
Let's say we start with the following file tree.
.
├── package.json
├── gel.toml
├── index.ts
├── dbschema
└── queries
└── getUser.edgeql
└── getMovies.edgeqlThe following command will run the generator in --file mode.
$
npx @gel/generate queries --fileA single file will be generated that exports two functions, getUser and getMovies. By default this file is generated into the dbschema directory.
.
├── package.json
├── gel.toml
├── index.ts
├── dbschema
│ └── queries.ts <-- generated file
└── queries
└── getUser.edgeql
└── getMovies.edgeqlWe can now use these functions in our code.
import * as queries from "./dbschema/queries";
import { createClient } from "gel";
const client = await createClient();
const movies = await queries.getMovies(client);
const user = await queries.getUser(client, {
user_id: "00000000-0000-0000-0000-000000000000"
});To override the file path and name, you can optionally pass a value to the --file flag. Note that you should exclude the extension.
$
npx @gel/generate queries --file path/to/myqueriesThe file extension is determined by the generator --target and will be automatically appended to the provided path. Extensionless "absolute" paths will work; relative paths will be resolved relative to the current working directory.
This will result in the following file tree.
.
├── package.json
├── gel.toml
├── path
│ └── to
│ └── myqueries.ts
├── queries
│ └── getUser.edgeql
│ └── getMovies.edgeql
└── index.ts