use crate::priv_prelude::*;
#[derive(Clone, Debug, Serialize)]
pub struct Annotated<T> {
pub attributes: Vec<AttributeDecl>,
pub value: T,
}
pub const STORAGE_ATTRIBUTE_NAME: &str = "storage";
pub const STORAGE_READ_ARG_NAME: &str = "read";
pub const STORAGE_WRITE_ARG_NAME: &str = "write";
pub const INLINE_ATTRIBUTE_NAME: &str = "inline";
pub const INLINE_NEVER_ARG_NAME: &str = "never";
pub const INLINE_ALWAYS_ARG_NAME: &str = "always";
pub const PAYABLE_ATTRIBUTE_NAME: &str = "payable";
pub const FALLBACK_ATTRIBUTE_NAME: &str = "fallback";
pub const DOC_COMMENT_ATTRIBUTE_NAME: &str = "doc-comment";
pub const TEST_ATTRIBUTE_NAME: &str = "test";
pub const TEST_SHOULD_REVERT_ARG_NAME: &str = "should_revert";
pub const ALLOW_ATTRIBUTE_NAME: &str = "allow";
pub const ALLOW_DEAD_CODE_ARG_NAME: &str = "dead_code";
pub const ALLOW_DEPRECATED_ARG_NAME: &str = "deprecated";
pub const CFG_ATTRIBUTE_NAME: &str = "cfg";
pub const CFG_TARGET_ARG_NAME: &str = "target";
pub const CFG_PROGRAM_TYPE_ARG_NAME: &str = "program_type";
pub const DEPRECATED_ATTRIBUTE_NAME: &str = "deprecated";
pub const DEPRECATED_NOTE_ARG_NAME: &str = "note";
pub const ERROR_TYPE_ATTRIBUTE_NAME: &str = "error_type";
pub const ERROR_ATTRIBUTE_NAME: &str = "error";
pub const ERROR_M_ARG_NAME: &str = "m";
pub const TRACE_ATTRIBUTE_NAME: &str = "trace";
pub const TRACE_NEVER_ARG_NAME: &str = "never";
pub const TRACE_ALWAYS_ARG_NAME: &str = "always";
pub const ABI_NAME_ATTRIBUTE_NAME: &str = "abi_name";
pub const ABI_NAME_NAME_ARG_NAME: &str = "name";
pub const EVENT_ATTRIBUTE_NAME: &str = "event";
pub const INDEXED_ATTRIBUTE_NAME: &str = "indexed";
pub const REQUIRE_ATTRIBUTE_NAME: &str = "require";
pub const REQUIRE_ARG_NAME_TRIVIALLY_ENCODABLE: &str = "trivially_encodable";
pub const REQUIRE_ARG_NAME_TRIVIALLY_DECODABLE: &str = "trivially_decodable";
pub const KNOWN_ATTRIBUTE_NAMES: &[&str] = &[
STORAGE_ATTRIBUTE_NAME,
DOC_COMMENT_ATTRIBUTE_NAME,
TEST_ATTRIBUTE_NAME,
INLINE_ATTRIBUTE_NAME,
PAYABLE_ATTRIBUTE_NAME,
ALLOW_ATTRIBUTE_NAME,
CFG_ATTRIBUTE_NAME,
DEPRECATED_ATTRIBUTE_NAME,
FALLBACK_ATTRIBUTE_NAME,
ABI_NAME_ATTRIBUTE_NAME,
EVENT_ATTRIBUTE_NAME,
INDEXED_ATTRIBUTE_NAME,
REQUIRE_ATTRIBUTE_NAME,
];
#[derive(Clone, Debug, Serialize)]
pub struct AttributeDecl {
pub hash_kind: AttributeHashKind,
pub attribute: SquareBrackets<Punctuated<Attribute, CommaToken>>,
}
impl AttributeDecl {
pub fn new_outer_doc_comment(span: Span, content_span: Span) -> Self {
Self::new_doc_comment(
span.clone(),
content_span,
AttributeHashKind::Outer(HashToken::new(span)),
)
}
pub fn new_inner_doc_comment(span: Span, content_span: Span) -> Self {
Self::new_doc_comment(
span.clone(),
content_span,
AttributeHashKind::Inner(HashBangToken::new(span)),
)
}
fn new_doc_comment(span: Span, content_span: Span, hash_kind: AttributeHashKind) -> Self {
let name = Ident::new_no_trim(content_span.clone());
AttributeDecl {
hash_kind,
attribute: SquareBrackets::new(
Punctuated::single(Attribute {
name: Ident::new_with_override(
DOC_COMMENT_ATTRIBUTE_NAME.to_string(),
span.clone(),
),
args: Some(Parens::new(
Punctuated::single(AttributeArg { name, value: None }),
content_span,
)),
}),
span,
),
}
}
pub fn is_doc_comment(&self) -> bool {
self.attribute.inner.value_separator_pairs.is_empty()
&& self
.attribute
.inner
.final_value_opt
.as_ref()
.is_some_and(|attr| attr.is_doc_comment())
}
pub fn is_inner(&self) -> bool {
matches!(self.hash_kind, AttributeHashKind::Inner(_))
}
}
impl Spanned for AttributeDecl {
fn span(&self) -> Span {
let hash_span = match &self.hash_kind {
AttributeHashKind::Inner(hash_bang_token) => hash_bang_token.span(),
AttributeHashKind::Outer(hash_token) => hash_token.span(),
};
Span::join(hash_span, &self.attribute.span())
}
}
#[derive(Clone, Debug, Serialize)]
pub enum AttributeHashKind {
Inner(HashBangToken),
Outer(HashToken),
}
impl Spanned for AttributeHashKind {
fn span(&self) -> Span {
match self {
AttributeHashKind::Inner(token) => token.span(),
AttributeHashKind::Outer(token) => token.span(),
}
}
}
#[derive(Clone, Debug, Serialize)]
pub struct AttributeArg {
pub name: Ident,
pub value: Option<Literal>,
}
impl Spanned for AttributeArg {
fn span(&self) -> Span {
if let Some(value) = &self.value {
Span::join(self.name.span(), &value.span())
} else {
self.name.span()
}
}
}
#[derive(Clone, Debug, Serialize)]
pub struct Attribute {
pub name: Ident,
pub args: Option<Parens<Punctuated<AttributeArg, CommaToken>>>,
}
impl Attribute {
pub fn is_doc_comment(&self) -> bool {
self.name.as_str() == DOC_COMMENT_ATTRIBUTE_NAME
}
pub fn is_cfg(&self) -> bool {
self.name.as_str() == CFG_ATTRIBUTE_NAME
}
}
impl Spanned for Attribute {
fn span(&self) -> Span {
self.args
.as_ref()
.map(|args| Span::join(self.name.span(), &args.span()))
.unwrap_or_else(|| self.name.span())
}
}