Skip to content

Conversation

@benjie
Copy link
Member

@benjie benjie commented Sep 11, 2023

@robrichard raised an issue for @defer whereby under the current draft specification if you have this query:

query HeroNameQuery {
  hero {
    ...NameFragment
    ...NameFragment @defer
  }
}
fragment NameFragment on Hero {
  name
}

then the @defer will never be visited; but if you move the @defer earlier:

query HeroNameQuery {
  hero {
    ...NameFragment @defer
    ...NameFragment
  }
}
fragment NameFragment on Hero {
  name
}

then the non-deferred version will never be visited.


@mjmahone has an RFC open about fragment arguments which suffers a similar issue: visits to a fragment aren't equivalent depending on the variables.

query HeroNameQuery {
  hero {
    ...ComponentOne
    ...ComponentTwo
  }
}
fragment ComponentOne on Hero {
  id
  ...HeroDetails(includeBio: true, includeAvatar: false)
}
fragment ComponentTwo on Hero {
  ...HeroDetails(includeBio: false, includeAvatar: true)
}
fragment HeroDetails($includeBio: Boolean!, $includeAvatar: Boolean!) on Hero {
  superheroName
  realName
  initials
  ... @include(if: $includeAvatar) {
    avatarLarge 
    avatarSmall
  }
  ... @include(if: $includeBio) {
    bioTitle
    bioSubtitle
    bioEntryParagraph
    bioQuote
    bioMainText
  }
}

Matt's solution takes a Relay-esque approach and generates a key for the fragments based on the values of their arguments.


Another consideration is where there are user-defined directives; e.g. { id ...Profile name ...Profile @live } might benefit from walking Profile again now that it has @live attached.

A simple solution to this is rather than each of these problems having their own solution, to simply navigate the fragment spread again - it's only a single layer that we need to worry about, so the cost is likely to be marginal. This RFC proposes this with a tiny change to the spec - after adding the fragmentSpreadName to visitedFragments, and calling CollectFields, we then remove it again.

@benjie benjie added the 💭 Strawman (RFC 0) RFC Stage 0 (See CONTRIBUTING.md) label Sep 11, 2023
@netlify
Copy link

netlify bot commented Sep 11, 2023

Deploy Preview for graphql-spec-draft ready!

Name Link
🔨 Latest commit 34ac2da
🔍 Latest deploy log https://app.netlify.com/projects/graphql-spec-draft/deploys/68668c6cc7979c000782ed61
😎 Deploy Preview https://deploy-preview-1045--graphql-spec-draft.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify project configuration.

@martinbonnin
Copy link
Contributor

Interesing!

Another consideration is where there are user-defined directives; e.g. { id ...Profile name ...Profile @live } might benefit from walking Profile again now that it has @live attached.

Do we have the same issue for fields? Assuming @live can be applied to fields, am I correct assuming the second name will not reach the resolver here?

{
  id
  name 
  # This is not passed to the resolver
  name @live
}

Because of this:

ExecuteField
1. Let field be the first entry in fields.

So the second name is still visited by CollectFields but not ExecuteField. If @live is important to the resolver, it is lost forever?

@benjie
Copy link
Member Author

benjie commented Sep 11, 2023

@martinbonnin Speaking from GraphQL.js' POV the entire of fields is made available to the resolver (via resolveInfo), so the resolver can determine that @live is present on some of the selections that represent the currently executing field. GraphQL spec does not pass any directives to resolvers, independent of order, so @live is never indicated to a resolver - that's something for the execution engine to take care of.

@benjie
Copy link
Member Author

benjie commented Sep 11, 2023

Counterpoint to my proposal:

query Malicious {
  listOfStuff {
    ...ManyFields
    ...ManyFields
    ...ManyFields
    ...ManyFields
    ...ManyFields
    ...ManyFields
    ...ManyFields
    ...ManyFields
    ...ManyFields
    ...ManyFields
    ...ManyFields
    ...ManyFields
  }    
}
fragment ManyFields on ListItem {
  a
  b
  c
  # ...
  zx
  zy
  zz
}

This could multiply up expensiveness easily. But then again, assuming it's memoized it'd be negligible.

@yaacovCR
Copy link
Contributor

Really interesting!

Additional potential counterpoint

Below example is attempt to explore how nested (but non-cyclic!) fragments might be handled.

Nested fragment depth is linear with respect to overall query length, but server field collection work would be non-linear with respect to the query length, potentially opening up an attack vector.

query Malicious {
  object {
    ...F1
    ...<10 more times>...
    ...F1
  }    
}

fragment F1 on SomeObject {
  ...F2
    ...<10 more times>...
  ...F2
}

fragment F2 on SomeObject {
  ...F3
    ...<10 more times>...
  ...F3
}

fragment F3 on SomeObject {
  ...F4
  ...<10 more times>...
  ...F4
}

fragment F4 on SomeObject {
  ...F5
  ...<10 more times>...
  ...F5
}

fragment F5 on SomeObject {
  ...FewFields
  ...<10 more times>...
  ...FewFields
}

fragment FewFields on SomeObject {
  a
  b
  c
}

Definitely appreciate the idea of a common solution.

One approach would be to adopt the fragment argument Relay-esque approach to include execution-level directives on fragments like @defer (and potentially @live?). This works completely with regard to fragment arguments, where every new combination of arguments is meaningful, but doesn't quite make sense with regard to @defer (and presumably @live) in that a fragment spread without @defer supersedes a spread with @defer (and presumably the opposite in the case of @live.

