All docs and tutorials usually show simple examples of mutations that look like this:
extend type Mutation {
edit(postId: String): String
}
But this way the edit
method has to be unique across all entities, which to me seems like not a very robust way to write things. I would like to describe mutation similar to how we describe Queries, something like this:
type PostMutation {
edit(postId: String): String
}
extend type Mutation {
post: PostMutation
}
This seems to be a valid schema (it compiles and I can see it reflected in the generated graph-i-ql docs). But I can't find a way to make resolvers work with this schema.
Is this a supported case for GraphQL?
It's possible but generally not a good idea because:
It breaks convention. By convention, mutations are always at the root. To distinguish between doing the same action on different types, you name your mutations, for example,
editPost
andeditComment
, instead of justedit
.Having the mutations at the root makes sense conceptually. Whatever action you're doing (liking a post, verifying an email, submitting an order, etc.) doesn't rely on GraphQL having to resolve additional fields before the action is taken. This is unlike when you're actually querying data. For example, to get comments on a post, we may have to resolve a
user
field, then aposts
field and then finally thecomments
field for each post. At each "level", the field's contents are dependent on the value the parent field resolved to. This normally is not the case with mutations.Under the hood, mutations are resolved sequentially. This is contrary to normal field resolution which happens in parallel. That means, for example, the
firstName
andlastName
of aUser
type are resolved at the same time. However, if your operation type ismutation
, the root fields will all be resolved one at a time. So in a query like this:Each mutation will happen one at a time, in the order that they appear in the document. However, this only works for the root and only when the operation is a
mutation
, so these three fields will resolve in parallel:If you still want to do it, despite the above, this is how you do it when using
makeExecutableSchema
, which is what Apollo uses under the hood:Your schema defined
PostMutation
as an object type, so GraphQL is expecting that field to return an object. If you omit the resolver forpost
, it will return null, which means none of the resolvers for the returning type (PostMutation
) will be fired. That also means, we can also write:which does nothing but is still a valid query. Which is yet another reason to avoid this sort of schema structure.
Absolutely disagree with Daniel!
This is an amazing approach which helps to frontenders fastly understand what operations have one or another resource/model. And do not list loooong lists of mutations.
Calling multiple mutations in one request is common antipattern. For such cases better to create one complex mutation.
But even if you need to do such operation with several mutations you may use aliases:
See the following test case: https://github.com/nodkz/conf-talks/blob/master/articles/graphql/schema-design/tests/mutations-test.js
Methods like/unlike are async with timeouts and works sequentially