title |
---|
Setup and options |
To get started with Apollo and React, you will need to install a few packages from npm or use apollo-boost
.
If you are first getting started with Apollo Client, the preset package is a quick and easy way to try it out!
# installing the preset package and react integration
npm install apollo-boost react-apollo graphql --save
# installing each piece independently
npm install apollo-client apollo-cache-inmemory apollo-link-http react-apollo graphql-tag graphql --save
Note: You don't have to do anything special to get Apollo Client to work in React Native, just install and import it as usual.
To get started using Apollo with React, we need to create an ApolloClient
and an ApolloProvider
. The ApolloProvider
is specific to the React integration, to learn how to include Apollo Client with your view layer of choice, see the corresponding guide!
ApolloClient
is the center of using GraphQL in your app! It manages all of your data so you can focus on features!ApolloProvider
makes it easy to use Apollo anywhere in your React component hierarchy.
To get started, create an ApolloClient
instance and point it at your GraphQL server:
import { ApolloClient } from 'apollo-client';
import { HttpLink } from 'apollo-link-http';
import { InMemoryCache } from 'apollo-cache-inmemory';
const client = new ApolloClient({
// By default, this client will send queries to the
// `/graphql` endpoint on the same host
link: new HttpLink(),
cache: new InMemoryCache()
});
The client takes a variety of options, but in particular, if you want to change the URL of the GraphQL server, you can customize your Apollo Link
:
import { ApolloClient } from 'apollo-client';
import { HttpLink } from 'apollo-link-http';
import { InMemoryCache } from 'apollo-cache-inmemory';
const client = new ApolloClient({
link: new HttpLink({ uri: 'https://api.example.com/graphql' }),
cache: new InMemoryCache()
});
ApolloClient
has some other options which control the behavior of the client, and we'll see examples of their use throughout this guide.
To connect your client to your component tree, use an ApolloProvider
component. We suggest putting the ApolloProvider
somewhere high in your app, above any places where you need to access GraphQL data. For example, it could be outside of your root route component if you're using React Router.
import { ApolloProvider } from 'react-apollo';
import { ApolloClient } from 'apollo-client';
import { HttpLink } from 'apollo-link-http';
import { InMemoryCache } from 'apollo-cache-inmemory';
const client = new ApolloClient({
link: new HttpLink(),
cache: new InMemoryCache()
});
ReactDOM.render(
<ApolloProvider client={client}>
<MyAppComponent />
</ApolloProvider>,
document.getElementById('root')
)
import gql from 'graphql-tag';
The gql
template tag is what you use to define GraphQL queries in your Apollo Client apps. It parses your GraphQL query into the GraphQL.js AST format which may then be consumed by Apollo Client methods. Whenever Apollo Client is asking for a GraphQL query you will always want to wrap it in a gql
template tag.
You can embed a GraphQL document containing only fragments inside of another GraphQL document using template string interpolation. This allows you to use fragments defined in one part of your codebase inside of a query define in a completely different file. See the example below for a demonstration of how this works.
Example:
Notice how in the query
variable we not only include the fragments
variable through template string interpolation (${fragments}
), but we also include a spread for the foo
fragment in our query.
const fragments = gql`
fragment foo on Foo {
a
b
c
...bar
}
fragment bar on Bar {
d
e
f
}
`;
const query = gql`
query {
...foo
}
${fragments}
`;
For more information about using fragments, checkout the guide and even some of the different ways to write GraphQL operations in your app using babel or webpack.
Apollo Client makes it super easy to request data using GraphQL. You can read, update, and even subscribe to whatever information your app needs using the client directly, or integrating it with your components.
If you want to see how easy it is to fetch data from a GraphQL server with Apollo, you can use the `query` method on your client. It is as easy as this:import { ApolloClient } from 'apollo-client';
import { HttpLink } from 'apollo-link-http';
import { InMemoryCache } from 'apollo-cache-inmemory';
const client = new ApolloClient({
link: new HttpLink({ uri: 'https://q80vw8qjp.lp.gql.zone/graphql' }),
cache: new InMemoryCache()
});
client.query({ query: gql`{ hello }` }).then(console.log);
The basic usage of graphql()
is as follows:
import React, { Component } from 'react';
import { graphql } from 'react-apollo';
import gql from 'graphql-tag';
// here we create a query opearation
const MY_QUERY = gql`query { todos { text } }`;
// We then can use the graphql container to pass the query results returned by MY_QUERY
// to a component as a prop (and update them as the results change)
const MyComponentWithData = graphql(MY_QUERY)(props => <div>...</div>);
The overall idea of using graphql
is to pair a description (GraphQL Operation) of data you want, with the presentation (React Component) you want to show to your users! That is what makes the API as simple as graphql
+ (operation)
+ (component)
! Over the course of the rest of the docs, you will see the graphql
function used almost everywhere as the best way to use Apollo and React together.
link
: Apollo Client requires an Apollo Link to serve as the network layer. For more information about creating links, read the docs.cache
: The second required argument for using Apollo Client is an instance of an Apollo Cache. The default cache is theapollo-cache-inmemory
which exports an{ InMemoryCache }
. For more information read the cache docs.ssrMode
: When using the client for server side rendering, passssrMode
astrue
so that React Apollo'sgetDataFromTree
can work effectively.ssrForceFetchDelay
: determines the time interval before Apollo Client force fetchs queries after a server side render.connectToDevTools
: This argument allows the Apollo Client Devtools to connect to your application's Apollo Client. You can set this to betrue
to use the tools in production (they are on by default in dev mode).queryDeduplication
: If set to false, this argument will force a query to still be sent to the server even if a query with identical parameters (query, variables, operationName) is already in flight.defaultOptions
: If you want to set application wide defaults for the options supplied towatchQuery
,query
, ormutate
, you can pass them as adefaultOptions
object. An example object looks like this:
const defaultOptions = {
watchQuery: {
fetchPolicy: 'cache-and-network',
errorPolicy: 'ignore',
},
query: {
fetchPolicy: 'network-only',
errorPolicy: 'all',
},
mutate: {
errorPolicy: 'all'
}
}
These options will be merged with options supplied with each request.
React-Apollo includes a component for providing a client instance to a React component tree, and a higher-order component for retrieving that client instance.import { ApolloProvider } from 'react-apollo';
Makes the GraphQL client available to any of your components enhanced by the graphql()
function. The <ApolloProvider/>
component works the same as the [react-redux
<Provider/>
component][]. It provides an ApolloClient
instance to all of your GraphQL components that either use the graphql()
function, or the withApollo
function.
If you do not add this component to the root of your React tree then your components enhanced with Apollo capabilities will not be able to function.
The <ApolloProvider/>
component takes the following props:
client
: The requiredApolloClient
instance. ThisApolloClient
instance will be used by all of your components enhanced with GraphQL capabilties.
If you want to get direct access to your ApolloClient
instance that is provided by <ApolloProvider/>
in your components then be sure to look at the withApollo()
enhancer function.
Example:
ReactDOM.render(
<ApolloProvider client={client}>
<MyRootComponent />
</ApolloProvider>,
document.getElementById('root'),
);
import { graphql } from 'react-apollo';
The graphql()
function is the most important thing exported by react-apollo
. With this function you can create higher-order components that can execute queries and update reactively based on the data in your Apollo store. The graphql()
function returns a function which will “enhance” any component with reactive GraphQL capabilities. This follows the React higher-order component pattern which is also used by react-redux
’s connect
function.
The graphql()
function may be used like this:
function TodoApp({ data: { todos } }) {
return (
<ul>
{todos.map(({ id, text }) => (
<li key={id}>{text}</li>
))}
</ul>
);
}
export default graphql(gql`
query TodoAppQuery {
todos {
id
text
}
}
`)(TodoApp);
You may also define an intermediate function and hook up your component with the graphql()
function like this:
// Create our enhancer function.
const withTodoAppQuery = graphql(gql`query { ... }`);
// Enhance our component.
const TodoAppWithData = withTodoAppQuery(TodoApp);
// Export the enhanced component.
export default TodoAppWithData;
The graphql()
function will only be able to provide access to your GraphQL data if there is a <ApolloProvider/>
component higher up in your tree to provide an ApolloClient
instance that will be used to fetch your data.
The behavior of your component enhanced with the graphql()
function will be different depending on if your GraphQL operation is a query, a mutation, or a subscription. Go to the appropriate API documentation for more information about the functionality and available options for each type.
Before we look into the specific behaviors of each operation, let us look at the config
object.
The config
object is the second argument you pass into the graphql()
function, after your GraphQL document. The config is optional and allows you to add some custom behavior to your higher order component.
export default graphql(
gql`{ ... }`,
config, // <- The `config` object.
)(MyComponent);
Lets go through all of the properties that may live on your config
object.
config.options
is an object or a function that allows you to define the specific behavior your component should use in handling your GraphQL data.
The specific options available for configuration depend on the operation you pass as the first argument to graphql()
. There are options specific to queries and mutations.
You can define config.options
as a plain object, or you can compute your options from a function that takes the component’s props as an argument.
Example:
export default graphql(gql`{ ... }`, {
options: {
// Options go here.
},
})(MyComponent);
export default graphql(gql`{ ... }`, {
options: (props) => ({
// Options are computed from `props` here.
}),
})(MyComponent);
The config.props
property allows you to define a map function that takes your props including the props added by the graphql()
function (props.data
for queries and props.mutate
for mutations) and allows you to compute a new props object that will be provided to the component that graphql()
is wrapping.
The function you define behaves almost exactly like mapProps
from Recompose providing the same benefits without the need for another library.
config.props
is most useful when you want to abstract away complex functions calls into a simple prop that you can pass down to your component.
Another benefit of config.props
is that it also allows you to decouple your pure UI components from your GraphQL and Apollo concerns. You can write your pure UI components in one file and then keep the logic required for them to interact with the store in a completely different place in your project. You can accomplish this by your pure UI components only asking for the props needed to render and config.props
can contain the logic to provide exactly the props your pure component needs from the data provided by your GraphQL API.
Example:
This example uses props.data.fetchMore
.
export default graphql(gql`{ ... }`, {
props: ({ data: { fetchMore } }) => ({
onLoadMore: () => {
fetchMore({ ... });
},
}),
})(MyComponent);
function MyComponent({ onLoadMore }) {
return (
<button onClick={onLoadMore}>
Load More!
</button>
);
}
If config.skip
is true then all of the React Apollo code will be skipped entirely. It will be as if the graphql()
function were a simple identity function. Your component will behave as if the graphql()
function were not there at all.
Instead of passing a boolean to config.skip
, you may also pass a function to config.skip
. The function will take your components props and should return a boolean. If the boolean returns true then the skip behavior will go into effect.
config.skip
is especially useful if you want to use a different query based on some prop. You can see this in an example below.
Example:
export default graphql(gql`{ ... }`, {
skip: props => !!props.skip,
})(MyComponent);
The following example uses the compose()
function to use multiple graphql()
enhancers at once.
export default compose(
graphql(gql`query MyQuery1 { ... }`, { skip: props => !props.useQuery1 }),
graphql(gql`query MyQuery2 { ... }`, { skip: props => props.useQuery1 }),
)(MyComponent);
function MyComponent({ data }) {
// The data may be from `MyQuery1` or `MyQuery2` depending on the value
// of the prop `useQuery1`.
console.log(data);
}
This property allows you to configure the name of the prop that gets passed down to your component. By default if the GraphQL document you pass into graphql()
is a query then your prop will be named data
. If you pass a mutation then your prop will be named mutate
. While appropriate these default names collide when you are trying to use multiple queries or mutations with the same component. To avoid collisions you may use config.name
to provide the prop from each query or mutation HOC a new name.
Example:
This example uses the compose
function to use multiple graphql()
HOCs together.
export default compose(
graphql(gql`mutation (...) { ... }`, { name: 'createTodo' }),
graphql(gql`mutation (...) { ... }`, { name: 'updateTodo' }),
graphql(gql`mutation (...) { ... }`, { name: 'deleteTodo' }),
)(MyComponent);
function MyComponent(props) {
// Instead of the default prop name, `mutate`,
// we have three different prop names.
console.log(props.createTodo);
console.log(props.updateTodo);
console.log(props.deleteTodo);
return null;
}
By setting config.withRef
to true you will be able to get the instance of your wrapped component from your higher-order GraphQL component using a getWrappedInstance
method available on the instance of your higher-order GraphQL component.
You may want to set this to true when you want to call functions or get access to properties that are defined on your wrapped component’s class instance.
Below you can see an example of this behavior.
Example:
This example uses the React ref
feature.
class MyComponent extends Component {
saySomething() {
console.log('Hello, world!');
}
render() {
// ...
}
}
const MyGraphQLComponent = graphql(
gql`{ ... }`,
{ withRef: true },
)(MyComponent);
class MyContainerComponent extends Component {
render() {
return (
<MyGraphQLComponent
ref={component => {
assert(component.getWrappedInstance() instanceof MyComponent);
// We can call methods on the component class instance.
component.saySomething();
}}
/>
);
}
}
By default the display name for React Apollo components is Apollo(${WrappedComponent.displayName})
. This is a pattern used by most React libraries that make use of higher order components. However, it may get a little confusing when you are using more then one higher order components and you look at the React Devtools.
To configure the name of your higher order component wrapper, you may use the config.alias
property. So for example, if you set config.alias
to 'withCurrentUser'
your wrapper component display name would be withCurrentUser(${WrappedComponent.displayName})
instead of Apollo(${WrappedComponent.displayName})
.
Example:
This example uses the compose
function to use multiple graphql()
HOCs together.
export default compose(
graphql(gql`{ ... }`, { alias: 'withCurrentUser' }),
graphql(gql`{ ... }`, { alias: 'withList' }),
)(MyComponent);
import { compose } from 'react-apollo';
For utility purposes, react-apollo
exports a compose
function. Using this function you may cleanly use several component enhancers at once. Including multiple graphql()
, withApollo()
, or Redux connect()
enhancers. This should clean up your code when you use multiple enhancers. Redux also exports a compose
function, and so does Recompose so you may choose to use the function from whichever library feels most appropriate.
An important note is that compose()
executes the last enhancer first and works its way backwards through the list of enhancers. To illustrate calling three functions like this: funcC(funcB(funcA(component)))
is equivalent to calling compose()
like this: compose(funcC, funcB, funcA)(component)
. If this does not make sense to you consider using flowRight()
from Lodash which otherwise has the same behavior.
Example:
export default compose(
withApollo,
graphql(`query { ... }`),
graphql(`mutation { ... }`),
connect(...),
)(MyComponent);
import { withApollo } from 'react-apollo';
A simple enhancer which provides direct access to your ApolloClient
instance. This is useful if you want to do custom logic with Apollo. Such as calling one-off queries. By calling this function with the component you want to enhance, withApollo()
will create a new component which passes in an instance of ApolloClient
as a client
prop.
If you are wondering when to use withApollo()
and when to use graphql()
the answer is that most of the time you will want to use graphql()
. graphql()
provides many of the advanced features you need to work with your GraphQL data. You should only use withApollo()
if you want the GraphQL client without any of the other features.
This will only be able to provide access to your client if there is an <ApolloProvider/>
component higher up in your tree to actually provide the client.
Example:
export default withApollo(MyComponent);
function MyComponent({ client }) {
console.log(client);
}