spin_trigger_http/
wagi.rs

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
use std::{io::Cursor, net::SocketAddr};

use anyhow::{ensure, Context, Result};
use http_body_util::BodyExt;
use hyper::{Request, Response};
use spin_factor_wasi::WasiFactor;
use spin_factors::RuntimeFactors;
use spin_http::{config::WagiTriggerConfig, routes::RouteMatch, wagi};
use tracing::{instrument, Level};
use wasmtime_wasi::pipe::MemoryOutputPipe;
use wasmtime_wasi_http::body::HyperIncomingBody as Body;

use crate::{headers::compute_default_headers, server::HttpExecutor, TriggerInstanceBuilder};

#[derive(Clone)]
pub struct WagiHttpExecutor {
    pub wagi_config: WagiTriggerConfig,
}

impl HttpExecutor for WagiHttpExecutor {
    #[instrument(name = "spin_trigger_http.execute_wagi", skip_all, err(level = Level::INFO), fields(otel.name = format!("execute_wagi_component {}", route_match.component_id())))]
    async fn execute<F: RuntimeFactors>(
        &self,
        mut instance_builder: TriggerInstanceBuilder<'_, F>,
        route_match: &RouteMatch,
        req: Request<Body>,
        client_addr: SocketAddr,
    ) -> Result<Response<Body>> {
        let component = route_match.component_id();

        tracing::trace!(
            "Executing request using the Wagi executor for component {}",
            component
        );

        let uri_path = req.uri().path();

        // Build the argv array by starting with the config for `argv` and substituting in
        // script name and args where appropriate.
        let script_name = uri_path.to_string();
        let args = req.uri().query().unwrap_or_default().replace('&', " ");
        let argv = self
            .wagi_config
            .argv
            .clone()
            .replace("${SCRIPT_NAME}", &script_name)
            .replace("${ARGS}", &args);

        let (parts, body) = req.into_parts();

        let body = body.collect().await?.to_bytes().to_vec();
        let len = body.len();

        // TODO
        // The default host and TLS fields are currently hard-coded.
        let mut headers =
            wagi::build_headers(route_match, &parts, len, client_addr, "default_host", false);

        let default_host = http::HeaderValue::from_str("localhost")?;
        let host = std::str::from_utf8(
            parts
                .headers
                .get("host")
                .unwrap_or(&default_host)
                .as_bytes(),
        )?;

        // Add the default Spin headers.
        // This sets the current environment variables Wagi expects (such as
        // `PATH_INFO`, or `X_FULL_URL`).
        // Note that this overrides any existing headers previously set by Wagi.
        for (keys, val) in compute_default_headers(&parts.uri, host, route_match, client_addr)? {
            headers.insert(keys[1].to_string(), val);
        }

        let stdout = MemoryOutputPipe::new(usize::MAX);

        let wasi_builder = instance_builder
            .factor_builder::<WasiFactor>()
            .context("The wagi HTTP trigger was configured without the required wasi support")?;

        // Set up Wagi environment
        wasi_builder.args(argv.split(' '));
        wasi_builder.env(headers);
        wasi_builder.stdin_pipe(Cursor::new(body));
        wasi_builder.stdout(stdout.clone());

        let (instance, mut store) = instance_builder.instantiate(()).await?;

        let command = wasmtime_wasi::bindings::Command::new(&mut store, &instance)?;

        tracing::trace!("Calling Wasm entry point");
        if let Err(()) = command
            .wasi_cli_run()
            .call_run(&mut store)
            .await
            .or_else(ignore_successful_proc_exit_trap)?
        {
            tracing::error!("Wagi main function returned unsuccessful result");
        }
        tracing::info!("Wagi execution complete");

        // Drop the store so we're left with a unique reference to `stdout`:
        drop(store);

        let stdout = stdout.try_into_inner().unwrap();
        ensure!(
            !stdout.is_empty(),
            "The {component:?} component is configured to use the WAGI executor \
             but did not write to stdout. Check the `executor` in spin.toml."
        );

        wagi::compose_response(&stdout)
    }
}

fn ignore_successful_proc_exit_trap(guest_err: anyhow::Error) -> Result<Result<(), ()>> {
    match guest_err
        .root_cause()
        .downcast_ref::<wasmtime_wasi::I32Exit>()
    {
        Some(trap) => match trap.0 {
            0 => Ok(Ok(())),
            _ => Err(guest_err),
        },
        None => Err(guest_err),
    }
}