Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Distinguishing between nulls and undefindes in variables #348

Closed
mbirkegaard opened this issue May 17, 2022 · 5 comments · Fixed by #426
Closed

Distinguishing between nulls and undefindes in variables #348

mbirkegaard opened this issue May 17, 2022 · 5 comments · Fixed by #426

Comments

@mbirkegaard
Copy link
Collaborator

mbirkegaard commented May 17, 2022

Currently nullable variables in mutations and queries become optional on the rescript-relay side.

Suppose there's mutation that looks something like

input MyMytationInput {
  id: ID!
  a: Int
  b: Int
}

myMutation(input: MyMutationInput): string

which edits some thing. According to coerce argument values in the graphql spec, there's a difference between executing

mutation myMutation(input: {
  id: "my-id",
  a: 7,
  b: null
})

and

mutation myMutation(input: {
  id: "my-id",
  a: 7,
})

However, on the rescript-relay side the generated mutation input has type

type myMutationInput = {
  id: string,
  a: option<int>
  b: option<int>
}

This disallows sending an explicit null. Setting b: None when calling the mutation turns it into undefined. Our current solution to this problem is to have the resolver for the mutation turn undefined into null depending on the use case.

I think my preference is an explicit approach where all nullable variables are of type Js.Nullable.t<'value> and handled when serialising, but I understand that may not be ergonomic enough for everyone and that an escape hatch solution like jeddeloh/rescript-apollo-client#27 is prefered.

@zth
Copy link
Owner

zth commented May 19, 2022

My initial hunch for this is having something similar to this:

mutation X($someVariable: SomeType!) @rescriptRelayAllowNull {

...which would make all variables in the operation modelled as Js.Nullable.t rather than option. It'd work for all operations, so queries as well.

Thoughts?

@mbirkegaard
Copy link
Collaborator Author

mbirkegaard commented May 20, 2022

I think that could work. It would have to be nullables all way down to cover our concrete use case, where the nullable fields are on a deeper type.

Js.Nullable.t could work, but option<Js.Null.t> is also worth considering.

If the generated type is something like (for some new schema, not the one above)

module Types {
  ...
  type rec input {
     id: string
     inputA: inputA
  } and
  type inputA {
    a: int,
    b: option<Js.Null.t<int>>,
    c: option<Js.Null.t<int>>
  }
}

...

module Utils = {
  ...
  @live @obj
  external make_input: (
    ~id: string,
    ~inputA: inputA
  ) => input = ""
  @live @obj
  external make_inputA: (
    ~a: int,
    ~b: Js.Null.t<int>=?
    ~c: Js.Null.t<int>=?
    unit,
  ) => inputA = ""

I think usage would be a bit more ergonomic, i.e.

make_inputA(~a=1, ~b=Js.Null.return(2), ())

compared to using Js.Nullable.t:

make_inputA(~a=1, ~b=Js.Nullable.return(2), ~c=Js.Nullable.undefined)

I'm not sure how this interacts with #349. Would all of the GraphQL nullable fields in the schema assets file be modeled as Js.Nullable.t or option<Js.Null.t>? If not, I think that breaks sharing types. Maybe the directive just changes which make functions are created and the types are always modeled to allow null and undefined?

@mununki
Copy link

mununki commented Dec 6, 2022

@zth I'd like to join this discussion. AFAIK, the option type for the nullable field in the mutation input is not distinguishing between null and undefined: the undefined means omitting the field when we request the mutation. So, we can't distinguish the mutation between removing and not updating for the nullable input field.
I've tried to send the null value for the nullable input field as below, but it omits the field on the network request anyway.

input={a: Some(Obj.magic(Js.Nullable.null)), ...}

IMHO, it would need to get a type definition as Js.Nullable.t<'a> with optional field record syntax for the nullable field in the mutation input. Maybe we can introduce a new type for the input field:

type input = {
  a: int // required
  b?: Js.Nullable.t<int> // nullable
}

let i0 = {
  a: 1,
  b: Js.Nullable.null, // null
}
let i1 = {
  a: 1,
  // undefined
}

I guess it lets us modeling both cases of null and undefined.

@mununki
Copy link

mununki commented Dec 6, 2022

FYR, graphql/graphql-spec#83

@zth
Copy link
Owner

zth commented Feb 27, 2023

I did an implementation intended to solve this here: #426

@mununki @mbirkegaard what do you think?

@zth zth closed this as completed in #426 Mar 11, 2023
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging a pull request may close this issue.

3 participants