use bindings::Guest;
use dilla_renderer::render as dilla_render;
use std::env;
use std::fs;
use std::path::Path;
#[cfg(feature = "describer")]
use dilla_describer::describe as dilla_describe;
#[cfg(feature = "prettify")]
use html_minifier::minify;
#[cfg(feature = "prettify")]
use serde_json::Value;
mod bindings;
#[cfg(feature = "debug")]
use dilla_renderer::DESIGN_SYSTEM;
#[cfg(feature = "debug")]
const VERSION: &str = env!("CARGO_PKG_VERSION");
struct Component;
impl Guest for Component {
fn render(payload: String) -> String {
#[cfg(feature = "debug")]
info("Guest");
#[cfg(feature = "prettify")]
let value: Value = serde_json::from_str(&result).unwrap();
#[cfg(feature = "prettify")]
return serde_json::to_string_pretty(&value).unwrap();
#[cfg(not(feature = "prettify"))]
dilla_render(&payload, "json").unwrap_or("Dilla engine::render error!".to_string())
}
fn render_html(payload: String) -> String {
#[cfg(feature = "debug")]
info("Guest");
#[cfg(feature = "prettify")]
return minify(result).expect("Failed to minify string");
#[cfg(not(feature = "prettify"))]
dilla_render(&payload, "full").unwrap_or("Dilla engine::render error!".to_string())
}
fn describe(req: String) -> String {
#[cfg(feature = "debug")]
info("Guest");
main_describe(&req)
}
}
fn main() {
let help = r#"Dilla WASM Component requires two arguments.
USAGE:
wasmtime MY_WASM.wasm <function> <request>
ARGUMENTS:
<function> - can be one of the following:
render: render the payload with Dilla Engine and return a json response.
render_html: render the payload with Dilla Engine and return a text HTML response.
for a DEV build:
describe: return a JSON response from the Dilla Describe API.
<request> - request to the function:
'render'
- JSON payload file to load
- Optional flag to silence output (for benchmark)
'describe': Artefact and Id separated by '::', ie: `component::alert`
EXAMPLES:
wasmtime MY_WASM.wasm render ./payload.json
wasmtime MY_WASM.wasm render ./payload.json true
wasmtime MY_WASM.wasm render_html ./payload.json
wasmtime MY_WASM.wasm describe component::alert
wasmtime MY_WASM.wasm describe component::_list
"#;
let args: Vec<String> = env::args().collect();
#[cfg(feature = "debug")]
println!("[DEBUG] {:?}", args);
match args.len() {
3 => {
let result = dispatch(args[1].to_owned(), args[2].to_owned(), "");
#[cfg(feature = "debug")]
info("Main");
println!("{}", result);
}
4 => {
let result = dispatch(args[1].to_owned(), args[2].to_owned(), args[3].as_str());
#[cfg(feature = "debug")]
info("Main");
println!("{}", result);
}
_ => print!("{}", help),
}
}
fn dispatch(function: String, req: String, silent: &str) -> String {
match function.as_str() {
"render" => main_render(&req, silent),
"render_html" => main_render_html(&req),
"describe" => main_describe(&req),
_ => format!("Unknown function: {}", function),
}
}
fn main_render(name: &str, silent: &str) -> String {
let payload = get_payload(name);
if silent.is_empty() {
#[cfg(feature = "prettify")]
let value: Value = serde_json::from_str(&result).unwrap();
#[cfg(feature = "prettify")]
return serde_json::to_string_pretty(&value).unwrap();
#[cfg(not(feature = "prettify"))]
dilla_render(&payload, "json").unwrap_or("Dilla engine::render error!".to_string())
} else {
dilla_render(&payload, "json").unwrap_or("Dilla engine::render error!".to_string());
"".to_string()
}
}
fn main_render_html(name: &str) -> String {
let payload = get_payload(name);
#[cfg(feature = "prettify")]
return minify(result).expect("Failed to minify string");
#[cfg(not(feature = "prettify"))]
dilla_render(&payload, "full").unwrap_or("Dilla engine::render error!".to_string())
}
fn main_describe(req: &str) -> String {
let parts: Vec<&str> = req.split("::").collect();
match parts.len() {
0 => dilla_describe("", ""),
1 => dilla_describe(parts[0], ""),
_ => dilla_describe(parts[0], parts[1]),
}
}
#[cfg(not(feature = "describer"))]
fn dilla_describe(_: &str, _: &str) -> String {
"Not a DEV build, `describe` is not implemented in this Dilla Engine WASM!".to_string()
}
fn get_payload(name: &str) -> String {
if Path::new(name).exists() {
#[cfg(feature = "debug")]
println!("[DEBUG] Load file {}", name);
fs::read_to_string(name).unwrap_or_else(|_| panic!("Error opening the file: {}", name))
} else {
#[cfg(feature = "debug")]
println!("[DEBUG] Load payload");
name.to_string()
}
}
#[cfg(feature = "debug")]
fn info(scope: &str) {
#[cfg(feature = "describer")]
println!(
"[DEBUG] Dilla DEV Component {} v{VERSION} | ds: {}",
scope, DESIGN_SYSTEM
);
#[cfg(not(feature = "describer"))]
println!(
"[DEBUG] Dilla Component {} v{VERSION} | ds: {}",
scope, DESIGN_SYSTEM
);
}