use crate::{module::ModuleDef, Ctx, Error, Module, Result};
use std::{collections::HashMap, fmt::Debug};
use super::Loader;
type LoadFn = for<'js> fn(Ctx<'js>, Vec<u8>) -> Result<Module<'js>>;
#[derive(Debug, Default)]
pub struct ModuleLoader {
modules: HashMap<String, LoadFn>,
}
impl ModuleLoader {
fn load_func<'js, D: ModuleDef>(ctx: Ctx<'js>, name: Vec<u8>) -> Result<Module<'js>> {
Module::declare_def::<D, _>(ctx, name)
}
pub fn add_module<N: Into<String>, M: ModuleDef>(&mut self, name: N, _module: M) -> &mut Self {
self.modules.insert(name.into(), Self::load_func::<M>);
self
}
#[must_use]
pub fn with_module<N: Into<String>, M: ModuleDef>(mut self, name: N, module: M) -> Self {
self.add_module(name, module);
self
}
}
impl Loader for ModuleLoader {
fn load<'js>(&mut self, ctx: &Ctx<'js>, path: &str) -> Result<Module<'js>> {
let load = self
.modules
.remove(path)
.ok_or_else(|| Error::new_loading(path))?;
(load)(ctx.clone(), Vec::from(path))
}
}