1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
use crate::StartDiagramToken;
use crate::{wr, ParseContainer};
use nom::branch::alt;
use nom::character::complete::{line_ending, space0};
use nom::combinator::eof;
use nom::sequence::tuple;
use nom::IResult;

/// `@start[a-z]*(id=MY_OWN_ID)` | `@start[a-z]* MY_OWN_ID` (e.g. `@startuml`)
#[derive(Clone, Debug)]
pub struct StartLine {
    token: StartDiagramToken,
}

impl StartLine {
    /// parse like `" @startuml\n"`, `" @startuml ID\n"`, `" @startuml(id=ID)\n"`
    pub fn parse(input: ParseContainer) -> IResult<ParseContainer, (Vec<ParseContainer>, Self)> {
        let (rest, parsed) = tuple((
            wr!(space0),
            StartDiagramToken::parse,
            wr!(space0),
            alt((wr!(eof), wr!(line_ending))),
        ))(input)?;

        let (token_raws, token_parsed) = parsed.1;

        let ret0 = [parsed.0]
            .into_iter()
            .chain(token_raws)
            .chain([parsed.2, parsed.3])
            .collect();

        let ret1 = Self {
            token: token_parsed,
        };

        Ok((rest, (ret0, ret1)))
    }

    pub fn diagram_kind(&self) -> &str {
        self.token.diagram_kind()
    }

    pub fn id(&self) -> Option<&str> {
        self.token.id()
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn test_parse_start_line() -> anyhow::Result<()> {
        let testdata = "@startuml\n";
        let (rest, (parsed, StartLine { token })) = StartLine::parse(testdata.into())?;
        assert_eq!(rest, "");
        assert_eq!(testdata, parsed.join(""));
        assert_eq!(token.diagram_kind(), "uml");
        assert_eq!(token.id(), None);

        let testdata = " \t@startuml\n";
        let (rest, (parsed, StartLine { token })) = StartLine::parse(testdata.into())?;
        assert_eq!(rest, "");
        assert_eq!(testdata, parsed.join(""));
        assert_eq!(token.diagram_kind(), "uml");
        assert_eq!(token.id(), None);

        let testdata = "@startuml \t\n";
        let (rest, (parsed, StartLine { token })) = StartLine::parse(testdata.into())?;
        assert_eq!(rest, "");
        assert_eq!(testdata, parsed.join(""));
        assert_eq!(token.diagram_kind(), "uml");
        assert_eq!(token.id(), None);

        let testdata = " \t@startuml \t\n";
        let (rest, (parsed, StartLine { token })) = StartLine::parse(testdata.into())?;
        assert_eq!(rest, "");
        assert_eq!(testdata, parsed.join(""));
        assert_eq!(token.diagram_kind(), "uml");
        assert_eq!(token.id(), None);

        let testdata = " \t@startuml id_foo\t\n";
        let (rest, (parsed, StartLine { token })) = StartLine::parse(testdata.into())?;
        assert_eq!(rest, "");
        assert_eq!(testdata, parsed.join(""));
        assert_eq!(token.diagram_kind(), "uml");
        assert_eq!(token.id(), Some("id_foo"));

        let testdata = " \t@startuml(id=id_bar)\t\n";
        let (rest, (parsed, token)) = StartLine::parse(testdata.into())?;
        assert_eq!(rest, "");
        assert_eq!(testdata, parsed.join(""));
        assert_eq!(token.diagram_kind(), "uml");
        assert_eq!(token.id(), Some("id_bar"));

        Ok(())
    }
}