ink_ir/ir/item/
storage.rs1use crate::{
16 ir,
17 ir::utils,
18};
19use proc_macro2::Ident;
20use syn::spanned::Spanned as _;
21
22#[derive(Debug, PartialEq, Eq)]
43pub struct Storage {
44 ast: syn::ItemStruct,
46}
47
48impl quote::ToTokens for Storage {
49 fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) {
52 self.ast.to_tokens(tokens)
53 }
54}
55
56impl Storage {
57 pub(super) fn is_ink_storage(
64 item_struct: &syn::ItemStruct,
65 ) -> Result<bool, syn::Error> {
66 if !ir::contains_ink_attributes(&item_struct.attrs) {
67 return Ok(false)
68 }
69 let attr = ir::first_ink_attribute(&item_struct.attrs)?
73 .expect("missing expected ink! attribute for struct");
74 Ok(matches!(attr.first().kind(), ir::AttributeArg::Storage))
75 }
76}
77
78impl TryFrom<syn::ItemStruct> for Storage {
79 type Error = syn::Error;
80
81 fn try_from(item_struct: syn::ItemStruct) -> Result<Self, Self::Error> {
82 let struct_span = item_struct.span();
83 let (_ink_attrs, other_attrs) = ir::sanitize_attributes(
84 struct_span,
85 item_struct.attrs,
86 &ir::AttributeArgKind::Storage,
87 |arg| {
88 match arg.kind() {
89 ir::AttributeArg::Storage => Ok(()),
90 _ => Err(None),
91 }
92 },
93 )?;
94 utils::ensure_pub_visibility("storage structs", struct_span, &item_struct.vis)?;
95 Ok(Self {
96 ast: syn::ItemStruct {
97 attrs: other_attrs,
98 ..item_struct
99 },
100 })
101 }
102}
103
104impl Storage {
105 pub fn attrs(&self) -> &[syn::Attribute] {
107 &self.ast.attrs
108 }
109
110 pub fn ident(&self) -> &Ident {
112 &self.ast.ident
113 }
114
115 pub fn generics(&self) -> &syn::Generics {
117 &self.ast.generics
118 }
119
120 pub fn fields(&self) -> syn::punctuated::Iter<syn::Field> {
122 self.ast.fields.iter()
123 }
124}
125
126#[cfg(test)]
127mod tests {
128 use super::*;
129
130 #[test]
131 fn simple_try_from_works() {
132 let item_struct: syn::ItemStruct = syn::parse_quote! {
133 #[ink(storage)]
134 pub struct MyStorage {
135 field_1: i32,
136 field_2: bool,
137 }
138 };
139 assert!(Storage::try_from(item_struct).is_ok())
140 }
141
142 fn assert_try_from_fails(item_struct: syn::ItemStruct, expected: &str) {
143 assert_eq!(
144 Storage::try_from(item_struct).map_err(|err| err.to_string()),
145 Err(expected.to_string())
146 )
147 }
148
149 #[test]
150 fn conflicting_attributes_fails() {
151 assert_try_from_fails(
152 syn::parse_quote! {
153 #[ink(storage)]
154 #[ink(event)]
155 pub struct MyStorage {
156 field_1: i32,
157 field_2: bool,
158 }
159 },
160 "encountered conflicting ink! attribute argument",
161 )
162 }
163
164 #[test]
165 fn duplicate_attributes_fails() {
166 assert_try_from_fails(
167 syn::parse_quote! {
168 #[ink(storage)]
169 #[ink(storage)]
170 pub struct MyStorage {
171 field_1: i32,
172 field_2: bool,
173 }
174 },
175 "encountered duplicate ink! attribute",
176 )
177 }
178
179 #[test]
180 fn wrong_first_attribute_fails() {
181 assert_try_from_fails(
182 syn::parse_quote! {
183 #[ink(event)]
184 #[ink(storage)]
185 pub struct MyStorage {
186 field_1: i32,
187 field_2: bool,
188 }
189 },
190 "unexpected first ink! attribute argument",
191 )
192 }
193
194 #[test]
195 fn missing_storage_attribute_fails() {
196 assert_try_from_fails(
197 syn::parse_quote! {
198 pub struct MyStorage {
199 field_1: i32,
200 field_2: bool,
201 }
202 },
203 "encountered unexpected empty expanded ink! attribute arguments",
204 )
205 }
206
207 #[test]
208 fn non_pub_storage_struct() {
209 assert_try_from_fails(
210 syn::parse_quote! {
211 #[ink(storage)]
212 struct PrivateStorage {
213 field_1: i32,
214 field_2: bool,
215 }
216 },
217 "non `pub` ink! storage structs are not supported",
218 )
219 }
220}