Forbid __typename on subscription root (#1001, #1000)

This commit is contained in:
ilslv 2021-12-13 15:27:14 +03:00 committed by GitHub
parent 09da50b143
commit e264cf509d
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 86 additions and 28 deletions

View file

@ -1,14 +1,12 @@
//! Checks that `__typename` field queries okay on root types.
//! Checks that `__typename` field queries okay (and not okay) on root types.
//! See [#372](https://github.com/graphql-rust/juniper/issues/372) for details.
use futures::{stream, FutureExt as _};
use futures::stream;
use juniper::{
execute, graphql_object, graphql_subscription, graphql_value, graphql_vars,
resolve_into_stream, RootNode,
resolve_into_stream, GraphQLError, RootNode,
};
use crate::util::extract_next;
pub struct Query;
#[graphql_object]
@ -102,12 +100,23 @@ async fn subscription_typename() {
let schema = RootNode::new(Query, Mutation, Subscription);
match resolve_into_stream(query, None, &schema, &graphql_vars! {}, &()).await {
Err(GraphQLError::ValidationError(mut errors)) => {
assert_eq!(errors.len(), 1);
let err = errors.pop().unwrap();
assert_eq!(
resolve_into_stream(query, None, &schema, &graphql_vars! {}, &())
.then(|s| extract_next(s))
.await,
Ok((graphql_value!({"__typename": "Subscription"}), vec![])),
err.message(),
"`__typename` may not be included as a root field in a \
subscription operation",
);
assert_eq!(err.locations()[0].index(), 15);
assert_eq!(err.locations()[0].line(), 0);
assert_eq!(err.locations()[0].column(), 15);
}
_ => panic!("Expected ValidationError"),
};
}
#[tokio::test]
@ -116,10 +125,21 @@ async fn explicit_subscription_typename() {
let schema = RootNode::new(Query, Mutation, Subscription);
match resolve_into_stream(query, None, &schema, &graphql_vars! {}, &()).await {
Err(GraphQLError::ValidationError(mut errors)) => {
assert_eq!(errors.len(), 1);
let err = errors.pop().unwrap();
assert_eq!(
resolve_into_stream(query, None, &schema, &graphql_vars! {}, &())
.then(|s| extract_next(s))
.await,
Ok((graphql_value!({"__typename": "Subscription"}), vec![])),
err.message(),
"`__typename` may not be included as a root field in a \
subscription operation"
);
assert_eq!(err.locations()[0].index(), 28);
assert_eq!(err.locations()[0].line(), 0);
assert_eq!(err.locations()[0].column(), 28);
}
_ => panic!("Expected ValidationError"),
};
}

View file

@ -7,6 +7,7 @@
- `#[graphql_object]` and `#[graphql_subscription]` macros expansion now preserves defined `impl` blocks "as is" and reuses defined methods in opaque way. ([#971](https://github.com/graphql-rust/juniper/pull/971))
- `rename = "<policy>"` attribute's argument renamed to `rename_all = "<policy>"`. ([#971](https://github.com/graphql-rust/juniper/pull/971))
- Upgrade `bson` feature to [2.0 version of its crate](https://github.com/mongodb/bson-rust/releases/tag/v2.0.0). ([#979](https://github.com/graphql-rust/juniper/pull/979))
- Forbid `__typename` field on `subscription` operations [accordingly to October 2021 spec](https://spec.graphql.org/October2021/#note-bc213). ([#1001](https://github.com/graphql-rust/juniper/pull/1001), [#1000](https://github.com/graphql-rust/juniper/pull/1000))
## Features

View file

@ -1,4 +1,3 @@
use futures::{future, stream};
use serde::Serialize;
use crate::{
@ -293,16 +292,6 @@ where
let response_name = f.alias.as_ref().unwrap_or(&f.name).item;
if f.name.item == "__typename" {
let typename =
Value::scalar(instance.concrete_type_name(executor.context(), info));
object.add_field(
response_name,
Value::Scalar(Box::pin(stream::once(future::ok(typename)))),
);
continue;
}
let meta_field = meta_type
.field_by_name(f.name.item)
.unwrap_or_else(|| {

View file

@ -4,6 +4,7 @@ use crate::{
schema::meta::MetaType,
validation::{ValidatorContext, Visitor},
value::ScalarValue,
Operation, OperationType, Selection,
};
pub struct FieldsOnCorrectType;
@ -16,6 +17,27 @@ impl<'a, S> Visitor<'a, S> for FieldsOnCorrectType
where
S: ScalarValue,
{
fn enter_operation_definition(
&mut self,
context: &mut ValidatorContext<'a, S>,
operation: &'a Spanning<Operation<S>>,
) {
// https://spec.graphql.org/October2021/#note-bc213
if let OperationType::Subscription = operation.item.operation_type {
for selection in &operation.item.selection_set {
if let Selection::Field(field) = selection {
if field.item.name.item == "__typename" {
context.report_error(
"`__typename` may not be included as a root \
field in a subscription operation",
&[field.item.name.start],
);
}
}
}
}
}
fn enter_field(
&mut self,
context: &mut ValidatorContext<'a, S>,
@ -357,4 +379,30 @@ mod tests {
"#,
);
}
#[test]
fn forbids_typename_on_subscription() {
expect_fails_rule::<_, _, DefaultScalarValue>(
factory,
r#"subscription { __typename }"#,
&[RuleError::new(
"`__typename` may not be included as a root field in a \
subscription operation",
&[SourcePosition::new(15, 0, 15)],
)],
);
}
#[test]
fn forbids_typename_on_explicit_subscription() {
expect_fails_rule::<_, _, DefaultScalarValue>(
factory,
r#"subscription SubscriptionRoot { __typename }"#,
&[RuleError::new(
"`__typename` may not be included as a root field in a \
subscription operation",
&[SourcePosition::new(32, 0, 32)],
)],
);
}
}