GraphQL
Refine provides a data provider for GraphQL APIs that has all the features of Refine without giving up the GraphQL features.
- GraphQL Data Provider expects
@urql/core
client instance. - You are responsible for passing
gqlQuery
,gqlMutation
.gql
export of@urql/core
can be used to write GraphQL Operations. - To learn more about data fetching in Refine, check out the Data Fetching guide.
Installation
- npm
- pnpm
- yarn
npm i @refinedev/graphql @urql/core graphql-ws
pnpm add @refinedev/graphql @urql/core graphql-ws
yarn add @refinedev/graphql @urql/core graphql-ws
Usage
We'll create a GraphQL Client with our API url and pass it to the dataProvider
function to create a data provider.
import Refine from "@refinedev/core";
import { Client, fetchExchange } from "@urql/core";
import createDataProvider from "@refinedev/graphql";
export const API_URL = "https://api.nestjs-query.refine.dev/graphql";
const gqlClient = new Client({ url: API_URL, exchanges: [fetchExchange] });
const dataProvider = createDataProvider(gqlClient);
const App = () => <Refine dataProvider={dataProvider}>{/* ... */}</Refine>;
Options
It's also possible to pass a 2nd parameter to GraphQL data provider. 2nd parameter is an object that consist of builder pieces for each data provider method such as getList, updateMany, etc... All fields in this options config are optional and fields that are provided will be deep merged into default options. So you can just pass certain methods you want to override, and the rest will fallback to default.
Let's say you have the following query:
query PostList($where: JSON, $sort: String) {
allBlogPosts(where: $where, sort: $sort) {
nodes {
id
title
content
category {
id
}
}
}
}
By default, our data provider expects a plural form of the resource in the response, so if you have allPosts
, you would need to swizzle GraphQL data provider and customize it yourself. With these options, we help you extract data from your response. So you don't need to create custom data provider for such cases.
import dataProvider, {
GraphQLClient,
defaultGetDataFn,
} from "@refinedev/graphql";
import camelCase from "camelcase";
const client = /** client init **/
const dataProvider = createDataProvider(client, {
getList: {
dataMapper: (response: OperationResult<any>, params: GetListParams) => {
// resource: blogPosts
const operationName = `all${camelcase(resource, {pascal: true})}`
// operationName: allBlogPosts
return response.data?.[operationName].nodes;
},
}
})
type ActionMethod = {
dataMapper: (
response: OperationResult<any>,
params: GetOneParams | GetListParams | etc,
) => {} | [];
buildVariables: (params: CreateParams | UpdateParams | etc) => {};
};
We have ActionMethod type for each of the data provider actions. Additionally, getOne
has convertMutationToQuery
and getList
has getTotalCount
methods.
convertMutationToQuery
method on getOne
might be needed because useForm
hook also uses it. useForm
hook has an optional gqlQuery field, we may only get gqlMutation
. For this reason, we need to convert mutation to query to get initial data on edit, if needed.
getTotalCount
can be used to extract total count of the list query from the response.
See all options
import type {
CreateManyParams,
CreateParams,
CustomParams,
DeleteManyParams,
DeleteOneParams,
GetListParams,
GetManyParams,
GetOneParams,
UpdateManyParams,
UpdateParams,
} from "@refinedev/core";
import type { OperationResult } from "@urql/core";
import { buildSorters, buildFilters, buildPagination } from "../utils";
export const defaultOptions = {
create: {
dataMapper: (response: OperationResult<any>, params: CreateParams<any>) =>
response,
buildVariables: (params: CreateParams<any>) => params.variables,
},
createMany: {
dataMapper: (
response: OperationResult<any>,
params: CreateManyParams<any>,
) => response,
buildVariables: (params: CreateManyParams<any>) => params.variables,
},
getOne: {
dataMapper: (response: OperationResult<any>, params: GetOneParams) =>
response,
buildVariables: (params: GetOneParams) => ({ id: params.id }),
convertMutationToQuery: (params: GetOneParams) => {},
},
getList: {
dataMapper: (response: OperationResult<any>, params: GetListParams) =>
response,
getTotalCount: (response: OperationResult<any>, params: GetListParams) => 0,
buildSorters: (params: GetListParams) => buildSorters(params.sorters),
buildFilters: (params: GetListParams) => buildFilters(params.filters),
buildPagination: (params: GetListParams) =>
buildPagination(params.pagination),
},
getMany: {
buildFilter: (params: GetManyParams) => ({
filter: { id: { in: params.ids } },
}),
dataMapper: (response: OperationResult<any>, params: GetManyParams) =>
response,
},
update: {
dataMapper: (response: OperationResult<any>, params: UpdateParams<any>) =>
response,
buildVariables: (params: UpdateParams<any>) => params.variables,
},
updateMany: {
dataMapper: (
response: OperationResult<any>,
params: UpdateManyParams<any>,
) => response,
buildVariables: (params: UpdateManyParams<any>) => params.variables,
},
deleteOne: {
dataMapper: (
response: OperationResult<any>,
params: DeleteOneParams<any>,
) => response,
},
deleteMany: {
dataMapper: (
response: OperationResult<any>,
params: DeleteManyParams<any>,
) => response,
buildVariables: (params: DeleteManyParams<any>) => params.variables,
custom: {
dataMapper: (response: OperationResult<any>, params: CustomParams) =>
response.data,
buildVariables: (params: CustomParams) => {},
},
},
};
Realtime
@refinedev/graphql
also provides a createLiveProvider
function to enable realtime features of Refine. These features are powered by GraphQL subscriptions and uses graphql-ws
to handle the connections.
import Refine from "@refinedev/core";
import { createLiveProvider } from "@refinedev/graphql";
import createClient from "graphql-ws";
const WSS_URL = "wss://api.nestjs-query.refine.dev/graphql";
const wsClient = createClient({ url: WSS_URL });
const liveProvider = createLiveProvider(wsClient);
const App = () => (
<Refine
liveProvider={liveProvider}
options={{ liveMode: "auto" }}
>
{/* ... */}
</Refine>
);
Queries and Mutations
You can use gql
export from @urql/core
to write your queries and mutations.
Refine hooks' meta
object has gqlQuery
and gqlMutation
properties, you can use them to write your queries and mutations.
As a best-practice, we suggest writing your queries/mutations in a separate file, next to the component that uses it.
import gql from "graphql-tag";
const POSTS_LIST_QUERY = gql`
query PostList($where: JSON, $sort: String) {
posts(where: $where, sort: $sort) {
id
title
content
category {
id
}
}
}
`;
const POST_CREATE_MUTATION = gql`
mutation createPost($input: createPostInput!) {
createPost(input: $input) {
id
title
content
category {
id
}
}
}
`;
import { useList } from "@refinedev/core";
import { POSTS_LIST_QUERY } from "./queries";
export const PostListPage () => {
const { data } = useList({
resource: "posts",
meta: { gqlQuery: POSTS_LIST_QUERY },
});
return (
<div>
{/* ... */}
</div>
);
}
import { useForm } from "@refinedev/core";
import { POST_CREATE_MUTATION } from "./queries";
export const PostCreatePage () => {
const { formProps } = useForm({
resource: "posts",
meta: { gqlMutation: POST_CREATE_MUTATION },
});
return (
<div>
{/* ... */}
</div>
);
}
Handling Errors
When using GraphQL there are two kinds of errors we may see. Network errors and GraphQL errors from the API. URQL provides a CombinedError class which holds these errors when encountered. The Refine GraphQL data provider package hooks identified these errors and provides them back along with a Refine specific error type for missing information. Each error type is prefixed with the [type]
so that the you can determine how they can be handled.
- Code errors
[Code]
: Error is when a required parameter such as a Query or a Mutation has not been provided - Network errors
[Network]
: Contains any error that would prevent making the network request. - GraphQL errors
[GraphQL]
: Any errors recevied from the errors array from the GraphQL API, converted into a string.
The errors are provided through the Notification provider if used or through the Refine's error handlers.
Managing Retries
Refine uses Tanstack Query which by default retries the API call 3 times before showing the error to the user. For [Code]
or [GraphQL]
errors, these retries can be avoided, and the users would only need this for [Network]
type errors using the queryOptions
parameter passed into the data fetch hooks using a function call can help prevent non-required API calls.
useList({
meta: {
gqlQuery: GET_LIST_QUERY,
},
queryOptions: {
retry(failureCount, error) {
// failureCount provides the number of times the request has failed
// error is the error thrown from the GraphQL provider
return error?.message.includes("[Network]") && failureCount <= 3;
},
},
});
Authentication
If your API uses authentication, the easiest way of passing in the token through the header is via the fetchOptions parameter passed into the GraphQL Client.
import createDataProvider from "@refinedev/graphql";
import { Client, fetchExchange } from "urql";
export const client = new Client({
url: API_URL,
exchanges: [fetchExchange],
fetchOptions: () => {
return {
headers: {
/**
* For demo purposes, we're using `localStorage` to access the token.
* You can use your own authentication logic here.
* In real world applications, you'll need to handle it in sync with your `authProvider`.
*/
Authorization: `Bearer ${localStorage.getItem("token")}`,
},
};
},
});
/**
* Create the data provider with the custom client.
*/
const dataProvider = graphqlDataProvider(client);
For more advanced authentication requirements you can use the urql authExchange as documented in https://commerce.nearform.com/open-source/urql/docs/advanced/authentication
Usage with Inferencer
You can also use @refinedev/inferencer
package to generate sample codes for your views. Since the GraphQL data providers rely on meta
fields, you'll need to provide some meta
values beforehand and then Inferencer will use these values to infer the fields of the data provider's response, generate a code and a preview.
Example
npm create refine-app@latest -- --example data-provider-graphql