Sort order as "type-then-name" in introspection output (#1239, #1134)

This commit is contained in:
Audun Halland 2024-01-17 15:04:17 +01:00 committed by GitHub
parent c54137f7b4
commit a1bc775e00
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
5 changed files with 1665 additions and 1627 deletions

View file

@ -103,6 +103,7 @@ All user visible changes to `juniper` crate will be documented in this file. Thi
- Incorrect error when explicit `null` provided for `null`able list input parameter. ([#1086], [#1085])
- Stack overflow on nested GraphQL fragments. ([CVE-2022-31173])
- Unstable definitions order in schema generated by `RootNode::as_sdl()`. ([#1237], [#1134])
- Unstable definitions order in schema generated by `introspect()` or other introspection queries. ([#1239], [#1134])
[#103]: /../../issues/103
[#113]: /../../issues/113
@ -165,6 +166,7 @@ All user visible changes to `juniper` crate will be documented in this file. Thi
[#1228]: /../../pull/1228
[#1235]: /../../pull/1235
[#1237]: /../../pull/1237
[#1239]: /../../pull/1239
[ba1ed85b]: /../../commit/ba1ed85b3c3dd77fbae7baf6bc4e693321a94083
[CVE-2022-31173]: /../../security/advisories/GHSA-4rx6-g5vg-5f3j

View file

@ -417,7 +417,13 @@ impl<'a, S> SchemaType<'a, S> {
/// Get a list of types.
pub fn type_list(&self) -> Vec<TypeType<S>> {
self.types.values().map(|t| TypeType::Concrete(t)).collect()
let mut types = self
.types
.values()
.map(|t| TypeType::Concrete(t))
.collect::<Vec<_>>();
sort_concrete_types(&mut types);
types
}
/// Get a list of concrete types.
@ -443,7 +449,9 @@ impl<'a, S> SchemaType<'a, S> {
/// Get a list of directives.
pub fn directive_list(&self) -> Vec<&DirectiveType<S>> {
self.directives.values().collect()
let mut directives = self.directives.values().collect::<Vec<_>>();
sort_directives(&mut directives);
directives
}
/// Get directive by name.
@ -685,6 +693,66 @@ impl<'a, S> fmt::Display for TypeType<'a, S> {
}
}
/// Sorts the provided [`TypeType`]s in the "type-then-name" manner.
fn sort_concrete_types<S>(types: &mut [TypeType<S>]) {
types.sort_by(|a, b| {
concrete_type_sort::by_type(a)
.cmp(&concrete_type_sort::by_type(b))
.then_with(|| concrete_type_sort::by_name(a).cmp(&concrete_type_sort::by_name(b)))
});
}
/// Sorts the provided [`DirectiveType`]s by name.
fn sort_directives<S>(directives: &mut [&DirectiveType<S>]) {
directives.sort_by(|a, b| a.name.cmp(&b.name));
}
/// Evaluation of a [`TypeType`] weights for sorting (for concrete types only).
///
/// Used for deterministic introspection output.
mod concrete_type_sort {
use crate::meta::MetaType;
use super::TypeType;
/// Returns a [`TypeType`] sorting weight by its type.
pub fn by_type<S>(t: &TypeType<S>) -> u8 {
match t {
TypeType::Concrete(MetaType::Enum(_)) => 0,
TypeType::Concrete(MetaType::InputObject(_)) => 1,
TypeType::Concrete(MetaType::Interface(_)) => 2,
TypeType::Concrete(MetaType::Scalar(_)) => 3,
TypeType::Concrete(MetaType::Object(_)) => 4,
TypeType::Concrete(MetaType::Union(_)) => 5,
// NOTE: The following types are not part of the introspected types.
TypeType::Concrete(
MetaType::List(_) | MetaType::Nullable(_) | MetaType::Placeholder(_),
) => 6,
// NOTE: Other variants will not appear since we're only sorting concrete types.
TypeType::List(..) | TypeType::NonNull(_) => 7,
}
}
/// Returns a [`TypeType`] sorting weight by its name.
pub fn by_name<'a, S>(t: &'a TypeType<'a, S>) -> Option<&'a str> {
match t {
TypeType::Concrete(MetaType::Enum(meta)) => Some(&meta.name),
TypeType::Concrete(MetaType::InputObject(meta)) => Some(&meta.name),
TypeType::Concrete(MetaType::Interface(meta)) => Some(&meta.name),
TypeType::Concrete(MetaType::Scalar(meta)) => Some(&meta.name),
TypeType::Concrete(MetaType::Object(meta)) => Some(&meta.name),
TypeType::Concrete(MetaType::Union(meta)) => Some(&meta.name),
TypeType::Concrete(
// NOTE: The following types are not part of the introspected types.
MetaType::List(_) | MetaType::Nullable(_) | MetaType::Placeholder(_),
)
// NOTE: Other variants will not appear since we're only sorting concrete types.
| TypeType::List(..)
| TypeType::NonNull(_) => None,
}
}
}
#[cfg(test)]
mod root_node_test {
#[cfg(feature = "schema-language")]

View file

@ -282,11 +282,11 @@ impl<'a, S: ScalarValue + 'a> TypeType<'a, S> {
TypeType::Concrete(&MetaType::Interface(InterfaceMeta {
name: ref iface_name,
..
})) => Some(
context
.concrete_type_list()
.iter()
.filter_map(|&ct| {
})) => {
let mut type_names = context
.types
.values()
.filter_map(|ct| {
if let MetaType::Object(ObjectMeta {
name,
interface_names,
@ -295,15 +295,21 @@ impl<'a, S: ScalarValue + 'a> TypeType<'a, S> {
{
interface_names
.iter()
.any(|name| name == iface_name)
.then(|| context.type_by_name(name))
.flatten()
.any(|iname| iname == iface_name)
.then(|| name.as_ref())
} else {
None
}
})
.collect(),
),
.collect::<Vec<_>>();
type_names.sort();
Some(
type_names
.into_iter()
.filter_map(|n| context.type_by_name(n))
.collect(),
)
}
_ => None,
}
}

View file

@ -1,5 +1,7 @@
use std::collections::HashSet;
use pretty_assertions::assert_eq;
use crate::{
graphql_vars,
introspection::IntrospectionFormat,
@ -184,14 +186,20 @@ async fn test_introspection_directives() {
EmptySubscription::<Database>::new(),
);
let mut result = crate::execute(q, None, &schema, &graphql_vars! {}, &database)
let result = crate::execute(q, None, &schema, &graphql_vars! {}, &database)
.await
.unwrap();
sort_schema_value(&mut result.0);
let mut expected = graphql_value!({
let expected = graphql_value!({
"__schema": {
"directives": [
{
"name": "deprecated",
"locations": [
"FIELD_DEFINITION",
"ENUM_VALUE",
],
},
{
"name": "include",
"locations": [
@ -208,13 +216,6 @@ async fn test_introspection_directives() {
"INLINE_FRAGMENT",
],
},
{
"name": "deprecated",
"locations": [
"FIELD_DEFINITION",
"ENUM_VALUE",
],
},
{
"name": "specifiedBy",
"locations": [
@ -224,7 +225,6 @@ async fn test_introspection_directives() {
],
},
});
sort_schema_value(&mut expected);
assert_eq!(result, (expected, vec![]));
}
@ -286,9 +286,9 @@ async fn test_builtin_introspection_query() {
EmptyMutation::<Database>::new(),
EmptySubscription::<Database>::new(),
);
let mut result = crate::introspect(&schema, &database, IntrospectionFormat::default()).unwrap();
sort_schema_value(&mut result.0);
let result = crate::introspect(&schema, &database, IntrospectionFormat::default()).unwrap();
let expected = schema_introspection_result();
assert_eq!(result, (expected, vec![]));
}
@ -301,9 +301,8 @@ async fn test_builtin_introspection_query_without_descriptions() {
EmptySubscription::<Database>::new(),
);
let mut result =
let result =
crate::introspect(&schema, &database, IntrospectionFormat::WithoutDescriptions).unwrap();
sort_schema_value(&mut result.0);
let expected = schema_introspection_result_without_descriptions();
assert_eq!(result, (expected, vec![]));

File diff suppressed because it is too large Load diff