spin_doctor/wasm/
missing.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
use std::process::Command;

use anyhow::{ensure, Context, Result};
use async_trait::async_trait;
use spin_common::ui::quoted_path;

use crate::{Diagnosis, PatientApp, Treatment};

use super::{PatientWasm, WasmDiagnostic};

/// WasmMissingDiagnostic detects missing Wasm sources.
#[derive(Default)]
pub struct WasmMissingDiagnostic;

#[async_trait]
impl WasmDiagnostic for WasmMissingDiagnostic {
    type Diagnosis = WasmMissing;

    async fn diagnose_wasm(
        &self,
        _app: &PatientApp,
        wasm: PatientWasm,
    ) -> anyhow::Result<Vec<Self::Diagnosis>> {
        if let Some(abs_path) = wasm.abs_source_path() {
            if !abs_path.exists() {
                return Ok(vec![WasmMissing(wasm)]);
            }
        }
        Ok(vec![])
    }
}

/// WasmMissing represents a missing Wasm source.
#[derive(Debug)]
pub struct WasmMissing(PatientWasm);

impl WasmMissing {
    fn build_cmd(&self, patient: &PatientApp) -> Result<Command> {
        let spin_bin = std::env::current_exe().context("Couldn't find spin executable")?;
        let mut cmd = Command::new(spin_bin);
        cmd.arg("build")
            .arg("-f")
            .arg(&patient.manifest_path)
            .arg("--component-id")
            .arg(self.0.component_id());
        Ok(cmd)
    }
}

impl Diagnosis for WasmMissing {
    fn description(&self) -> String {
        let id = self.0.component_id();
        let Some(rel_path) = self.0.source_path() else {
            unreachable!("unsupported source");
        };
        format!(
            "Component {id:?} source {} is missing",
            quoted_path(rel_path)
        )
    }

    fn treatment(&self) -> Option<&dyn Treatment> {
        self.0.has_build().then_some(self)
    }
}

#[async_trait]
impl Treatment for WasmMissing {
    fn summary(&self) -> String {
        "Run `spin build`".into()
    }

    async fn dry_run(&self, patient: &PatientApp) -> anyhow::Result<String> {
        let args = self
            .build_cmd(patient)?
            .get_args()
            .map(|arg| arg.to_string_lossy())
            .collect::<Vec<_>>()
            .join(" ");
        Ok(format!("Run `spin {args}`"))
    }

    async fn treat(&self, patient: &mut PatientApp) -> anyhow::Result<()> {
        let mut cmd = self.build_cmd(patient)?;
        let status = cmd.status()?;
        ensure!(status.success(), "Build command {cmd:?} failed: {status:?}");
        Ok(())
    }
}

#[cfg(test)]
mod tests {
    use crate::test::{assert_single_diagnosis, TestPatient};

    use super::*;

    const MINIMUM_VIABLE_MANIFEST: &str = r#"
            spin_manifest_version = "1"
            name = "wasm-missing-test"
            version = "0.0.0"
            trigger = { type = "test" }
            [[component]]
            id = "missing-source"
            source = "does-not-exist.wasm"
            trigger = {}
        "#;

    #[tokio::test]
    async fn test_without_build() {
        let patient = TestPatient::from_toml_str(MINIMUM_VIABLE_MANIFEST);
        let diag = assert_single_diagnosis::<WasmMissingDiagnostic>(&patient).await;
        assert!(diag.treatment().is_none());
    }

    #[tokio::test]
    async fn test_with_build() {
        let manifest = format!("{MINIMUM_VIABLE_MANIFEST}\nbuild.command = 'true'");
        let patient = TestPatient::from_toml_str(manifest);
        let diag = assert_single_diagnosis::<WasmMissingDiagnostic>(&patient).await;
        assert!(diag.treatment().is_some());
        assert!(diag
            .build_cmd(&patient)
            .unwrap()
            .get_args()
            .any(|arg| arg == "missing-source"));
    }
}