diff options
author | Nadrieril | 2019-03-26 22:35:46 +0100 |
---|---|---|
committer | Nadrieril | 2019-03-26 22:35:46 +0100 |
commit | 23e12ffc4421414abbd089759dab9c50aefeac0c (patch) | |
tree | da1cfb1f2744fd85f1f3f6b25858db99311a315f /dhall | |
parent | fd293b7919d84faa2ac0df05ddd25c0386dc4c67 (diff) |
Derive DhallType for structs
Diffstat (limited to '')
-rw-r--r-- | dhall/src/dhall_type.rs | 77 | ||||
-rw-r--r-- | dhall/src/lib.rs | 2 | ||||
-rw-r--r-- | dhall/tests/dhall_type.rs | 37 | ||||
-rw-r--r-- | dhall_core/src/dhall_type.rs | 10 | ||||
-rw-r--r-- | dhall_core/src/lib.rs | 2 | ||||
-rw-r--r-- | dhall_generator/Cargo.toml | 1 | ||||
-rw-r--r-- | dhall_generator/src/dhall_expr.rs | 4 | ||||
-rw-r--r-- | dhall_generator/src/dhall_type.rs | 86 |
8 files changed, 189 insertions, 30 deletions
diff --git a/dhall/src/dhall_type.rs b/dhall/src/dhall_type.rs new file mode 100644 index 0000000..8abef32 --- /dev/null +++ b/dhall/src/dhall_type.rs @@ -0,0 +1,77 @@ +use dhall_core::*; +use dhall_generator::*; + +#[derive(Debug, Clone)] +pub enum DhallConversionError {} + +pub trait DhallType: Sized { + fn dhall_type() -> DhallExpr; + // fn as_dhall(&self) -> DhallExpr; + // fn from_dhall(e: DhallExpr) -> Result<Self, DhallConversionError>; +} + +impl DhallType for bool { + fn dhall_type() -> DhallExpr { + dhall_expr!(Bool) + } +} + +impl DhallType for Natural { + fn dhall_type() -> DhallExpr { + dhall_expr!(Natural) + } +} + +impl DhallType for Integer { + fn dhall_type() -> DhallExpr { + dhall_expr!(Integer) + } +} + +impl DhallType for String { + fn dhall_type() -> DhallExpr { + dhall_expr!(Text) + } +} + +impl<A: DhallType, B: DhallType> DhallType for (A, B) { + fn dhall_type() -> DhallExpr { + let ta = A::dhall_type(); + let tb = B::dhall_type(); + dhall_expr!({ _1: ta, _2: tb }) + } +} + +impl<T: DhallType> DhallType for Option<T> { + fn dhall_type() -> DhallExpr { + let t = T::dhall_type(); + dhall_expr!(Optional t) + } +} + +impl<T: DhallType> DhallType for Vec<T> { + fn dhall_type() -> DhallExpr { + let t = T::dhall_type(); + dhall_expr!(List t) + } +} + +impl<'a, T: DhallType> DhallType for &'a T { + fn dhall_type() -> DhallExpr { + T::dhall_type() + } +} + +impl<T> DhallType for std::marker::PhantomData<T> { + fn dhall_type() -> DhallExpr { + dhall_expr!({}) + } +} + +impl<T: DhallType, E: DhallType> DhallType for Result<T, E> { + fn dhall_type() -> DhallExpr { + let tt = T::dhall_type(); + let te = E::dhall_type(); + dhall_expr!(< Ok: tt | Err: te>) + } +} diff --git a/dhall/src/lib.rs b/dhall/src/lib.rs index d9f9edb..c0c1d6f 100644 --- a/dhall/src/lib.rs +++ b/dhall/src/lib.rs @@ -10,8 +10,10 @@ mod normalize; pub use crate::normalize::*; pub mod binary; +mod dhall_type; pub mod imports; pub mod typecheck; +pub use dhall_type::*; pub use crate::imports::*; diff --git a/dhall/tests/dhall_type.rs b/dhall/tests/dhall_type.rs index cbb71a4..0e27ad0 100644 --- a/dhall/tests/dhall_type.rs +++ b/dhall/tests/dhall_type.rs @@ -1,15 +1,32 @@ #![feature(proc_macro_hygiene)] -use dhall_core::*; +use dhall::*; use dhall_generator::*; -#[derive(DhallType)] -struct A { - _field1: bool, - // field2: Option<bool>, -} - #[test] -fn test_dhall_type_a() { - assert_eq!(A::dhall_type(), dhall_expr!(False)); - // assert_eq!(A::dhall_type(), dhall_expr!({ field1: Bool })); +fn test_dhall_type() { + assert_eq!(bool::dhall_type(), dhall_expr!(Bool)); + assert_eq!(String::dhall_type(), dhall_expr!(Text)); + assert_eq!( + <(bool, Option<String>)>::dhall_type(), + dhall_expr!({ _1: Bool, _2: Optional Text }) + ); + + #[derive(DhallType)] + #[allow(dead_code)] + struct A { + field1: bool, + field2: Option<bool>, + } + assert_eq!( + A::dhall_type(), + dhall_expr!({ field1: Bool, field2: Optional Bool }) + ); + + #[derive(DhallType)] + #[allow(dead_code)] + struct B<'a, T: 'a> { + field1: &'a T, + field2: Option<T>, + } + assert_eq!(<B<'static, bool>>::dhall_type(), A::dhall_type()); } diff --git a/dhall_core/src/dhall_type.rs b/dhall_core/src/dhall_type.rs deleted file mode 100644 index 3635e67..0000000 --- a/dhall_core/src/dhall_type.rs +++ /dev/null @@ -1,10 +0,0 @@ -use crate::*; - -#[derive(Debug, Clone)] -pub enum DhallConversionError {} - -pub trait DhallType: Sized { - fn dhall_type() -> DhallExpr; - // fn as_dhall(&self) -> DhallExpr; - // fn from_dhall(e: DhallExpr) -> Result<Self, DhallConversionError>; -} diff --git a/dhall_core/src/lib.rs b/dhall_core/src/lib.rs index b15c644..0874b09 100644 --- a/dhall_core/src/lib.rs +++ b/dhall_core/src/lib.rs @@ -18,6 +18,4 @@ mod printer; pub use crate::printer::*; mod parser; pub use crate::parser::*; -mod dhall_type; -pub use dhall_type::*; pub mod context; diff --git a/dhall_generator/Cargo.toml b/dhall_generator/Cargo.toml index ada6707..2a6ca67 100644 --- a/dhall_generator/Cargo.toml +++ b/dhall_generator/Cargo.toml @@ -12,4 +12,5 @@ doctest = false itertools = "0.8.0" quote = "0.6.11" proc-macro2 = "0.4.27" +syn = "0.15.29" dhall_core = { path = "../dhall_core" } diff --git a/dhall_generator/src/dhall_expr.rs b/dhall_generator/src/dhall_expr.rs index 1ee4d1a..bc9da22 100644 --- a/dhall_generator/src/dhall_expr.rs +++ b/dhall_generator/src/dhall_expr.rs @@ -89,6 +89,10 @@ fn dhall_to_tokenstream( let m = map_to_tokenstream(m, ctx); quote! { dhall_core::Expr::RecordLit(#m) } } + UnionType(m) => { + let m = map_to_tokenstream(m, ctx); + quote! { dhall_core::Expr::UnionType(#m) } + } e => unimplemented!("{:?}", e), } } diff --git a/dhall_generator/src/dhall_type.rs b/dhall_generator/src/dhall_type.rs index 7cfd945..b305ca0 100644 --- a/dhall_generator/src/dhall_type.rs +++ b/dhall_generator/src/dhall_type.rs @@ -1,15 +1,85 @@ extern crate proc_macro; // use dhall_core::*; -// use proc_macro2::TokenStream; -use quote::quote; +use proc_macro::TokenStream; +use quote::{quote, quote_spanned}; +use syn::spanned::Spanned; +use syn::{parse_macro_input, parse_quote, DeriveInput}; -pub fn derive_dhall_type(_input: proc_macro::TokenStream) -> proc_macro::TokenStream { - (quote!{ - impl DhallType for A { +pub fn derive_dhall_type(input: TokenStream) -> TokenStream { + let input = parse_macro_input!(input as DeriveInput); + + // List of types that must impl DhallType + let mut constraints = vec![]; + + let dhall_type = match &input.data { + syn::Data::Struct(data) => match &data.fields { + syn::Fields::Named(fields) => { + let fields = fields.named.iter() + .map(|f| { + let name = f.ident.as_ref().unwrap().to_string(); + let ty = &f.ty; + constraints.push(ty.clone()); + quote!( m.insert(dhall_core::Label::from(#name), <#ty as dhall::DhallType>::dhall_type()); ) + }); + quote! { dhall_core::rc(dhall_core::Expr::RecordType({ + use std::collections::BTreeMap; + let mut m = BTreeMap::new(); + #(#fields)* + m + })) } + } + _ => quote!(dhall_generator::dhall_expr!(Bool)), + }, + _ => quote!(dhall_generator::dhall_expr!(Bool)), + }; + + let mut generics = input.generics.clone(); + generics.make_where_clause(); + let (impl_generics, ty_generics, orig_where_clause) = + generics.split_for_impl(); + let orig_where_clause = orig_where_clause.unwrap(); + + // Hygienic errors + let assertions = constraints.iter().enumerate().map(|(i, ty)| { + // Ensure that ty: DhallType, with an appropriate span + let assert_name = + syn::Ident::new(&format!("_AssertDhallType{}", i), ty.span()); + let mut local_where_clause = orig_where_clause.clone(); + local_where_clause + .predicates + .push(parse_quote!(#ty: dhall::DhallType)); + let phantoms = generics.params.iter().map(|param| match param { + syn::GenericParam::Type(syn::TypeParam { ident, .. }) => { + quote!(#ident) + } + syn::GenericParam::Lifetime(syn::LifetimeDef { + lifetime, .. + }) => quote!(&#lifetime ()), + _ => unimplemented!(), + }); + quote_spanned! {ty.span()=> + struct #assert_name #impl_generics #local_where_clause { + _phantom: std::marker::PhantomData<(#(#phantoms),*)> + }; + } + }); + + // Ensure that all the fields have a DhallType impl + let mut where_clause = orig_where_clause.clone(); + for ty in constraints.iter() { + where_clause + .predicates + .push(parse_quote!(#ty: dhall::DhallType)); + } + + let ident = &input.ident; + let tokens = quote! { + impl #impl_generics dhall::DhallType for #ident #ty_generics #where_clause { fn dhall_type() -> dhall_core::DhallExpr { - dhall_core::rc(dhall_core::Expr::BoolLit(false)) + #(#assertions)* + #dhall_type } } - }).into() + }; + TokenStream::from(tokens) } - |