rune/ast/
item_mod.rs
1use crate::ast::prelude::*;
2
3#[test]
4#[cfg(not(miri))]
5fn ast_parse() {
6 rt::<ast::ItemMod>("mod ruins {}");
7
8 let item = rt::<ast::ItemMod>("#[cfg(test)] mod tests {}");
9 assert_eq!(item.attributes.len(), 1);
10
11 let item = rt::<ast::ItemMod>("mod whiskey_bravo { #![allow(dead_code)] fn x() {} }");
12 assert_eq!(item.attributes.len(), 0);
13 assert!(matches!(item.body, ast::ItemModBody::InlineBody(..)));
14}
15
16#[derive(Debug, TryClone, PartialEq, Eq, Parse, ToTokens, Spanned)]
18#[rune(parse = "meta_only")]
19#[non_exhaustive]
20pub struct ItemMod {
21 #[rune(iter, meta)]
23 pub attributes: Vec<ast::Attribute>,
24 #[rune(option, meta)]
26 pub visibility: ast::Visibility,
27 pub mod_token: T![mod],
29 pub name: ast::Ident,
31 pub body: ItemModBody,
33 #[rune(skip)]
35 pub(crate) id: ItemId,
36}
37
38impl ItemMod {
39 pub(crate) fn name_span(&self) -> Span {
41 if let Some(span) = self.visibility.option_span() {
42 span.join(self.name.span())
43 } else {
44 self.mod_token.span().join(self.name.span())
45 }
46 }
47}
48
49item_parse!(Mod, ItemMod, "mod item");
50
51#[derive(Debug, TryClone, PartialEq, Eq, ToTokens, Spanned)]
53#[non_exhaustive]
54pub enum ItemModBody {
55 EmptyBody(T![;]),
57 InlineBody(ItemInlineBody),
59}
60
61impl Parse for ItemModBody {
62 fn parse(p: &mut Parser<'_>) -> Result<Self> {
63 Ok(match p.nth(0)? {
64 K!['{'] => Self::InlineBody(p.parse()?),
65 _ => Self::EmptyBody(p.parse()?),
66 })
67 }
68}
69
70#[derive(Debug, TryClone, PartialEq, Eq, ToTokens, Parse, Spanned)]
72#[non_exhaustive]
73pub struct ItemInlineBody {
74 pub open: T!['{'],
76 #[rune(option)]
78 pub file: Box<ast::File>,
79 pub close: T!['}'],
81}
82
83impl Peek for ItemInlineBody {
84 fn peek(p: &mut Peeker<'_>) -> bool {
85 <T!['{']>::peek(p)
86 }
87}