ink_ir/ir/storage_item/
mod.rs1mod config;
16
17use crate::utils::find_storage_key_salt;
18use config::StorageItemConfig;
19use proc_macro2::TokenStream as TokenStream2;
20use quote::{
21 quote,
22 ToTokens,
23};
24use std::collections::HashSet;
25
26pub struct StorageItem {
28 ast: syn::DeriveInput,
29 config: StorageItemConfig,
30}
31
32impl StorageItem {
33 pub fn new(config: TokenStream2, item: TokenStream2) -> Result<Self, syn::Error> {
35 let ast = syn::parse2::<syn::DeriveInput>(item)?;
36 let parsed_config = syn::parse2::<crate::ast::AttributeArgs>(config)?;
37 let config = StorageItemConfig::try_from(parsed_config)?;
38
39 for attr in &ast.attrs {
40 if attr
41 .path()
42 .to_token_stream()
43 .to_string()
44 .contains("storage_item")
45 {
46 return Err(format_err_spanned!(
47 attr,
48 "only one `ink::storage_item` is allowed",
49 ))
50 }
51 }
52
53 Ok(Self { ast, config })
54 }
55
56 pub fn ast(&self) -> &syn::DeriveInput {
58 &self.ast
59 }
60
61 pub fn all_used_types(&self) -> Vec<syn::Type> {
63 let res: Vec<_> = match self.data().clone() {
64 syn::Data::Struct(st) => {
65 st.fields.iter().map(|field| field.ty.clone()).collect()
66 }
67 syn::Data::Enum(en) => {
68 en.variants
69 .iter()
70 .flat_map(|variant| variant.fields.iter())
71 .map(|field| field.ty.clone())
72 .collect()
73 }
74 syn::Data::Union(un) => {
75 un.fields
76 .named
77 .iter()
78 .map(|field| field.ty.clone())
79 .collect()
80 }
81 };
82 let mut set = HashSet::new();
83 res.into_iter()
84 .filter(|ty| {
85 if !set.contains(ty) {
86 set.insert(ty.clone());
87 true
88 } else {
89 false
90 }
91 })
92 .collect()
93 }
94
95 pub fn config(&self) -> &StorageItemConfig {
97 &self.config
98 }
99
100 pub fn vis(&self) -> &syn::Visibility {
102 &self.ast.vis
103 }
104
105 pub fn attrs(&self) -> &[syn::Attribute] {
107 &self.ast.attrs
108 }
109
110 pub fn ident(&self) -> &syn::Ident {
112 &self.ast.ident
113 }
114
115 pub fn generics(&self) -> TokenStream2 {
117 let types = self.ast.generics.clone();
118 let (_, _, where_closure) = self.ast.generics.split_for_impl();
120 quote! {
121 #types #where_closure
122 }
123 }
124
125 pub fn data(&self) -> &syn::Data {
127 &self.ast.data
128 }
129
130 pub fn salt(&self) -> TokenStream2 {
132 if let Some(param) = find_storage_key_salt(&self.ast) {
133 param.ident.to_token_stream()
134 } else {
135 quote! { () }
136 }
137 }
138}