spin_trigger/loader.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 129 130 131 132
use anyhow::Context as _;
use spin_common::{ui::quoted_path, url::parse_file_url};
use spin_core::{async_trait, wasmtime, Component};
use spin_factors::AppComponent;
#[derive(Default)]
pub struct ComponentLoader {
_private: (),
#[cfg(feature = "unsafe-aot-compilation")]
aot_compilation_enabled: bool,
}
impl ComponentLoader {
/// Create a new `ComponentLoader`
pub fn new() -> Self {
Self::default()
}
/// Updates the TriggerLoader to load AOT precompiled components
///
/// **Warning: This feature may bypass important security guarantees of the
/// Wasmtime security sandbox if used incorrectly! Read this documentation
/// carefully.**
///
/// Usually, components are compiled just-in-time from portable Wasm
/// sources. This method causes components to instead be loaded
/// ahead-of-time as Wasmtime-precompiled native executable binaries.
/// Precompiled binaries must be produced with a compatible Wasmtime engine
/// using the same Wasmtime version and compiler target settings - typically
/// by a host with the same processor that will be executing them. See the
/// Wasmtime documentation for more information:
/// https://docs.rs/wasmtime/latest/wasmtime/struct.Module.html#method.deserialize
///
/// # Safety
///
/// This method is marked as `unsafe` because it enables potentially unsafe
/// behavior if used to load malformed or malicious precompiled binaries.
/// Loading sources from an incompatible Wasmtime engine will fail but is
/// otherwise safe. This method is safe if it can be guaranteed that
/// `<TriggerLoader as Loader>::load_component` will only ever be called
/// with a trusted `LockedComponentSource`. **Precompiled binaries must
/// never be loaded from untrusted sources.**
#[cfg(feature = "unsafe-aot-compilation")]
pub unsafe fn enable_loading_aot_compiled_components(&mut self) {
self.aot_compilation_enabled = true;
}
#[cfg(feature = "unsafe-aot-compilation")]
fn load_precompiled_component(
&self,
engine: &wasmtime::Engine,
path: &std::path::Path,
) -> anyhow::Result<Component> {
assert!(self.aot_compilation_enabled);
match engine.detect_precompiled_file(path)? {
Some(wasmtime::Precompiled::Component) => unsafe {
Component::deserialize_file(engine, path)
},
Some(wasmtime::Precompiled::Module) => {
anyhow::bail!("expected AOT compiled component but found module");
}
None => {
anyhow::bail!("expected AOT compiled component but found other data");
}
}
}
}
#[async_trait]
impl spin_factors_executor::ComponentLoader for ComponentLoader {
async fn load_component(
&self,
engine: &wasmtime::Engine,
component: &AppComponent,
) -> anyhow::Result<Component> {
let source = component
.source()
.content
.source
.as_ref()
.context("LockedComponentSource missing source field")?;
let path = parse_file_url(source)?;
#[cfg(feature = "unsafe-aot-compilation")]
if self.aot_compilation_enabled {
return self
.load_precompiled_component(engine, &path)
.with_context(|| format!("error deserializing component from {path:?}"));
}
let composed = spin_compose::compose(&ComponentSourceLoader, component.locked)
.await
.with_context(|| {
format!(
"failed to resolve dependencies for component {:?}",
component.locked.id
)
})?;
spin_core::Component::new(engine, composed)
.with_context(|| format!("failed to compile component from {}", quoted_path(&path)))
}
}
struct ComponentSourceLoader;
#[async_trait]
impl spin_compose::ComponentSourceLoader for ComponentSourceLoader {
async fn load_component_source(
&self,
source: &spin_app::locked::LockedComponentSource,
) -> anyhow::Result<Vec<u8>> {
let source = source
.content
.source
.as_ref()
.context("LockedComponentSource missing source field")?;
let path = parse_file_url(source)?;
let bytes: Vec<u8> = tokio::fs::read(&path).await.with_context(|| {
format!(
"failed to read component source from disk at path {}",
quoted_path(&path)
)
})?;
let component = spin_componentize::componentize_if_necessary(&bytes)?;
Ok(component.into())
}
}