For differing directive usage on sibling spreads, I see 3 different scenarios:

  1. Fragment spreads should fail validation if they inconsistently use a directive (akin to how we have a validation rule requiring stream directive usage to be identical for overlapping fields).
  2. Fragment spreads that only sometimes use the directive should be considered AS NOT using the directive => desired behavior for @defer.
  3. Fragment spreads that only sometimes use the directive should be considered AS IF using the directive => desired behavior for @live.

Self-referencing fragments with differing arguments/directives

Another point to consider is that presumably self-referencing fragment spreads with different arguments should also be allowed, as they do not form a true cycle. So we may require the Relay-esque style behavior anyway. Modifying your example above:

fragment ComponentOne on Hero {
  id
  ...HeroDetails(includeBio: true, includeAvatar: false)
  ...SubComponentA
}
fragment SubComponentA on Hero {
  ...HeroDetails(includeBio: false, includeAvatar: true)
}

For @defer, currently nesting a deferred fragment at the same level under a deferred or vice a versa would be meaningless. For @live, it could be used to ensure that wherever the fragment is used, it would also get the directive. Not sure if that should be encouraged!

@benjie
Copy link
Member Author

benjie commented Sep 12, 2023

I don't think we'll allow a fragment to reference itself even if it has different arguments.

I think we should allow fragments to be specified multiple times with different directives, since different components may use the same data in different ways (e.g. the logged in user's avatar may be rendered in multiple places on a web page, in some it may be deferrable, and others not - that's up to the components themselves to decide). The issue of having to align ! and ? across components is one that bothers me with CCN, but that's a different topic 😉

Assuming field collection is memoized (where the implementation knows to key it by the arguments and directives applied) then your non-linear example wouldn't be much of an issue. I considered explicitly defining this in the spec like Matt does for fragment arguments, but the risk is that someone would use a Base64-encoded avatar image as an argument to a fragment, and suddenly in your malicious query example but extended to use fragment arguments we'd have to generate that large key many times!

A more subtle option could be to track "fragments with no arguments and no directives" separately from those with either or both.

@yaacovCR
Copy link
Contributor

yaacovCR commented Sep 12, 2023 via email

@benjie
Copy link
Member Author

benjie commented Sep 12, 2023

Yep, exact same page on that! (From an observable behavior point of view. Not necessarily from the algorithm's point of view.)

@benjie
Copy link
Member Author

benjie commented Jul 17, 2025

This was rejected at todays WG in favour of changing visitedFragments to a string-boolean map.

The map would be populated such that (pseudocode):

  • Let {deferred} be {true} if the fragment is deferred, {false} otherwise.
  • Let {visited} be the value for key {fragmentSpreadName} in {visitedFragments}.
  • If either {visited} does not exist, or if both {visited} is {true} and {deferred} is {false} then:
    • Set the value for {fragmentSpreadName} in {visitedFragments} to {deferred} (overwriting if necessary).
    • Visit the fragment.

@benjie benjie closed this Jul 17, 2025
@github-project-automation github-project-automation bot moved this from Needs representation at WG to Done in Benjie's GraphQL tasks Jul 17, 2025
@benjie benjie added 🗑 Rejected (RFC X) RFC Stage X (See CONTRIBUTING.md) and removed 💭 Strawman (RFC 0) RFC Stage 0 (See CONTRIBUTING.md) labels Jul 17, 2025
yaacovCR added a commit to yaacovCR/graphql-js that referenced this pull request Jul 22, 2025
This PR changes the fragment spread visiting logic to avoid re-visiting fragment spreads whenever possible.

Previously, fragment spreads that have been visited in a non-deferred capacity were not revisited, but fragment spreads that had been visited in a deferred capacity could be revisited.

This PR updates the logic so that fragment spreads that had been visited in a deferred capacity are not revisited in a deferred capacity.

See graphql/graphql-spec#1045 (comment)
@yaacovCR
Copy link
Contributor

in favour of changing visitedFragments to a string-boolean map.

implemented in graphql/graphql-js#4462

yaacovCR added a commit to graphql/graphql-js that referenced this pull request Jul 24, 2025
This PR changes the fragment spread visiting logic to avoid re-visiting fragment spreads in additional cases.

Previously, fragment spreads that have been visited in a non-deferred capacity were not revisited, but fragment spreads that had been visited in a deferred capacity could be revisited.

This PR updates the logic so that fragment spreads that had been visited in a deferred capacity are not revisited in a deferred capacity.

See: graphql/graphql-spec#1045 (comment)
yaacovCR added a commit to yaacovCR/graphql-js that referenced this pull request Jul 30, 2025
This PR changes the fragment spread visiting logic to avoid re-visiting fragment spreads in additional cases.

Previously, fragment spreads that have been visited in a non-deferred capacity were not revisited, but fragment spreads that had been visited in a deferred capacity could be revisited.

This PR updates the logic so that fragment spreads that had been visited in a deferred capacity are not revisited in a deferred capacity.

See: graphql/graphql-spec#1045 (comment)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

🗑 Rejected (RFC X) RFC Stage X (See CONTRIBUTING.md)

Projects

None yet

Development

Successfully merging this pull request may close these issues.

4 participants