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
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
//! Import other modules into the current namespace
//!
//! The import statement allowes to import modules from
//! the standard lib, local modules or thid-party modules
//!
//! ```py
//! import math
//! import lib.functools
//!
//! external import frosch.handler as fhandler
//! ```
use crate::parser::{
    literals::{parse_variable, sp},
    tokens::{dot, external, import, las},
    Res, Span,
};

use crate::literals::Variable;
use nom::{
    combinator::opt,
    error::context,
    multi::separated_list1,
    sequence::{preceded, tuple},
};

/// The import struct represents one import statement
#[derive(PartialEq, Debug, Clone)]
pub struct Import<'a> {
    /// If a module is a third-party module, therefore not
    /// in the local project or part of the standard lib
    /// it has to be marked as an external module.
    pub external: bool,

    /// The path defines where to look for the import.
    /// A path is represented as a list of Identifier names
    pub path: Vec<Variable<'a>>,

    /// For convenience can a import be aliased, which
    /// represented the imported module
    pub alias: Option<Variable<'a>>,
}

/// Parses the input inot a Import struct
pub fn parse_import(input: Span) -> Res<Import> {
    context(
        "Import",
        preceded(
            sp,
            tuple((
                opt(preceded(sp, external)),
                preceded(
                    preceded(sp, import),
                    separated_list1(preceded(sp, dot), preceded(sp, parse_variable)),
                ),
                opt(preceded(preceded(sp, las), preceded(sp, parse_variable))),
            )),
        ),
    )(input)
    .map(|(next_input, res)| {
        (
            next_input,
            Import {
                external: res.0.is_some(),
                path: res.1,
                alias: res.2,
            },
        )
    })
}

#[cfg(test)]
mod tests {

    use super::*;
    use crate::literals::Token;

    #[test]
    fn test_import() {
        let string = "import hello";
        let (_, res) = parse_import(Span::new(string)).unwrap();

        assert_eq!(
            res,
            Import {
                external: false,
                path: vec![Token::new("hello", Span::new("hello"))],
                alias: None
            }
        )
    }

    #[test]
    fn test_import_multi() {
        let string = "import hello.world as tuna";
        let (_, res) = parse_import(Span::new(string)).unwrap();

        assert_eq!(
            res,
            Import {
                external: false,
                path: vec![
                    Token::new("hello", Span::new("hello")),
                    Token::new("world", Span::new("world"))
                ],
                alias: Some(Token::new("tuna", Span::new("tuna")))
            }
        )
    }

    #[test]
    fn test_import_external() {
        let string = "external import hello";
        let (_, res) = parse_import(Span::new(string)).unwrap();

        assert_eq!(
            res,
            Import {
                external: true,
                path: vec![Token::new("hello", Span::new("hello"))],
                alias: None
            }
        )
    }

    #[test]
    fn test_import_alias() {
        let string = "import hello as h";
        let (_, res) = parse_import(Span::new(string)).unwrap();
        assert_eq!(
            res,
            Import {
                external: false,
                path: vec![Token::new("hello", Span::new("hello"))],
                alias: Some(Token::new("h", Span::new("h")))
            }
        )
    }
}