1use super::Selector;
16use crate::{
17 ast,
18 ast::MetaNameValue,
19 error::ExtError as _,
20 format_err,
21};
22use proc_macro2::Span;
23use std::collections::HashMap;
24use syn::spanned::Spanned;
25
26pub fn ensure_pub_visibility(
33 name: &str,
34 parent_span: Span,
35 vis: &syn::Visibility,
36) -> Result<(), syn::Error> {
37 let bad_visibility = match vis {
38 syn::Visibility::Inherited => Some(parent_span),
39 syn::Visibility::Restricted(vis_restricted) => Some(vis_restricted.span()),
40 syn::Visibility::Public(_) => None,
41 };
42 if let Some(bad_visibility) = bad_visibility {
43 return Err(format_err!(
44 bad_visibility,
45 "non `pub` ink! {} are not supported",
46 name
47 ))
48 }
49 Ok(())
50}
51
52pub fn local_message_id(ident: &syn::Ident) -> u32 {
59 let input = ident.to_string().into_bytes();
60 let selector = Selector::compute(&input);
61 selector.into_be_u32()
62}
63
64#[derive(Debug, PartialEq, Eq)]
67pub struct WhitelistedAttributes(pub HashMap<String, ()>);
68
69impl Default for WhitelistedAttributes {
70 fn default() -> Self {
71 Self(HashMap::from([
72 ("cfg".to_string(), ()),
74 ("cfg_attr".to_string(), ()),
75 ("allow".to_string(), ()),
77 ("warn".to_string(), ()),
78 ("deny".to_string(), ()),
79 ("forbid".to_string(), ()),
80 ("deprecated".to_string(), ()),
81 ("must_use".to_string(), ()),
82 ("doc".to_string(), ()),
84 ("rustfmt".to_string(), ()),
86 ]))
87 }
88}
89
90impl WhitelistedAttributes {
91 pub fn parse_arg_value(&mut self, arg: &MetaNameValue) -> Result<(), syn::Error> {
95 if let ast::MetaValue::Lit(syn::Lit::Str(attributes)) = &arg.value {
96 attributes.value().split(',').for_each(|attribute| {
97 self.0.insert(attribute.trim().to_string(), ());
98 });
99 Ok(())
100 } else {
101 Err(format_err_spanned!(
102 arg,
103 "expected a string with attributes separated by `,`",
104 ))
105 }
106 }
107
108 pub fn filter_attr(&self, attrs: Vec<syn::Attribute>) -> Vec<syn::Attribute> {
111 attrs
112 .into_iter()
113 .filter(|attr| {
114 if let Some(ident) = attr.path().get_ident() {
115 self.0.contains_key(&ident.to_string())
116 } else {
117 false
118 }
119 })
120 .collect()
121 }
122}
123
124pub fn duplicate_config_err<F, S>(
126 first: F,
127 second: S,
128 name: &str,
129 ink_attr: &str,
130) -> syn::Error
131where
132 F: Spanned,
133 S: Spanned,
134{
135 format_err!(
136 second.span(),
137 "encountered duplicate ink! {} `{}` configuration argument",
138 ink_attr,
139 name,
140 )
141 .into_combine(format_err!(
142 first.span(),
143 "first `{}` configuration argument here",
144 name
145 ))
146}
147
148pub fn find_storage_key_salt(input: &syn::DeriveInput) -> Option<syn::TypeParam> {
152 input.generics.params.iter().find_map(|param| {
153 if let syn::GenericParam::Type(type_param) = param {
154 if let Some(syn::TypeParamBound::Trait(trait_bound)) =
155 type_param.bounds.first()
156 {
157 let segments = &trait_bound.path.segments;
158 if let Some(last) = segments.last() {
159 if last.ident == "StorageKey" {
160 return Some(type_param.clone())
161 }
162 }
163 }
164 }
165 None
166 })
167}
168
169pub fn extract_cfg_attributes(
171 attrs: &[syn::Attribute],
172 span: Span,
173) -> Vec<proc_macro2::TokenStream> {
174 attrs
175 .iter()
176 .filter(|a| a.path().is_ident(super::CFG_IDENT))
177 .map(|a| quote::quote_spanned!(span=> #a ))
178 .collect()
179}
180
181pub fn extract_cfg_syn_attributes(attrs: &[syn::Attribute]) -> Vec<syn::Attribute> {
183 attrs
184 .iter()
185 .filter(|a| a.path().is_ident(super::CFG_IDENT))
186 .cloned()
187 .collect()
188}