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
use std::collections::HashMap;

use gc::Gc;

use context::Context;
use symbol::Symbol;
use util::{as_sym_list, is_shl};
use value::Value;

/// An error occurring from a call to
/// [`parse_module_header`](fn.parse_module_header.html).
#[derive(Clone, Debug, Eq, Finalize, Hash, PartialEq, Trace)]
pub enum ModuleHeaderError {
    /// An `import` form was invalid.
    BadImportForm,

    /// The `module` form was invalid.
    BadModuleForm,

    /// No `module` form was present.
    NoModuleForm,
}

/// Parses out the `module` and `import` forms from the given values, returning the body.
pub fn parse_module_header<C: 'static + Context>(
    mut values: Vec<Gc<Value<C>>>,
) -> Result<(Symbol, Vec<Symbol>, HashMap<Symbol, Vec<Symbol>>, Vec<Gc<Value<C>>>), ModuleHeaderError> {
    fn parse_header_form<C: 'static + Context>(
        import: bool,
        value: Gc<Value<C>>,
    ) -> Result<(Symbol, Vec<Symbol>), ModuleHeaderError> {
        if let Some(mut sym_list) = as_sym_list(value) {
            if sym_list.len() >= 2 {
                let head = sym_list.remove(0);
                if import && head != Symbol::from("import") {
                    Err(ModuleHeaderError::BadImportForm)
                } else if !import && head != Symbol::from("module") {
                    Err(ModuleHeaderError::BadModuleForm)
                } else {
                    let name = sym_list.remove(0);
                    Ok((name, sym_list))
                }
            } else {
                Err(if import {
                    ModuleHeaderError::BadImportForm
                } else {
                    ModuleHeaderError::BadModuleForm
                })
            }
        } else {
            Err(if import {
                ModuleHeaderError::BadImportForm
            } else {
                ModuleHeaderError::BadModuleForm
            })
        }
    }

    if values.len() == 0 {
        return Err(ModuleHeaderError::NoModuleForm);
    }

    let import_sym = Symbol::from("import");
    let (name, exports) = parse_header_form(false, values.remove(0))?;
    let end = values
        .iter()
        .cloned()
        .position(|x| !is_shl(import_sym, x))
        .unwrap_or_else(|| values.len());

    let imports = values
        .drain(..end)
        .map(|v| parse_header_form(true, v))
        .collect::<Result<_, _>>()?;
    Ok((name, exports, imports, values))
}