extendr_macros/
extendr_conversion.rs

1use crate::{extendr_options::ExtendrOptions, wrappers};
2use proc_macro::TokenStream;
3use quote::quote;
4use syn::Attribute;
5use syn::Generics;
6use syn::Ident;
7use syn::Item;
8
9struct TypeFields {
10    ident: Ident,
11    generics: Generics,
12    attrs: Vec<Attribute>,
13}
14
15fn pull_fields(item: &Item) -> syn::Result<TypeFields> {
16    let fields = match item {
17        Item::Struct(str) => TypeFields {
18            ident: str.ident.clone(),
19            generics: str.generics.clone(),
20            attrs: str.attrs.clone(),
21        },
22        Item::Enum(str) => TypeFields {
23            ident: str.ident.clone(),
24            generics: str.generics.clone(),
25            attrs: str.attrs.clone(),
26        },
27        _ => {
28            return Err(syn::Error::new_spanned(
29                item,
30                "#[extendr] conversions can only be built for `struct` and `enum`",
31            ))
32        }
33    };
34    Ok(fields)
35}
36
37pub(crate) fn extendr_type_conversion(item: Item, opts: &ExtendrOptions) -> TokenStream {
38    match do_extendr_type_conversion(item, opts) {
39        Ok(result) => result,
40        Err(e) => e.to_compile_error().into(),
41    }
42}
43
44fn do_extendr_type_conversion(item: Item, _opts: &ExtendrOptions) -> syn::Result<TokenStream> {
45    let TypeFields {
46        ident: self_ty,
47        generics,
48        attrs,
49    } = pull_fields(&item)?;
50    if generics.const_params().count() != 0 {
51        return Err(syn::Error::new_spanned(
52            item,
53            "const params not allowed in #[extendr] impl",
54        ));
55    }
56    let mut self_ty_name = self_ty.to_string();
57    for gen in generics.type_params() {
58        self_ty_name.push('_');
59        self_ty_name.push_str(gen.ident.to_string().as_str());
60    }
61    // residual code syntax from extendr_impl.rs
62    // let mut _prefix = format!("{}__", self_ty_name);
63
64    // To do: Should documenting the struct be moved to R?
65    // At the moment, only documentattion above the impl
66    // block makes it to R.
67    let _doc_string = wrappers::get_doc_string(&attrs);
68
69    let conversion_impls = quote! {
70        // Output conversion function for this type.
71
72        impl TryFrom<Robj> for &#self_ty {
73            type Error = extendr_api::Error;
74
75            fn try_from(robj: Robj) -> extendr_api::Result<Self> {
76                Self::try_from(&robj)
77            }
78        }
79
80        impl TryFrom<Robj> for &mut #self_ty {
81            type Error = extendr_api::Error;
82
83            fn try_from(mut robj: Robj) -> extendr_api::Result<Self> {
84                Self::try_from(&mut robj)
85            }
86        }
87
88        // Output conversion function for this type.
89        impl TryFrom<&Robj> for &#self_ty {
90            type Error = extendr_api::Error;
91            fn try_from(robj: &Robj) -> extendr_api::Result<Self> {
92                use extendr_api::ExternalPtr;
93                unsafe {
94                    let external_ptr: &ExternalPtr<#self_ty> = robj.try_into()?;
95                    external_ptr.try_addr()
96                }
97            }
98        }
99
100        // Input conversion function for a mutable reference to this type.
101        impl TryFrom<&mut Robj> for &mut #self_ty {
102            type Error = extendr_api::Error;
103            fn try_from(robj: &mut Robj) -> extendr_api::Result<Self> {
104                use extendr_api::ExternalPtr;
105                unsafe {
106                    let external_ptr: &mut ExternalPtr<#self_ty> = robj.try_into()?;
107                    external_ptr.try_addr_mut()
108                }
109            }
110        }
111    };
112
113    let output = TokenStream::from(quote! {
114        #item
115
116        #conversion_impls
117
118        // Output conversion function for this type.
119        impl From<#self_ty> for Robj {
120            fn from(value: #self_ty) -> Self {
121                use extendr_api::ExternalPtr;
122                unsafe {
123                    let mut res: ExternalPtr<#self_ty> = ExternalPtr::new(value);
124                    res.set_attrib(class_symbol(), #self_ty_name).unwrap();
125                    res.into()
126                }
127            }
128        }
129
130    });
131    Ok(output)
132}