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

Better handle errors from the client and detect multiple operations #435

Merged
merged 9 commits into from
May 9, 2024

Conversation

jerelmiller
Copy link
Member

Fixes #431

Better reports errors if they originate from Apollo Client itself rather than throwing an obscure error during the manifest generation. This change also adds detects multiple operations in a single document and will report those as errors.

@jerelmiller jerelmiller requested a review from glasser May 3, 2024 19:15
@jerelmiller jerelmiller requested a review from a team as a code owner May 3, 2024 19:15
Copy link

changeset-bot bot commented May 3, 2024

🦋 Changeset detected

Latest commit: 864fdef

The changes in this PR will be included in the next version bump.

This PR includes changesets to release 1 package
Name Type
@apollo/generate-persisted-query-manifest Patch

Not sure what this means? Click here to learn what changesets are.

Click here if you're a maintainer who wants to add another changeset to this PR

Copy link

codesandbox-ci bot commented May 3, 2024

This pull request is automatically built and testable in CodeSandbox.

To see build info of the built libraries, click here or the icon next to each commit SHA.

@jerelmiller jerelmiller force-pushed the jerel/handle-apollo-errors branch from ff1af21 to b4b6b92 Compare May 3, 2024 19:19
addError(source, ERROR_MESSAGES.multipleOperations());
return BREAK;
}

if (!name) {
Copy link
Member

Choose a reason for hiding this comment

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

Hmm, I'm wondering why we don't have any sort of error or warning on anonymous operations rather than just not including them in the manifest and having them break...

Copy link
Member Author

Choose a reason for hiding this comment

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

Hmmm, I'm not sure I follow what you're asking (or I'm reading your comment wrong). Currently we error on anonymous operations and force you to name all of them. Here is a test that shows this behavior:

test("errors on anonymous operations", async () => {
const { cleanup, runCommand, writeFile } = await setup();
const query = gql`
query {
greeting
}
`;
await writeFile("./src/query.graphql", print(query));
const { code, stderr } = await runCommand();
expect(code).toBe(1);
expect(stderr).toMatchInlineSnapshot(`
[
"src/query.graphql",
"1:1 error Anonymous GraphQL operations are not supported. Please name your query.",
"✖ 1 error",
]
`);
await cleanup();
});

Are you seeing something different in testing?

Copy link
Member Author

Choose a reason for hiding this comment

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

Finally realized what I was missing which is that this check was done in a different code block that traversed the AST in fromFilepathList. For now I moved the check for the number of operations into that same codeblock. It looks like GraphQL Codegen will split those operations into separate entries in the manifest.

If we decide to include a fail-safe and would prefer we don't allow this at all, we can move the check back and update the comment explaining why we do this check in the original place rather than moving it to the function that returns the document sources.

// We delegate validation to the functions that return the document sources.
// We just need to record the operations here to sort them in the manifest
// output.
visit(source.node, {
OperationDefinition(node) {
const name = node.name?.value;

if (++documentCount > 1) {
addError(source, ERROR_MESSAGES.multipleOperations());
Copy link
Member

Choose a reason for hiding this comment

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

Should we put the error on the second operation node as opposed to on "source" which appears in test to mean that all errors are showing up as 1:1?

Copy link
Member Author

@jerelmiller jerelmiller May 6, 2024

Choose a reason for hiding this comment

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

Looking back through the implementation of addError, it looks like we really don't care about node in that output and only use file and location as part of the error message, so it doesn't look like it would matter which node we use here (though I suppose we can override location to be the operation node location).

On the note about errors 1:1 with operation nodes, I considered that (and actually implemented that first), but I found this only makes sense if we include the operation name in the error output, otherwise the error just looks repetitive. Here is what something like that looks like:

  expect(stderr).toMatchInlineSnapshot(`
    [
      "src/mixed.graphql",
      "1:1  error  Cannot declare multiple operations in a single document. Please remove the 'TestQuery' operation or move it into a different document.",
      "1:1  error  Cannot declare multiple operations in a single document. Please remove the 'TestSubscription' operation or move it into a different document.",
      "src/mutations.graphql",
      "1:1  error  Cannot declare multiple operations in a single document. Please remove the 'SayGoodbye' operation or move it into a different document.",
      "src/queries.graphql",
      "1:1  error  Cannot declare multiple operations in a single document. Please remove the 'GoodbyeQuery' operation or move it into a different document.",
      "src/subscriptions.graphql",
      "1:1  error  Cannot declare multiple operations in a single document. Please remove the 'GoodbyeSubscription' operation or move it into a different document.",
      "✖ 5 errors",
    ]
  `);

vs what it is now:

  expect(stderr).toMatchInlineSnapshot(`
    [
      "src/mixed.graphql",
      "1:1  error  Cannot declare multiple operations in a single document.",
      "src/mutations.graphql",
      "1:1  error  Cannot declare multiple operations in a single document.",
      "src/queries.graphql",
      "1:1  error  Cannot declare multiple operations in a single document.",
      "src/subscriptions.graphql",
      "1:1  error  Cannot declare multiple operations in a single document.",
      "✖ 4 errors",
    ]
  `);

I went with the simpler approach because it felt pretty self explanatory on what action you needed to take, but I can be convinced differently.

Thoughts?

@jerelmiller
Copy link
Member Author

jerelmiller commented May 8, 2024

@glasser would love for you to take another look here when you have some time 🙂. Let me know if the latest change alleviates your concerns.

@jerelmiller jerelmiller merged commit 9c3d761 into main May 9, 2024
11 checks passed
@jerelmiller jerelmiller deleted the jerel/handle-apollo-errors branch May 9, 2024 23:37
@github-actions github-actions bot mentioned this pull request May 9, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
2 participants