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

Add a mechanism to specify options on input-type fields #124

Merged
merged 2 commits into from
Oct 1, 2021
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
3 changes: 3 additions & 0 deletions docs/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -22,10 +22,13 @@ When releasing a new version:

### Breaking changes:

- Previously, `# @genqlient` directives applied to entire operations applied inconsistently to fields of input types used by those operations. Specifically, `pointer: true`, when applied to the operation, would affect all input-field arguments, but `omitempty: true` would not. Now, all options apply to fields of input types; this is a behavior change in the case of `omitempty`.

### New features:

- genqlient's types are now safe to JSON-marshal, which can be useful for putting them in a cache, for example. See the [docs](FAQ.md#-let-me-json-marshal-my-response-objects) for details.
- The new `flatten` option in the `# @genqlient` directive allows for a simpler form of type-sharing using fragment spreads. See the [docs](FAQ.md#-shared-types-between-different-parts-of-the-query) for details.
- The new `for` option in the `# @genqlient` directive allows applying options to a particular field anywhere it appears in the query. This is especially useful for fields of input types, for which there is otherwise no way to specify options; see the [documentation on handling nullable fields](FAQ.md#-nullable-fields) for an example, and the [`# @genqlient` directive reference](genqlient_directive.graphql) for the full details.

### Bug fixes:

Expand Down
37 changes: 35 additions & 2 deletions docs/FAQ.md
Original file line number Diff line number Diff line change
Expand Up @@ -144,7 +144,7 @@ query MyQuery(
}
```

You can also put the `# @genqlient(omitempty: true)` on the first line, which will apply it to all arguments in the query.
You can also put the `# @genqlient(omitempty: true)` on the first line, which will apply it to all arguments in the query, or `# @genqlient(for: "MyInput.myField", omitempty: true)` on the first line to apply it to a particular field of a particular input type used by the query (for which there would otherwise be no place to put the directive, as the field never appears explicitly in the query, but only in the schema).

If you need to distinguish null from the empty string (or generally from the Go zero value of your type), you can tell genqlient to use a pointer for the field or argument like this:
```graphql
Expand All @@ -157,7 +157,40 @@ query MyQuery(
}
```

This will generate a Go field `MyString *string`, and set it to `nil` if the server returns null (and in reverse for arguments). Such fields can be harder to work with in Go, but allow a clear distinction between null and the Go zero value. Again, you can put the directive on the first line to apply it to everything in the query, although this usually gets cumbersome.
This will generate a Go field `MyString *string`, and set it to `nil` if the server returns null (and in reverse for arguments). Such fields can be harder to work with in Go, but allow a clear distinction between null and the Go zero value. Again, you can put the directive on the first line to apply it to everything in the query, although this usually gets cumbersome, or use `for` to apply it to a specific input-type field.

As an example of using all these options together:
```graphql
# @genqlient(omitempty: true)
# @genqlient(for: "MyInputType.id", omitempty: false, pointer: true)
# @genqlient(for: "MyInputType.name", omitempty: false, pointer: true)
query MyQuery(
arg1: MyInputType!,
# @genqlient(pointer: true)
arg2: String!,
# @genqlient(omitempty: false)
arg3: String!,
) {
myString(arg1: $arg1, arg2: $arg2, arg3: $arg3)
}
```
This will generate:
```go
func MyQuery(
ctx context.Context,
client graphql.Client,
arg1 MyInputType,
arg2 *string, // omitempty
arg3 string,
) (*MyQueryResponse, error)
type MyInputType struct {
Id *string `json:"id"`
Name *string `json:"name"`
Title string `json:"title,omitempty"`
Age int `json:"age,omitempty"`
}
```

See [genqlient_directive.graphql](genqlient_directive.graphql) for complete documentation on these options.

Expand Down
59 changes: 47 additions & 12 deletions docs/genqlient_directive.graphql
Original file line number Diff line number Diff line change
Expand Up @@ -7,11 +7,12 @@
# by the server, not by the client, so it would reject a real @genqlient
# directive as nonexistent.)
#
# Directives may be applied to fields, arguments, or the entire query.
# Directives on the line preceding the query apply to all relevant nodes in
# the query; other directives apply to all nodes on the following line. (In
# all cases it's fine for there to be other comments in between the directive
# and the node(s) to which it applies.) For example, in the following query:
# Directives may be applied to fields, arguments, or the entire query or named
# fragment. Directives on the line preceding the query or a named fragment
# apply to all relevant nodes in the query; other directives apply to all nodes
# on the following line. (In all cases it's fine for there to be other
# comments in between the directive and the node(s) to which it applies.) For
# example, in the following query:
# # @genqlient(n: "a")
#
# # @genqlient(n: "b")
Expand Down Expand Up @@ -39,16 +40,48 @@
# entire query (so "d", "e", and "f" take precedence over "b" and "c"), and
# multiple directives on the same node ("b" and "c") must not conflict. Note
# that directives on nodes do *not* apply to their "children", so "d" does not
# apply to the fields of MyInput, and "f" does not apply to field4.
# apply to the fields of MyInput, and "f" does not apply to field4. (But
# directives on operations and fragments do: both "b" and "c" apply to fields
# of MyInput and to field4.)
directive genqlient(

# If set, this argument will be omitted if it has an empty value, defined
# (the same as in encoding/json) as false, 0, a nil pointer, a nil interface
# value, and any empty array, slice, map, or string.
# If set to a string "MyType.myField", this entire @genqlient directive
# will be treated as if it were applied to the specified field of the
# specified type. It must be applied to an entire operation or fragment.
#
# This is especially useful for input-type options like omitempty and
# pointer, which are equally meaningful on input-type fields as on arguments,
# but there's no natural syntax to put them on fields.
#
# Note that for input types, unless the type has the "typename" option set,
# all operations and fragments in the same package which use this type should
# have matching directives. (This is to avoid needing to give them more
# complex type-names.) This is not currently validated, but will be
# validated in the future (see issue #123).
#
# For example, given the following query:
# # @genqlient(for: "MyInput.myField", omitempty: true)
# # @genqlient(for: "MyInput.myOtherField", pointer: true)
# # @genqlient(for: "MyOutput.id", bind: "path/to/pkg.MyOutputID")
# query MyQuery($arg: MyInput) { ... }
# genqlient will generate a type
# type MyInput struct {
# MyField <type> `json:"myField,omitempty"`
# MyOtherField *<type> `json:"myField"`
# MyThirdField <type> `json:"myThirdField"`
# }
# and use it for the argument to MyQuery, and similarly if `MyOutput.id` is
# ever requested in the response, it will be set to use the given type.
for: String

# If set, this argument (or input-type field, see "for") will be omitted if
# it has an empty value, defined (the same as in encoding/json) as false, 0,
# a nil pointer, a nil interface value, and any empty array, slice, map, or
# string.
#
# For example, given the following query:
# # @genqlient(omitempty: true)
# query MyQuery(arg: String) { ... }
# query MyQuery($arg: String) { ... }
# genqlient will generate a function
# MyQuery(ctx context.Context, client graphql.Client, arg string) ...
# which will pass {"arg": null} to GraphQL if arg is "", and the actual
Expand Down Expand Up @@ -161,8 +194,10 @@ directive genqlient(
# type-name in multiple places unless they request the exact same fields, or
# if your type-name conflicts with an autogenerated one (again, unless they
# request the exact same fields). They must even have the fields in the
# same order. Fragments are often easier to use (see the discussion of
# code-sharing in FAQ.md, and the "flatten" option above).
# same order. They should also have matching @genqlient directives, although
# this is not currently validated (see issue #123). Fragments are often
# easier to use (see the discussion of code-sharing in FAQ.md, and the
# "flatten" option above).
#
# Note that unlike most directives, if applied to the entire operation,
# typename affects the overall response type, rather than being propagated
Expand Down
27 changes: 14 additions & 13 deletions generate/convert.go
Original file line number Diff line number Diff line change
Expand Up @@ -164,11 +164,10 @@ func (g *generator) convertArguments(
name := "__" + operation.Name + "Input"
fields := make([]*goStructField, len(operation.VariableDefinitions))
for i, arg := range operation.VariableDefinitions {
_, directive, err := g.parsePrecedingComment(arg, arg.Position)
_, options, err := g.parsePrecedingComment(arg, nil, arg.Position, queryOptions)
if err != nil {
return nil, err
}
options := queryOptions.merge(directive)

goName := upperFirst(arg.Variable)
// Some of the arguments don't apply here, namely the name-prefix (see
Expand Down Expand Up @@ -386,19 +385,21 @@ func (g *generator) convertDefinition(
}

for i, field := range def.Fields {
_, fieldOptions, err := g.parsePrecedingComment(
field, def, field.Position, queryOptions)
if err != nil {
return nil, err
}

goName := upperFirst(field.Name)
// There are no field-specific options for inputs (yet, see #14),
// but we still need to merge with an empty directive to clear out
// any query-options that shouldn't apply here (namely "typename").
fieldOptions := queryOptions.merge(newGenqlientDirective(pos))
// Several of the arguments don't really make sense here:
// (note field.Type is necessarily a scalar, input, or enum)
// - namePrefix is ignored for input types and enums (see
// names.go) and for scalars (they use client-specified
// names)
// - selectionSet is ignored for input types, because we
// just use all fields of the type; and it's nonexistent
// for scalars and enums, our only other possible types,
// for scalars and enums, our only other possible types
// TODO(benkraft): Can we refactor to avoid passing the values that
// will be ignored? We know field.Type is a scalar, enum, or input
// type. But plumbing that is a bit tricky in practice.
Expand All @@ -414,8 +415,7 @@ func (g *generator) convertDefinition(
JSONName: field.Name,
GraphQLName: field.Name,
Description: field.Description,
// TODO(benkraft): set Omitempty once we have a way for the
// user to specify it.
Omitempty: fieldOptions.GetOmitempty(),
}
}
return goType, nil
Expand Down Expand Up @@ -506,12 +506,11 @@ func (g *generator) convertSelectionSet(
) ([]*goStructField, error) {
fields := make([]*goStructField, 0, len(selectionSet))
for _, selection := range selectionSet {
_, selectionDirective, err := g.parsePrecedingComment(
selection, selection.GetPosition())
_, selectionOptions, err := g.parsePrecedingComment(
selection, nil, selection.GetPosition(), queryOptions)
if err != nil {
return nil, err
}
selectionOptions := queryOptions.merge(selectionDirective)

switch selection := selection.(type) {
case *ast.Field:
Expand Down Expand Up @@ -705,6 +704,8 @@ func (g *generator) convertFragmentSpread(
}
}

// TODO(benkraft): Set directive here if we ever allow @genqlient
// directives on fragment-spreads.
return &goStructField{GoName: "" /* i.e. embedded */, GoType: typ}, nil
}

Expand All @@ -713,7 +714,7 @@ func (g *generator) convertFragmentSpread(
func (g *generator) convertNamedFragment(fragment *ast.FragmentDefinition) (goType, error) {
typ := g.schema.Types[fragment.TypeCondition]

comment, directive, err := g.parsePrecedingComment(fragment, fragment.Position)
comment, directive, err := g.parsePrecedingComment(fragment, nil, fragment.Position, nil)
if err != nil {
return nil, err
}
Expand Down
2 changes: 1 addition & 1 deletion generate/generate.go
Original file line number Diff line number Diff line change
Expand Up @@ -244,7 +244,7 @@ func (g *generator) addOperation(op *ast.OperationDefinition) error {
f := formatter.NewFormatter(&builder)
f.FormatQueryDocument(queryDoc)

commentLines, directive, err := g.parsePrecedingComment(op, op.Position)
commentLines, directive, err := g.parsePrecedingComment(op, nil, op.Position, nil)
if err != nil {
return err
}
Expand Down
Loading