use ink_analyzer_macro::{FromInkAttribute, FromSyntax};
use ra_ap_syntax::ast;
use crate::traits::{FromInkAttribute, FromSyntax, IsInkFn};
use crate::tree::utils;
use crate::{InkArg, InkArgKind, InkAttrData, InkAttribute};
#[derive(Debug, Clone, PartialEq, Eq, FromInkAttribute, FromSyntax)]
pub struct Extension {
#[arg_kind(Extension)]
ink_attr: InkAttrData<ast::Fn>,
}
impl IsInkFn for Extension {
fn fn_item(&self) -> Option<&ast::Fn> {
self.ink_attr.parent_ast()
}
}
impl Extension {
pub fn id(&self) -> Option<u32> {
self.extension_arg()?.value()?.as_u32()
}
pub fn extension_arg(&self) -> Option<InkArg> {
utils::ink_arg_by_kind(self.syntax(), InkArgKind::Extension)
}
pub fn handle_status_arg(&self) -> Option<InkArg> {
utils::ink_arg_by_kind(self.syntax(), InkArgKind::HandleStatus)
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::test_utils::*;
use test_utils::quote_as_str;
#[test]
fn cast_works() {
for (code, has_handle_status) in [
(
quote_as_str! {
#[ink(extension=1)]
fn my_extension();
},
false,
),
(
quote_as_str! {
#[ink(extension=0x1)]
fn my_extension();
},
false,
),
(
quote_as_str! {
#[ink(extension=1, handle_status=false)]
fn my_extension();
},
true,
),
(
quote_as_str! {
#[ink(extension=1)]
#[ink(handle_status=true)]
fn my_extension();
},
true,
),
] {
let ink_attr = parse_first_ink_attribute(code);
let extension = Extension::cast(ink_attr).unwrap();
assert!(extension.extension_arg().is_some());
assert_eq!(extension.handle_status_arg().is_some(), has_handle_status);
assert!(extension.fn_item().is_some());
}
}
}