parent
e740646b8c
commit
d1d5829b44
4 changed files with 107 additions and 10 deletions
|
@ -13,6 +13,7 @@ use schema::meta::{Argument, EnumMeta, EnumValue, Field, InputObjectMeta, Interf
|
||||||
use schema::model::{RootNode, SchemaType};
|
use schema::model::{RootNode, SchemaType};
|
||||||
|
|
||||||
use types::base::GraphQLType;
|
use types::base::GraphQLType;
|
||||||
|
use types::name::Name;
|
||||||
|
|
||||||
/// A type registry used to build schemas
|
/// A type registry used to build schemas
|
||||||
///
|
///
|
||||||
|
@ -21,7 +22,7 @@ use types::base::GraphQLType;
|
||||||
/// into `Type` instances and automatically registers them.
|
/// into `Type` instances and automatically registers them.
|
||||||
pub struct Registry<'r> {
|
pub struct Registry<'r> {
|
||||||
/// Currently registered types
|
/// Currently registered types
|
||||||
pub types: HashMap<String, MetaType<'r>>,
|
pub types: HashMap<Name, MetaType<'r>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
|
@ -408,7 +409,7 @@ where
|
||||||
|
|
||||||
impl<'r> Registry<'r> {
|
impl<'r> Registry<'r> {
|
||||||
/// Construct a new registry
|
/// Construct a new registry
|
||||||
pub fn new(types: HashMap<String, MetaType<'r>>) -> Registry<'r> {
|
pub fn new(types: HashMap<Name, MetaType<'r>>) -> Registry<'r> {
|
||||||
Registry { types: types }
|
Registry { types: types }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -421,12 +422,16 @@ impl<'r> Registry<'r> {
|
||||||
T: GraphQLType,
|
T: GraphQLType,
|
||||||
{
|
{
|
||||||
if let Some(name) = T::name(info) {
|
if let Some(name) = T::name(info) {
|
||||||
if !self.types.contains_key(&name.to_string()) {
|
let validated_name = name.parse::<Name>().unwrap();
|
||||||
self.insert_placeholder(&name, Type::NonNullNamed(Cow::Owned(name.to_string())));
|
if !self.types.contains_key(name) {
|
||||||
|
self.insert_placeholder(
|
||||||
|
validated_name.clone(),
|
||||||
|
Type::NonNullNamed(Cow::Owned(name.to_string())),
|
||||||
|
);
|
||||||
let meta = T::meta(info, self);
|
let meta = T::meta(info, self);
|
||||||
self.types.insert(name.to_string(), meta);
|
self.types.insert(validated_name, meta);
|
||||||
}
|
}
|
||||||
self.types[&name.to_string()].as_type()
|
self.types[name].as_type()
|
||||||
} else {
|
} else {
|
||||||
T::meta(info, self).as_type()
|
T::meta(info, self).as_type()
|
||||||
}
|
}
|
||||||
|
@ -483,10 +488,10 @@ impl<'r> Registry<'r> {
|
||||||
Argument::new(name, self.get_type::<Option<T>>(info)).default_value(value.to())
|
Argument::new(name, self.get_type::<Option<T>>(info)).default_value(value.to())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn insert_placeholder(&mut self, name: &str, of_type: Type<'r>) {
|
fn insert_placeholder(&mut self, name: Name, of_type: Type<'r>) {
|
||||||
if !self.types.contains_key(name) {
|
if !self.types.contains_key(&name) {
|
||||||
self.types.insert(
|
self.types.insert(
|
||||||
name.to_owned(),
|
name,
|
||||||
MetaType::Placeholder(PlaceholderMeta { of_type: of_type }),
|
MetaType::Placeholder(PlaceholderMeta { of_type: of_type }),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,6 +2,7 @@ use std::collections::HashMap;
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
|
|
||||||
use types::base::GraphQLType;
|
use types::base::GraphQLType;
|
||||||
|
use types::name::Name;
|
||||||
use executor::{Context, Registry};
|
use executor::{Context, Registry};
|
||||||
use ast::Type;
|
use ast::Type;
|
||||||
use schema::meta::{Argument, InterfaceMeta, MetaType, ObjectMeta, PlaceholderMeta, UnionMeta};
|
use schema::meta::{Argument, InterfaceMeta, MetaType, ObjectMeta, PlaceholderMeta, UnionMeta};
|
||||||
|
@ -25,7 +26,7 @@ pub struct RootNode<'a, QueryT: GraphQLType, MutationT: GraphQLType> {
|
||||||
|
|
||||||
/// Metadata for a schema
|
/// Metadata for a schema
|
||||||
pub struct SchemaType<'a> {
|
pub struct SchemaType<'a> {
|
||||||
types: HashMap<String, MetaType<'a>>,
|
types: HashMap<Name, MetaType<'a>>,
|
||||||
query_type_name: String,
|
query_type_name: String,
|
||||||
mutation_type_name: Option<String>,
|
mutation_type_name: Option<String>,
|
||||||
directives: HashMap<String, DirectiveType<'a>>,
|
directives: HashMap<String, DirectiveType<'a>>,
|
||||||
|
|
|
@ -3,3 +3,4 @@ pub mod scalars;
|
||||||
pub mod pointers;
|
pub mod pointers;
|
||||||
pub mod containers;
|
pub mod containers;
|
||||||
pub mod utilities;
|
pub mod utilities;
|
||||||
|
pub mod name;
|
||||||
|
|
90
juniper/src/types/name.rs
Normal file
90
juniper/src/types/name.rs
Normal file
|
@ -0,0 +1,90 @@
|
||||||
|
use std::borrow::Borrow;
|
||||||
|
use std::error::Error;
|
||||||
|
use std::fmt::{Display, Formatter, Result as FmtResult};
|
||||||
|
use std::str::FromStr;
|
||||||
|
|
||||||
|
// Helper functions until the corresponding AsciiExt methods
|
||||||
|
// stabilise (https://github.com/rust-lang/rust/issues/39658).
|
||||||
|
|
||||||
|
fn is_ascii_alphabetic(c: char) -> bool {
|
||||||
|
return c >= 'a' && c <= 'z' || c >= 'A' && c <= 'Z';
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_ascii_digit(c: char) -> bool {
|
||||||
|
return c >= '0' && c <= '9';
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
||||||
|
pub struct Name(String);
|
||||||
|
|
||||||
|
impl Name {
|
||||||
|
pub fn is_valid(input: &str) -> bool {
|
||||||
|
for (i, c) in input.chars().enumerate() {
|
||||||
|
if i == 0 {
|
||||||
|
if !is_ascii_alphabetic(c) && c != '_' {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if !is_ascii_alphabetic(c) && !is_ascii_digit(c) && c != '_' {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return input.len() > 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
|
||||||
|
pub struct NameParseError(String);
|
||||||
|
|
||||||
|
impl Display for NameParseError {
|
||||||
|
fn fmt(&self, f: &mut Formatter) -> FmtResult {
|
||||||
|
self.0.fmt(f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Error for NameParseError {
|
||||||
|
fn description(&self) -> &str {
|
||||||
|
&self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FromStr for Name {
|
||||||
|
type Err = NameParseError;
|
||||||
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
||||||
|
if Name::is_valid(s) {
|
||||||
|
Ok(Name(s.to_string()))
|
||||||
|
} else {
|
||||||
|
Err(NameParseError(format!(
|
||||||
|
"Names must match /^[_a-zA-Z][_a-zA-Z0-9]*$/ but \"{}\" does not",
|
||||||
|
s
|
||||||
|
)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Borrow<String> for Name {
|
||||||
|
fn borrow(&self) -> &String {
|
||||||
|
&self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Borrow<str> for Name {
|
||||||
|
fn borrow(&self) -> &str {
|
||||||
|
&self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_name_is_valid() {
|
||||||
|
assert!(Name::is_valid("Foo"));
|
||||||
|
assert!(Name::is_valid("foo42"));
|
||||||
|
assert!(Name::is_valid("_Foo"));
|
||||||
|
assert!(Name::is_valid("_Foo42"));
|
||||||
|
assert!(Name::is_valid("_foo42"));
|
||||||
|
assert!(Name::is_valid("_42Foo"));
|
||||||
|
|
||||||
|
assert!(!Name::is_valid("42_Foo"));
|
||||||
|
assert!(!Name::is_valid("Foo-42"));
|
||||||
|
assert!(!Name::is_valid("Foo???"));
|
||||||
|
}
|
Loading…
Reference in a new issue