pub(crate) mod package_wizard;
pub(crate) mod prompts;
pub(crate) mod render;
pub(crate) mod timestamp;
pub(crate) mod unpack;
use std::{
path::{Path, PathBuf},
str::FromStr,
};
use anyhow::{bail, Context as _, Result};
use once_cell::sync::Lazy;
use regex::Regex;
use wasmer_wasix::runners::MappedDirectory;
fn retrieve_alias_pathbuf(alias: &str, real_dir: &str) -> Result<MappedDirectory> {
let pb = PathBuf::from(&real_dir).canonicalize()?;
if let Ok(pb_metadata) = pb.metadata() {
if !pb_metadata.is_dir() {
bail!("\"{}\" exists, but it is not a directory", &real_dir);
}
} else {
bail!("Directory \"{}\" does not exist", &real_dir);
}
Ok(MappedDirectory {
guest: alias.to_string(),
host: pb,
})
}
pub fn parse_mapdir(entry: &str) -> Result<MappedDirectory> {
if let [alias, real_dir] = entry.split("::").collect::<Vec<&str>>()[..] {
retrieve_alias_pathbuf(alias, real_dir)
}
else if let [alias, real_dir] = entry.splitn(2, ':').collect::<Vec<&str>>()[..] {
retrieve_alias_pathbuf(alias, real_dir)
} else {
bail!(
"Directory mappings must consist of two paths separate by a `::` or `:`. Found {}",
&entry
)
}
}
pub fn parse_envvar(entry: &str) -> Result<(String, String)> {
let entry = entry.trim();
match entry.find('=') {
None => bail!(
"Environment variable must be of the form `<name>=<value>`; found `{}`",
&entry
),
Some(0) => bail!(
"Environment variable is not well formed, the `name` is missing in `<name>=<value>`; got `{}`",
&entry
),
Some(position) if position == entry.len() - 1 => bail!(
"Environment variable is not well formed, the `value` is missing in `<name>=<value>`; got `{}`",
&entry
),
Some(position) => Ok((entry[..position].into(), entry[position + 1..].into())),
}
}
pub(crate) const DEFAULT_PACKAGE_MANIFEST_FILE: &str = "wasmer.toml";
pub fn load_package_manifest(
path: &Path,
) -> Result<Option<(PathBuf, wasmer_config::package::Manifest)>, anyhow::Error> {
let file_path = if path.is_file() {
path.to_owned()
} else {
path.join(DEFAULT_PACKAGE_MANIFEST_FILE)
};
let contents = match std::fs::read_to_string(&file_path) {
Ok(c) => c,
Err(err) if err.kind() == std::io::ErrorKind::NotFound => return Ok(None),
Err(err) => {
return Err(err).with_context(|| {
format!(
"Could not read package manifest at '{}'",
file_path.display()
)
})
}
};
let manifest = wasmer_config::package::Manifest::parse(&contents).with_context(|| {
format!(
"Could not parse package config at: '{}' - full config: {}",
file_path.display(),
contents
)
})?;
Ok(Some((file_path, manifest)))
}
#[derive(Debug, Clone, PartialEq)]
pub(crate) struct Identifier {
pub name: String,
pub owner: Option<String>,
pub version: Option<String>,
}
impl Identifier {
pub fn new(name: impl Into<String>) -> Self {
Identifier {
name: name.into(),
owner: None,
version: None,
}
}
pub fn with_owner(self, owner: impl Into<String>) -> Self {
Identifier {
owner: Some(owner.into()),
..self
}
}
pub fn with_version(self, version: impl Into<String>) -> Self {
Identifier {
version: Some(version.into()),
..self
}
}
}
impl FromStr for Identifier {
type Err = anyhow::Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
const PATTERN: &str = r"^(?x)
(?:
(?P<owner>[a-zA-Z][\w\d_.-]*)
/
)?
(?P<name>[a-zA-Z][\w\d_.-]*)
(?:
@
(?P<version>[\w\d.]+)
)?
$
";
static RE: Lazy<Regex> = Lazy::new(|| Regex::new(PATTERN).unwrap());
let caps = RE.captures(s).context(
"Invalid package identifier, expected something like namespace/package@version",
)?;
let mut identifier = Identifier::new(&caps["name"]);
if let Some(owner) = caps.name("owner") {
identifier = identifier.with_owner(owner.as_str());
}
if let Some(version) = caps.name("version") {
identifier = identifier.with_version(version.as_str());
}
Ok(identifier)
}
}
pub(crate) fn merge_yaml_values(a: &serde_yaml::Value, b: &serde_yaml::Value) -> serde_yaml::Value {
use serde_yaml::Value as V;
match (a, b) {
(V::Mapping(a), V::Mapping(b)) => {
let mut m = a.clone();
for (k, v) in b.iter() {
let newval = if let Some(old) = a.get(k) {
merge_yaml_values(old, v)
} else {
v.clone()
};
m.insert(k.clone(), newval);
}
V::Mapping(m)
}
_ => b.clone(),
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_merge_yaml_values() {
use serde_yaml::Value;
let v1 = r#"
a: a
b:
b1: b1
c: c
"#;
let v2 = r#"
a: a1
b:
b2: b2
"#;
let v3 = r#"
a: a1
b:
b1: b1
b2: b2
c: c
"#;
let a: Value = serde_yaml::from_str(v1).unwrap();
let b: Value = serde_yaml::from_str(v2).unwrap();
let c: Value = serde_yaml::from_str(v3).unwrap();
let merged = merge_yaml_values(&a, &b);
assert_eq!(merged, c);
}
#[test]
fn parse_valid_identifiers() {
let inputs = [
("python", Identifier::new("python")),
(
"syrusakbary/python",
Identifier::new("python").with_owner("syrusakbary"),
),
(
"wasmer/wasmer.io",
Identifier::new("wasmer.io").with_owner("wasmer"),
),
(
"syrusakbary/python@1.2.3",
Identifier::new("python")
.with_owner("syrusakbary")
.with_version("1.2.3"),
),
(
"python@1.2.3",
Identifier::new("python").with_version("1.2.3"),
),
];
for (src, expected) in inputs {
let identifier = Identifier::from_str(src).expect(src);
assert_eq!(identifier, expected);
}
}
#[test]
fn invalid_package_identifiers() {
let inputs = ["", "$", "python/", "/python", "python@", "."];
for input in inputs {
let result = Identifier::from_str(input);
assert!(result.is_err(), "Got {result:?} from {input:?}");
}
}
#[test]
fn test_parse_envvar() {
assert_eq!(
parse_envvar("A").unwrap_err().to_string(),
"Environment variable must be of the form `<name>=<value>`; found `A`"
);
assert_eq!(
parse_envvar("=A").unwrap_err().to_string(),
"Environment variable is not well formed, the `name` is missing in `<name>=<value>`; got `=A`"
);
assert_eq!(
parse_envvar("A=").unwrap_err().to_string(),
"Environment variable is not well formed, the `value` is missing in `<name>=<value>`; got `A=`"
);
assert_eq!(parse_envvar("A=B").unwrap(), ("A".into(), "B".into()));
assert_eq!(parse_envvar(" A=B\t").unwrap(), ("A".into(), "B".into()));
assert_eq!(
parse_envvar("A=B=C=D").unwrap(),
("A".into(), "B=C=D".into())
);
}
}