-
-
Notifications
You must be signed in to change notification settings - Fork 527
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(graphql_analyze): noAnonymousOperations
Disallow operations without name, since GraphQL clients usually use operations' name to cache.
- Loading branch information
1 parent
7fffb27
commit aa33e72
Showing
16 changed files
with
361 additions
and
100 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
220 changes: 121 additions & 99 deletions
220
crates/biome_configuration/src/analyzer/linter/rules.rs
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
120 changes: 120 additions & 0 deletions
120
crates/biome_graphql_analyze/src/lint/nursery/no_anonymous_operations.rs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,120 @@ | ||
use biome_analyze::{ | ||
context::RuleContext, declare_lint_rule, ActionCategory, Ast, FixKind, Rule, RuleDiagnostic, | ||
RuleSource, RuleSourceKind, | ||
}; | ||
use biome_console::markup; | ||
use biome_graphql_factory::make; | ||
use biome_graphql_syntax::GraphqlOperationDefinition; | ||
use biome_rowan::{AstNode, AstNodeExt, BatchMutationExt}; | ||
use biome_string_case::Case; | ||
|
||
use crate::GraphqlRuleAction; | ||
|
||
declare_lint_rule! { | ||
/// Require specifying name for GraphQL operations. | ||
/// | ||
/// This is useful since most GraphQL client libraries are using the operation name for caching purposes.. | ||
/// | ||
/// ## Examples | ||
/// | ||
/// ### Invalid | ||
/// | ||
/// ```graphql,expect_diagnostic | ||
/// query {} | ||
/// ``` | ||
/// | ||
/// ### Valid | ||
/// | ||
/// ```graphql | ||
/// query Human { | ||
/// name | ||
/// } | ||
/// ``` | ||
/// | ||
pub NoAnonymousOperations { | ||
version: "next", | ||
name: "noAnonymousOperations", | ||
language: "graphql", | ||
sources: &[RuleSource::EslintGraphql("no-anonymous-operations")], | ||
source_kind: RuleSourceKind::SameLogic, | ||
recommended: true, | ||
fix_kind: FixKind::Unsafe, | ||
} | ||
} | ||
|
||
impl Rule for NoAnonymousOperations { | ||
type Query = Ast<GraphqlOperationDefinition>; | ||
type State = NoAnonymousOperationsState; | ||
type Signals = Option<Self::State>; | ||
type Options = (); | ||
|
||
fn run(ctx: &RuleContext<Self>) -> Option<Self::State> { | ||
let node = ctx.query(); | ||
let operation_type = node.ty().ok()?.text(); | ||
if node.name().is_some() { | ||
None | ||
} else { | ||
Some(NoAnonymousOperationsState { | ||
operation_type: operation_type.clone(), | ||
suggested_name: get_suggested_name(node, operation_type), | ||
}) | ||
} | ||
} | ||
|
||
fn diagnostic(ctx: &RuleContext<Self>, state: &Self::State) -> Option<RuleDiagnostic> { | ||
let node = ctx.query(); | ||
|
||
Some( | ||
RuleDiagnostic::new( | ||
rule_category!(), | ||
node.range(), | ||
markup! { | ||
"Anonymous GraphQL operations are forbidden. Make sure to name your " {state.operation_type}"!" | ||
}, | ||
) | ||
.note(markup! { | ||
"Rename this "{state.operation_type}" to "{state.suggested_name}"." | ||
}), | ||
) | ||
} | ||
|
||
fn action(ctx: &RuleContext<Self>, state: &Self::State) -> Option<GraphqlRuleAction> { | ||
let mut mutation = ctx.root().begin(); | ||
let node = ctx.query().clone(); | ||
let new_name = make::graphql_name_binding(make::ident(&state.suggested_name)); | ||
let new_node = node.clone().detach().with_name(Some(new_name)); | ||
mutation.replace_node(node, new_node); | ||
|
||
Some(GraphqlRuleAction::new( | ||
ActionCategory::QuickFix, | ||
ctx.metadata().applicability(), | ||
markup! { | ||
"Rename this "{state.operation_type}" to "{state.suggested_name}"." | ||
}, | ||
mutation, | ||
)) | ||
} | ||
} | ||
|
||
fn get_suggested_name(operation: &GraphqlOperationDefinition, operation_type: String) -> String { | ||
let suggested_name = get_suggested_name_base_on_content(operation).unwrap_or(operation_type); | ||
Case::Pascal.convert(&suggested_name) | ||
} | ||
|
||
fn get_suggested_name_base_on_content(operation: &GraphqlOperationDefinition) -> Option<String> { | ||
let selection_set = operation.selection_set().ok()?; | ||
let first_field = selection_set | ||
.selections() | ||
.into_iter() | ||
.find_map(|selection| selection.as_graphql_field().cloned())?; | ||
|
||
first_field | ||
.alias() | ||
.map(|alias| alias.text()) | ||
.or(first_field.name().ok().map(|name| name.text())) | ||
} | ||
|
||
pub struct NoAnonymousOperationsState { | ||
operation_type: String, | ||
suggested_name: String, | ||
} |
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
3 changes: 3 additions & 0 deletions
3
crates/biome_graphql_analyze/tests/specs/nursery/noAnonymousOperations/invalid.graphql
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
query { human } | ||
mutation { ...Type } | ||
subscription {} |
71 changes: 71 additions & 0 deletions
71
crates/biome_graphql_analyze/tests/specs/nursery/noAnonymousOperations/invalid.graphql.snap
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,71 @@ | ||
--- | ||
source: crates/biome_graphql_analyze/tests/spec_tests.rs | ||
expression: invalid.graphql | ||
--- | ||
# Input | ||
```graphql | ||
query { human } | ||
mutation { ...Type } | ||
subscription {} | ||
``` | ||
|
||
# Diagnostics | ||
``` | ||
invalid.graphql:1:1 lint/nursery/noAnonymousOperations FIXABLE ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ | ||
! Anonymous GraphQL operations are forbidden. Make sure to name your query! | ||
> 1 │ query { human } | ||
│ ^^^^^^^^^^^^^^^ | ||
2 │ mutation { ...Type } | ||
3 │ subscription {} | ||
i Rename this query to Human. | ||
i Unsafe fix: Rename this query to Human. | ||
1 │ query·Human{·human·} | ||
│ +++++ | ||
``` | ||
|
||
``` | ||
invalid.graphql:2:1 lint/nursery/noAnonymousOperations FIXABLE ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ | ||
! Anonymous GraphQL operations are forbidden. Make sure to name your mutation! | ||
1 │ query { human } | ||
> 2 │ mutation { ...Type } | ||
│ ^^^^^^^^^^^^^^^^^^^^ | ||
3 │ subscription {} | ||
4 │ | ||
i Rename this mutation to Mutation. | ||
i Unsafe fix: Rename this mutation to Mutation. | ||
2 │ mutation·Mutation{·...Type·} | ||
│ ++++++++ | ||
``` | ||
|
||
``` | ||
invalid.graphql:3:1 lint/nursery/noAnonymousOperations FIXABLE ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ | ||
! Anonymous GraphQL operations are forbidden. Make sure to name your subscription! | ||
1 │ query { human } | ||
2 │ mutation { ...Type } | ||
> 3 │ subscription {} | ||
│ ^^^^^^^^^^^^^^^ | ||
4 │ | ||
i Rename this subscription to Subscription. | ||
i Unsafe fix: Rename this subscription to Subscription. | ||
3 │ subscription·Subscription{} | ||
│ ++++++++++++ | ||
``` |
4 changes: 4 additions & 0 deletions
4
crates/biome_graphql_analyze/tests/specs/nursery/noAnonymousOperations/valid.graphql
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
/* should not generate diagnostics */ | ||
query myQuery { a } | ||
mutation doSomething { a } | ||
subscription myData { a } |
12 changes: 12 additions & 0 deletions
12
crates/biome_graphql_analyze/tests/specs/nursery/noAnonymousOperations/valid.graphql.snap
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
--- | ||
source: crates/biome_graphql_analyze/tests/spec_tests.rs | ||
expression: valid.graphql | ||
--- | ||
# Input | ||
```graphql | ||
/* should not generate diagnostics */ | ||
query myQuery { a } | ||
mutation doSomething { a } | ||
subscription myData { a } | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
pub use crate::generated::node_factory::*; | ||
use biome_graphql_syntax::{GraphqlSyntaxKind, GraphqlSyntaxToken}; | ||
|
||
/// Create a new literal name token with no attached trivia | ||
pub fn ident(text: &str) -> GraphqlSyntaxToken { | ||
GraphqlSyntaxToken::new_detached(GraphqlSyntaxKind::IDENT, text, [], []) | ||
} |
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.