wasm-wave 0.249.0

WebAssembly Value Encoding
Documentation
#![cfg(feature = "wit")]

use anyhow::{Context, Result};
use std::{collections::HashMap, fmt::Write, fs, path::Path, sync::OnceLock};

use wasm_wave::{
    parser::ParserError,
    untyped::UntypedFuncCall,
    value::{FuncType, Value, resolve_wit_func_type},
    wasm::{DisplayValue, WasmFunc},
};
use wit_parser::Resolve;

fn test(path: &Path) -> Result<String> {
    let filename = path.file_name().unwrap().to_string_lossy();
    let inputs = std::fs::read_to_string(path)?;
    let mut output = String::new();
    let out = &mut output;
    let inputs = inputs.trim_end().trim_end_matches(';');
    for mut input in inputs.split(";\n") {
        // Copy leading comments into the output
        while input.starts_with("//") {
            let Some((comment, remainder)) = input.split_once('\n') else {
                break;
            };
            input = remainder;
            writeln!(out, "{comment}")?;
            continue;
        }

        fn parse_func_call(
            input: &str,
        ) -> Result<(String, &'static FuncType, Vec<Value>), ParserError> {
            let untyped_call = UntypedFuncCall::parse(input)?;
            let func_name = untyped_call.name().to_string();
            let func_type = get_func_type(&func_name).unwrap_or_else(|| {
                panic!("unknown test func {func_name:?}");
            });
            let param_types = func_type.params().collect::<Vec<_>>();
            let values = untyped_call.to_wasm_params::<Value>(&param_types)?;
            Ok((func_name, func_type, values))
        }

        match parse_func_call(input) {
            Ok((func_name, func_type, values)) => {
                assert!(
                    !filename.starts_with("reject-"),
                    "accepted input {input:?} in {filename}"
                );
                write!(out, "{func_name}(")?;
                let mut first = true;
                for (name, value) in func_type.param_names().zip(values) {
                    if first {
                        first = false;
                    } else {
                        write!(out, ", ")?;
                    }
                    write!(out, "{name}: {value}", value = DisplayValue(&value))?;
                }
                writeln!(out, ")")?;
            }
            Err(err) => {
                assert!(
                    !filename.starts_with("accept-"),
                    "rejected input {input:?} in {filename}: {err:#}"
                );
                writeln!(out, "{err}")?;
            }
        }
    }
    Ok(output)
}

fn get_func_type(func_name: &str) -> Option<&'static FuncType> {
    static FUNC_TYPES: OnceLock<HashMap<String, FuncType>> = OnceLock::new();
    FUNC_TYPES
        .get_or_init(|| {
            let mut resolve = Resolve::new();
            resolve.push_path("tests/ui/ui.wit").unwrap();
            resolve
                .interfaces
                .iter()
                .flat_map(|(_, i)| &i.functions)
                .map(|(name, func)| (name.clone(), resolve_wit_func_type(&resolve, func).unwrap()))
                .collect::<HashMap<_, _>>()
        })
        .get(func_name)
}

// TODO: this ought to be be migrated to use `libtest-mimic`, which other crates in this workspace
// use as well
#[test]
fn ui() -> Result<()> {
    for entry in fs::read_dir("tests/ui")? {
        let path = entry?.path();
        if path.extension().is_none() {
            continue;
        }
        if path.extension().unwrap() != "waves" {
            continue;
        }

        println!("testing {path:?}");
        let actual = test(&path)?;
        let expected = path.with_extension("out");
        if std::env::var_os("BLESS").is_some() {
            fs::write(&expected, actual)
                .with_context(|| format!("failed to write {expected:?}"))?;
        } else {
            let expected = fs::read_to_string(&expected)
                .with_context(|| format!("failed to read {expected:?}"))?;
            assert_eq!(
                expected, actual,
                "expectation `{expected}` did not match actual `{actual}`",
            );
        }
    }

    Ok(())
}