use super::ValidationRule;
use crate::ast::{visit_document, OperationVisitor, OperationVisitorContext};
use crate::static_graphql::query::*;
use crate::validation::utils::{ValidationError, ValidationErrorContext};
pub struct LoneAnonymousOperation;
impl LoneAnonymousOperation {
pub fn new() -> Self {
LoneAnonymousOperation
}
}
impl<'a> OperationVisitor<'a, ValidationErrorContext> for LoneAnonymousOperation {
fn enter_document(
&mut self,
_: &mut OperationVisitorContext,
user_context: &mut ValidationErrorContext,
document: &Document,
) {
let operations_count = document
.definitions
.iter()
.filter(|n| match n {
Definition::Operation(OperationDefinition::SelectionSet(_)) => true,
Definition::Operation(OperationDefinition::Query(_)) => true,
Definition::Operation(OperationDefinition::Mutation(_)) => true,
Definition::Operation(OperationDefinition::Subscription(_)) => true,
_ => false,
})
.count();
for definition in &document.definitions {
match definition {
Definition::Operation(OperationDefinition::SelectionSet(_)) => {
if operations_count > 1 {
user_context.report_error(ValidationError {error_code: self.error_code(),
message: "This anonymous operation must be the only defined operation."
.to_string(),
locations: vec![],
})
}
}
Definition::Operation(OperationDefinition::Query(query)) => {
if query.name == None && operations_count > 1 {
user_context.report_error(ValidationError {error_code: self.error_code(),
message: "This anonymous operation must be the only defined operation."
.to_string(),
locations: vec![query.position.clone()],
})
}
}
Definition::Operation(OperationDefinition::Mutation(mutation)) => {
if mutation.name == None && operations_count > 1 {
user_context.report_error(ValidationError {error_code: self.error_code(),
message: "This anonymous operation must be the only defined operation."
.to_string(),
locations: vec![mutation.position.clone()],
})
}
}
Definition::Operation(OperationDefinition::Subscription(subscription)) => {
if subscription.name == None && operations_count > 1 {
user_context.report_error(ValidationError {error_code: self.error_code(),
message: "This anonymous operation must be the only defined operation."
.to_string(),
locations: vec![subscription.position.clone()],
})
}
}
_ => {}
};
}
}
}
impl ValidationRule for LoneAnonymousOperation {
fn error_code<'a>(&self) -> &'a str {
"LoneAnonymousOperation"
}
fn validate<'a>(
&self,
ctx: &'a mut OperationVisitorContext,
error_collector: &mut ValidationErrorContext,
) {
visit_document(
&mut LoneAnonymousOperation::new(),
&ctx.operation,
ctx,
error_collector,
);
}
}
#[test]
fn no_operations() {
use crate::validation::test_utils::*;
let mut plan = create_plan_from_rule(Box::new(LoneAnonymousOperation {}));
let errors = test_operation_with_schema(
"fragment fragA on Type {
field
}",
TEST_SCHEMA,
&mut plan,
);
assert_eq!(get_messages(&errors).len(), 0);
}
#[test]
fn one_anon_operation() {
use crate::validation::test_utils::*;
let mut plan = create_plan_from_rule(Box::new(LoneAnonymousOperation {}));
let errors = test_operation_with_schema(
"{
field
}",
TEST_SCHEMA,
&mut plan,
);
assert_eq!(get_messages(&errors).len(), 0);
}
#[test]
fn mutiple_named() {
use crate::validation::test_utils::*;
let mut plan = create_plan_from_rule(Box::new(LoneAnonymousOperation {}));
let errors = test_operation_with_schema(
"query Foo {
field
}
query Bar {
field
}",
TEST_SCHEMA,
&mut plan,
);
assert_eq!(get_messages(&errors).len(), 0);
}
#[test]
fn anon_operation_with_fragment() {
use crate::validation::test_utils::*;
let mut plan = create_plan_from_rule(Box::new(LoneAnonymousOperation {}));
let errors = test_operation_with_schema(
"{
...Foo
}
fragment Foo on Type {
field
}",
TEST_SCHEMA,
&mut plan,
);
assert_eq!(get_messages(&errors).len(), 0);
}
#[test]
fn multiple_anon_operations() {
use crate::validation::test_utils::*;
let mut plan = create_plan_from_rule(Box::new(LoneAnonymousOperation {}));
let errors = test_operation_with_schema(
"{
fieldA
}
{
fieldB
}",
TEST_SCHEMA,
&mut plan,
);
let messages = get_messages(&errors);
assert_eq!(messages.len(), 2);
assert_eq!(
messages,
vec![
"This anonymous operation must be the only defined operation.",
"This anonymous operation must be the only defined operation."
]
);
}
#[test]
fn anon_operation_with_mutation() {
use crate::validation::test_utils::*;
let mut plan = create_plan_from_rule(Box::new(LoneAnonymousOperation {}));
let errors = test_operation_with_schema(
"{
fieldA
}
mutation Foo {
fieldB
}",
TEST_SCHEMA,
&mut plan,
);
let messages = get_messages(&errors);
assert_eq!(messages.len(), 1);
assert_eq!(
messages,
vec!["This anonymous operation must be the only defined operation."]
);
}
#[test]
fn anon_operation_with_subscription() {
use crate::validation::test_utils::*;
let mut plan = create_plan_from_rule(Box::new(LoneAnonymousOperation {}));
let errors = test_operation_with_schema(
"{
fieldA
}
subscription Foo {
fieldB
}",
TEST_SCHEMA,
&mut plan,
);
let messages = get_messages(&errors);
assert_eq!(messages.len(), 1);
assert_eq!(
messages,
vec!["This anonymous operation must be the only defined operation."]
);
}