Skip to content
This repository was archived by the owner on Nov 20, 2020. It is now read-only.

Feat/mutation error policy #127

Merged
merged 4 commits into from
Jun 20, 2020
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
43 changes: 25 additions & 18 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -181,22 +181,34 @@ module ScreamMutation = [%graphql {|
[@react.component]
let make = () => {
/* Both variant and records available */
let ( screamMutation, _simple, _full ) = useMutation(~variables=ScreamMutation.makeVariables(~screamLevel=10, ()), ScreamMutation.definition);
let ( screamMutation, simple, _full ) =
useMutation(~variables=ScreamMutation.makeVariables(~screamLevel=10, ()), ScreamMutation.definition);
let scream = (_) => {
screamMutation()
|> Js.Promise.then_(result => {
switch(result) {
| Data(data) => ...
| Error(error) => ...
| NoData => ...
}
Js.Promise.resolve()
})
|> Js.Promise.then_(((simple, _full)) => {
// Trigger side effects by chaining the promise returned by screamMutation()
switch (simple) {
// You *must* set the error policy to be able to handle errors
// in then_. See EditPersons.re for more
| ApolloHooks.Mutation.Errors(_theErrors) => Js.log("OH NO!")
| NoData => Js.log("NO DATA?")
| Data(_theData) => Js.log("DATA!")
};
Js.Promise.resolve();
})
|> ignore
}

// Use simple (and/or full) for (most) UI feedback
<div>
<button onClick={scream}>
{switch (simple) {
| NotCalled
| Data(_) => React.null
| Loading => <div> "Screaming!"->React.string </div>
| NoData
| Error(_) => <div> "Something went wrong!"->React.string </div>
}}
<button onClick={scream} disabled={simple === Loading}>
{React.string("You kids get off my lawn!")}
</button>
</div>
Expand All @@ -212,14 +224,9 @@ let make = () => {
let ( screamMutation, _simple, _full ) = useMutation(ScreamMutation.definition);
let scream = (_) => {
screamMutation(~variables=ScreamMutation.makeVariables(~screamLevel=10, ()), ())
|> Js.Promise.then_(result => {
switch(result) {
| Data(data) => ...
| Error(error) => ...
| NoData => ...
}
Js.Promise.resolve()
})
|> Js.Promise.then_(((simple, _full)) => {
...
})
|> ignore
}

Expand Down
27 changes: 25 additions & 2 deletions examples/persons/src/EditPerson.re
Original file line number Diff line number Diff line change
Expand Up @@ -71,7 +71,12 @@ let make = (~refetchQueries, ~update) => {
React.useReducer(reducer, {age: None, name: "", id: ""});

let (editPersonMutation, _simple, _full) =
useMutation(~refetchQueries, ~update, EditPersonMutation.definition);
useMutation(
~refetchQueries,
~update,
~errorPolicy=All, // See note below on error policies
EditPersonMutation.definition,
);

let handleSubmit = event => {
ReactEvent.Form.preventDefault(event);
Expand All @@ -89,8 +94,26 @@ let make = (~refetchQueries, ~update) => {
OptimisticResponse.make(~id=state.id, ~name=state.name, ~age),
(),
)
/* Setting error policy to All (or Ignore) means that errors show up
* in then_ in the promise returned by editPersonMutation
* Not setting it (or setting it to None) makes the promise reject
* on errors and you'll have to handle errors in Js.catch(e => ...) instead,
* where e is just Js.Promise.error and you won't get any help from the type system.
*
* See also: https://www.apollographql.com/docs/react/data/error-handling/#error-policies
*/
|> Js.Promise.then_(((simple, _full) as result) => {
switch (simple) {
| ApolloHooks.Mutation.Errors(_theErrors) => Js.log("OH NO!")
| NoData => Js.log("NO DATA?")
| Data(_theData) => Js.log("DATA!")
};
// If you don't need to handle the result elsewhere,
// the promise can just resolve to unit
Js.Promise.resolve(result);
})
|> ignore
| None => ignore()
| None => ()
};
};

Expand Down
6 changes: 6 additions & 0 deletions src/ApolloHooksMutation.re
Original file line number Diff line number Diff line change
Expand Up @@ -64,6 +64,8 @@ type options('a) = {
[@bs.optional]
optimisticResponse: Js.Json.t,
[@bs.optional]
errorPolicy: string,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

looks good, probably just leave a link for defining it as a poly. variant in some next PR?
https://www.apollographql.com/docs/react/data/error-handling/#error-policies

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Opened #132.

[@bs.optional]
context: Context.t,
};

Expand Down Expand Up @@ -97,6 +99,7 @@ let useMutation:
unit
=?,
~optimisticResponse: Js.Json.t=?,
~errorPolicy: ApolloHooksTypes.errorPolicy=?,
~context: Context.t=?,
ApolloHooksTypes.graphqlDefinition('data, _, _)
) =>
Expand All @@ -112,6 +115,7 @@ let useMutation:
~awaitRefetchQueries=?,
~update=?,
~optimisticResponse=?,
~errorPolicy=?,
~context=?,
(parse, query, _),
) => {
Expand All @@ -125,6 +129,8 @@ let useMutation:
~awaitRefetchQueries?,
~update?,
~optimisticResponse?,
~errorPolicy=?
errorPolicy->Belt.Option.map(ApolloHooksTypes.errorPolicyToJs),
~context?,
(),
),
Expand Down