1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
//! <div align=center>
//!   <img src="https://data.dilla.io/dilla.png" alt="" width=320>
//!   <p><strong>Share your design system in a tiny universal package.</strong></p>
//! </div>
//!
//! Dilla is a fast but minimal WASM builder based on the syntax and behavior
//! of the [Jinja2](https://jinja.palletsprojects.com/) implemented on top of
//! [Minijinja](https://docs.rs/minijinja/latest/minijinja). The goal is to
//! be able to pack your design system into a <strong>universal</strong>
//! package, executable through a simple <strong>declarative API</strong>, for
//! both server side and headless rendering.
//!
//! To know more about Dilla visit our website [dilla.io](https://dilla.io).
//!
//! ---
//!
//! **Dilla WASM builder with WASM Component model.**
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;

/**
 * Specific implementation for wit-bindgen and bindings generation.
 */
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)
    }
}

/**
 * Module will be used by Wasmtime or Node WASI runtime.
 */
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
    );
}