Recently we bought into GraphQL and use it in every one of our web apps, both on the client and server level. It’s been helpful reducing unnecessary communication needed between our different teams when it comes to knowing what our many, many different API’s do. This contributes to our async work strategy and keeps developers moving and focusing on difficult problems versus organizational bloat.
For use with our frontend applications, we opted for Apollo Client with React which seems to be the one true GraphQL client at this point. As the library is fairly new (the javascript ecosystem moves fast, who knew?) we have experienced our fair share of pains and troubleshooting. Some of which included:
- Using Apollo Local State on a large codebase and running in production
- Integration with Server Side Rendering (Next.js)
- Generating Typescript types on a stitched schema
This article focuses on the first point, managing frontend local state. When looking for state management solutions, Apollo Local State (formally apollo-link-state) popped up. A couple of reasons led us to using this library:
- The data models and store structure can be shared between the data fetching cache and the local state management cache
- This leads to sharing of Typescript types as well 😉
- Actions are performed with mutations, something that previous GraphQL users already understand.
- Staying within the Apollo ecosystem meant smooth integration with existing tools, meaning less overhead for developers.
- Backed by Apollo meant that support would be there for a significant amount of time.
Another good sign was this very attractive article which explains the advantages of keeping your local state close to the GraphQL schema vs using something like Redux.
Here comes the pain
A pattern established by Flux (the paradigm, not the library), has you splitting up actions for every event which happens in your app. User clicked a button, action is triggered, user scrolls down a certain length, action is triggered. Your app can observe these actions and manipulate the state accordingly. With Apollo Local State Mutations, this becomes much more intentional. No observability is given, each action is directly related to a resolver.
For example, you want to update a name on an issue (think GitHub just for examples sake) in the cache, this is Apollo’s term for state:
const IssueContainer: React.FC<{ issue: GithubIssue }> = ({ issue }) => (
<UpdateIssueNameMutation mutation={UPDATE_ISSUE_NAME}>
{(updateIssuename) => (
<IssueFields
issue={issue}
onChange={(name) => {
updateIssuename({
variables: {
input: {
id: issue.id,
name,
},
},
});
}}
/>
)}
</UpdateIssueNameMutation>
);
For this above mutation, here is an example of what we would need to write in Typescript, I break down what’s going on in the comments:
import gql from "graphql-tag";
import { IFieldResolver } from "graphql-tools";
import { ISSUE_PARTS } from "../issues";
import { IssueParts, UpdateIssueNameVariables } from "../../graphql-types";
import { ResolverContext } from ".";
export const UPDATE_ISSUE_NAME = gql`
mutation UpdateIssueName($input: UpdateIssueNameInput!) {
updateIssueName(input: $input) @client
}
`;
const ISSUE_FRAGMENT = gql`
${ISSUE_PARTS}
fragment IssueParts on Issue {
id @client
name @client
}
`;
/**
* Updates the name of a Github issue.
**/
const updateIssuename: IFieldResolver<void, ResolverContext, any> = (
_obj,
args: UpdateIssueNameVariables,
context
) => {
const { input } = args;
const { cache, getCacheKey } = context;
// 1. Get the id of the object in the cache using the actual issue id
const id = getCacheKey({
__typename: "Issue",
id: input.id,
});
// 2. Get the data from the cache
const issue: IssueParts | null = cache.readFragment({
fragment: ISSUE_FRAGMENT,
fragmentName: "IssueParts",
id,
});
if (!issue) {
return null;
}
// 3. Update the data locally
const updatedIssue = {
...issue,
name: input.name,
};
// 4. Write the data back to the cache
cache.writeFragment({
fragment: ISSUE_FRAGMENT,
fragmentName: "IssueParts",
id,
data: updatedIssue,
});
return null;
};
export default updateIssuename;
That’s 60 lines to update a single attribute on one data entity in the cache. After doing a couple of these, you will start to pull your hair out. Mutations as is, don’t do a whole lot for you, this results in a lot of boilerplate. Having all of the boilerplate code is not ideal, it leads to more bugs and thus more tests need to be written to avoid those bugs.
Looking for patterns
After writing about ten of these with plans to write a lot more, we wrote a small Yeoman generator to speed up the process. This made writing them a lot faster but didn’t solve the bloat in our codebase. Every mutation ended up doing the same thing as described in the comments above:
- Get the id of the object in the cache using the actual entity id
- Get the data from the cache
- Update the data locally
- Write the data back to the cache
The solution
Naturally, we wrote a helper which would help us refactor our resolvers.
import { IFieldResolver } from "graphql-tools";
import { DocumentNode } from "graphql";
import { ResolverContext } from ".";
interface InputVariablesShape<TInput> {
input: TInput;
}
/**
* Creates a client-side mutation resolver which reads one item from the cache,
* mutates it using the given mutation, then writes it back to the cache.
* @param reducer Func which mutates data and returns it. Must be a pure function.
* @param fragment Used to read/write data to apollo-link-state.
* @param fragmentName Name of fragment inside
* @param getId A func which returns the id of the entity to be used in the `reducer` func.
*/
export const createResolver = <InputShape, EntityType>(
reducer: (entity: EntityType, input: InputShape) => EntityType,
fragment: DocumentNode,
fragmentName: string,
getId: (input: InputShape) => string
) => {
const resolver: IFieldResolver<void, ResolverContext, any> = (
_obj,
args: InputVariablesShape<InputShape>,
{ cache, getCacheKey }
) => {
const { input } = args;
// 1. Get the id of the object in the cache using `getId`
const id = getCacheKey({ id: getId(input) });
// 2. Get the data from the cache
const entity: EntityType | null = cache.readFragment({
fragment,
fragmentName,
id,
});
if (!entity) {
return null;
}
// 3. Update the data locally
const newEntity: EntityType = reducer(entity, input);
// 4. Write the data back to the cache
cache.writeFragment({
fragment,
fragmentName,
id,
data: newEntity,
});
return null;
};
return resolver;
};
This resulted in the resolver code becoming a small pure func:
const fragment = gql`
${ISSUE_PARTS}
fragment BasicIssueParts on BasicIssue {
... on Node {
id
}
parameters {
id
value
}
}
`;
export const reducer = (
issue: IssueParts,
input: UpdateIssueNameInput
): IssueParts => ({
...issue,
name: input.name,
});
export default createResolver(
reducer,
fragment,
"IssueParts",
(input: UpdateIssueNameInput) => {
return input.id;
}
);
A two-line resolver which does the same as before. This covered about 90% of our use-cases!
The road ahead
If you found this blog post helpful, don’t hesitate to steal the gist of this code.