use marine_wasm_backend_traits::WasmBackend;
use marine_core::generic::HostImportDescriptor;
use std::collections::HashMap;
use std::collections::HashSet;
use std::path::Component;
use std::path::Path;
use std::path::PathBuf;
#[derive(Clone, Default, Debug)]
pub struct ConfigContext {
pub base_path: Option<PathBuf>,
}
pub struct WithContext<'c, T> {
pub context: &'c ConfigContext,
pub data: T,
}
impl ConfigContext {
pub fn wrapped<T>(&self, data: T) -> WithContext<'_, T> {
WithContext {
context: self,
data,
}
}
}
#[derive(Default)]
pub struct ModuleDescriptor<WB: WasmBackend> {
pub load_from: Option<PathBuf>,
pub file_name: String,
pub import_name: String,
pub config: MarineModuleConfig<WB>,
}
impl<WB: WasmBackend> ModuleDescriptor<WB> {
pub fn get_path(&self, modules_dir: &Option<PathBuf>) -> Result<PathBuf, MarineError> {
match &self.load_from {
None => match modules_dir {
Some(dir) => Ok(dir.join(Path::new(&self.file_name))),
None => Err(MarineError::InvalidConfig(format!(
r#""modules_dir" field is not defined, but it is required to load module "{}""#,
self.import_name
))),
},
Some(path) => {
if path.is_file() {
Ok(path.clone())
} else {
Ok(path.join(Path::new(&self.file_name)))
}
}
}
}
}
#[derive(Default)]
pub struct MarineConfig<WB: WasmBackend> {
pub modules_dir: Option<PathBuf>,
pub modules_config: Vec<ModuleDescriptor<WB>>,
pub default_modules_config: Option<MarineModuleConfig<WB>>,
}
#[derive(Default)]
pub struct MarineModuleConfig<WB: WasmBackend> {
pub mem_pages_count: Option<u32>,
pub max_heap_size: Option<u64>,
pub logger_enabled: bool,
pub host_imports: HashMap<String, HostImportDescriptor<WB>>,
pub wasi: Option<MarineWASIConfig>,
pub logging_mask: i32,
}
impl<WB: WasmBackend> MarineModuleConfig<WB> {
pub fn extend_wasi_envs(&mut self, new_envs: HashMap<String, String>) {
match &mut self.wasi {
Some(MarineWASIConfig { envs, .. }) => envs.extend(new_envs),
w @ None => {
*w = Some(MarineWASIConfig {
envs: new_envs,
preopened_files: HashSet::new(),
mapped_dirs: HashMap::new(),
})
}
};
}
pub fn extend_wasi_files(
&mut self,
new_preopened_files: HashSet<PathBuf>,
new_mapped_dirs: HashMap<String, PathBuf>,
) {
match &mut self.wasi {
Some(MarineWASIConfig {
preopened_files,
mapped_dirs,
..
}) => {
preopened_files.extend(new_preopened_files);
mapped_dirs.extend(new_mapped_dirs);
}
w @ None => {
*w = Some(MarineWASIConfig {
envs: HashMap::new(),
preopened_files: new_preopened_files,
mapped_dirs: new_mapped_dirs,
})
}
};
}
pub fn root_wasi_files_at(&mut self, root: &Path) {
match &mut self.wasi {
Some(MarineWASIConfig {
preopened_files,
mapped_dirs,
..
}) => {
mapped_dirs.values_mut().for_each(|path| {
*path = root.join(&path);
});
let mapped_preopens = preopened_files.iter().map(|path| {
let new_path = root.join(path);
(path.to_string_lossy().into(), new_path)
});
mapped_dirs.extend(mapped_preopens);
preopened_files.clear();
}
None => {}
}
}
}
#[derive(Debug, Clone, Default)]
pub struct MarineWASIConfig {
pub envs: HashMap<String, String>,
pub preopened_files: HashSet<PathBuf>,
pub mapped_dirs: HashMap<String, PathBuf>,
}
use super::TomlMarineConfig;
use super::TomlMarineModuleConfig;
use super::TomlWASIConfig;
use super::TomlMarineNamedModuleConfig;
use crate::MarineError;
use crate::MarineResult;
use crate::config::as_relative_to_base;
use itertools::Itertools;
use std::convert::TryFrom;
use std::convert::TryInto;
impl<WB: WasmBackend> TryFrom<TomlMarineConfig> for MarineConfig<WB> {
type Error = MarineError;
fn try_from(toml_config: TomlMarineConfig) -> Result<Self, Self::Error> {
let base_path = toml_config.base_path;
let context = ConfigContext {
base_path: Some(base_path),
};
let modules_dir = toml_config
.modules_dir
.map(|dir| as_relative_to_base(context.base_path.as_deref(), &dir))
.transpose()?;
let default_modules_config = toml_config
.default
.map(|m| context.wrapped(m).try_into())
.transpose()?;
let modules_config = toml_config
.module
.into_iter()
.map(|toml_module| ModuleDescriptor::try_from(context.wrapped(toml_module)))
.collect::<MarineResult<Vec<_>>>()?;
Ok(MarineConfig {
modules_dir,
modules_config,
default_modules_config,
})
}
}
impl<'c, WB: WasmBackend> TryFrom<WithContext<'c, TomlMarineNamedModuleConfig>>
for ModuleDescriptor<WB>
{
type Error = MarineError;
fn try_from(config: WithContext<'c, TomlMarineNamedModuleConfig>) -> Result<Self, Self::Error> {
let WithContext {
context,
data: config,
} = config;
let file_name = config.file_name.unwrap_or(format!("{}.wasm", config.name));
let load_from = config
.load_from
.map(|path| as_relative_to_base(context.base_path.as_deref(), &path))
.transpose()?;
Ok(ModuleDescriptor {
load_from,
file_name,
import_name: config.name,
config: context.wrapped(config.config).try_into()?,
})
}
}
impl<'c, WB: WasmBackend> TryFrom<WithContext<'c, TomlMarineModuleConfig>>
for MarineModuleConfig<WB>
{
type Error = MarineError;
fn try_from(toml_config: WithContext<'c, TomlMarineModuleConfig>) -> Result<Self, Self::Error> {
let WithContext {
context,
data: toml_config,
} = toml_config;
let mounted_binaries = toml_config.mounted_binaries.unwrap_or_default();
let mounted_binaries = mounted_binaries
.into_iter()
.map(|(import_func_name, host_cmd)| {
let host_cmd = host_cmd.try_into::<PathBuf>()?;
Ok((import_func_name, host_cmd))
})
.collect::<Result<Vec<_>, Self::Error>>()?;
let max_heap_size = toml_config.max_heap_size.map(|v| v.as_u64());
let mut host_cli_imports = HashMap::new();
for (import_name, host_cmd) in mounted_binaries {
let host_cmd = as_relative_to_base(context.base_path.as_deref(), &host_cmd)?;
host_cli_imports.insert(
import_name,
crate::host_imports::create_mounted_binary_import(host_cmd),
);
}
let wasi = toml_config.wasi.map(|w| w.try_into()).transpose()?;
Ok(MarineModuleConfig {
mem_pages_count: toml_config.mem_pages_count,
max_heap_size,
logger_enabled: toml_config.logger_enabled.unwrap_or(true),
host_imports: host_cli_imports,
wasi,
logging_mask: toml_config.logging_mask.unwrap_or(i32::max_value()),
})
}
}
impl TryFrom<TomlWASIConfig> for MarineWASIConfig {
type Error = MarineError;
fn try_from(toml_config: TomlWASIConfig) -> Result<Self, Self::Error> {
let to_string = |elem: (String, toml::Value)| -> Result<(String, String), Self::Error> {
let to = elem
.1
.try_into::<String>()
.map_err(MarineError::ParseConfigError)?;
Ok((elem.0, to))
};
let check_path = |path: PathBuf| -> Result<PathBuf, Self::Error> {
if path.is_absolute() {
return Err(MarineError::InvalidConfig(format!(
"Absolute paths are not supported in WASI section: {}",
path.display()
)));
}
if path.components().contains(&Component::ParentDir) {
return Err(MarineError::InvalidConfig(format!(
"Paths containing \"..\" are not supported in WASI section: {}",
path.display()
)));
}
Ok(path)
};
let to_path = |elem: (String, toml::Value)| -> Result<(String, PathBuf), Self::Error> {
let to = elem
.1
.try_into::<PathBuf>()
.map_err(MarineError::ParseConfigError)?;
let to = check_path(to)?;
Ok((elem.0, to))
};
let envs = toml_config.envs.unwrap_or_default();
let envs = envs
.into_iter()
.map(to_string)
.collect::<Result<HashMap<_, _>, _>>()?;
let preopened_files = toml_config.preopened_files.unwrap_or_default();
let preopened_files = preopened_files
.into_iter()
.map(check_path)
.collect::<Result<HashSet<_>, _>>()?;
let mapped_dirs = toml_config.mapped_dirs.unwrap_or_default();
let mapped_dirs = mapped_dirs
.into_iter()
.map(to_path)
.collect::<Result<HashMap<_, _>, _>>()?;
Ok(MarineWASIConfig {
envs,
preopened_files,
mapped_dirs,
})
}
}