rest0.3.3

Declarative REST resolver extension.

REST Extension

This is a REST extension for the Grafbase Gateway. It allows you to define REST endpoints and map them to GraphQL fields. It provides two directives:

  • @restEndpoint, which you can repeat on the schema, defines a REST endpoint.
  • @rest, defined on a field, specifies which endpoint the field uses, which path and method it uses and how it selects data from the REST response.

Add the following to your gateway configuration ("grafbase.toml"):

[extensions.rest] version = "0.3"

Then run grafbase extension install. The extension will be installed in the grafbase_extensions directory. That directory must be present when the gateway is started.

Build this extension manually and copy the artifacts to a location where the gateway can find them until we complete the Grafbase Extension Registry.

grafbase extension build

The build directory contains the resulting wasm component and manifest file.

build/ ├── extension.wasm └── manifest.json

In your gateway configuration, you can now load the extension from the build directory.

[extensions.rest] path = "/path/to/build"

This extension acts as a field resolver for the Grafbase Gateway. Use it as a subgraph to provide a REST translator to GraphQL for the gateway.

The extension provides two directives:

  • @restEndpoint, which you can repeat on the schema, defines a REST endpoint.
  • @rest, defined on a field, specifies which endpoint the field uses, which path and method it uses and how it selects data from the REST response.

Define your REST endpoint in your subgraph definition:

extend schema @link(url: "https://specs.apollo.dev/federation/v2.0", import: ["@key", "@shareable"]) @link(url: "file:///absolute/path/to/extension/build", import: ["@restEndpoint", "@rest"]) @restEndpoint( name: "restCountries", baseURL: "https://restcountries.com/v3.1" )

The @restEndpoint takes a unique name per subgraph, which you must refer to in the corresponding @rest directives, and a baseURL.

The @rest directive goes to a field directive:

type Country { name: String! } type Query { listAllCountries: [Country!]! @rest( method: GET endpoint: "restCountries" path: "/all" selection: "[.[] | { name: .name.official }]" ) }

The endpoint argument must match one @restEndpoint definition in the same subgraph. The http arguments define the method and a path for this endpoint, and the selection defines how the data transforms from the REST endpoint to a GraphQL subgraph response.

In our example we query countries from a public endpoint, which returns JSON. If we curl the API:

curl https://restcountries.com/v3.1/all |jq |head -n100

Our data looks something like this:

[ { "name": { "common": "South Georgia", "official": "South Georgia and the South Sandwich Islands", "nativeName": { "eng": { "official": "South Georgia and the South Sandwich Islands", "common": "South Georgia" } } }, ... } }

Now, our GraphQL type for Country has only one field: the name. The selection argument in the @rest directive supports jq filters. First try the selection filter in the terminal with curl and jq. Convert the API endpoint from what we see above to the following, which should fit our GraphQL type definition:

type Country { name: String! }

The extension JSON output should have objects with the name key and the values must be strings. No nulls are allowed.

Run the following curl command:

curl https://restcountries.com/v3.1/all |jq "[.[] | { name: .name.official }]"

You'll get output:

[ { "name": "South Georgia and the South Sandwich Islands" }, { "name": "Grenada" }, ... ]

Save the subgraph schema we just defined into a file, and publish it to the Grafbase platform

grafbase publish --name countries -m init my-org/my-federated-graph

You can omit the --url parameter from a subgraph that only acts as a virtual graph for an extension.

The REST extension sends headers to the REST endpoint. The resolver filters the request headers and uses the resulting subgraph headers.

When you compose the REST subgraph to your federated graph, and you give it a name, you can define the header rules in your gateway configuration.

Use the body argument to send data to the REST endpoint. The body argument accepts a JSON object or a selection that maps data from the input arguments.

To send dynamic data from the input arguments, add a selection to the body. The extension looks for a body in an argument named input. Use this name to follow the expected convention:

type Mutation { createCountry(input: Country!): Country! @rest( endpoint: "restCountries", http: { method: POST, path: "/create" }, selection: "{ name: .name.official }" ) }

You can also use static data in the body:

type Mutation { createCountry: Country! @rest( endpoint: "restCountries", method: POST, path: "/create" body: { static: { name: "Georgia" } }, selection: "{ name: .name.official }" ) }

The extension checks static data first, then searches for a body in an argument named input.

The path argument is used to specify the path to the REST endpoint. You can use the input arguments to construct the path:

type Mutation { getCountry(id: Int!): Country @rest( endpoint: "restCountries", method: GET, path: "/fetch/{{ args.id }}" selection: "{ name: .name.official }" ) }

The extension will generate the path based on the id argument.

11 Mar, 2025
Tom Houlé
InstallAdd this to your TOML configuration file:
[extensions] rest = "0.3.3"