spin_manifest/schema/
common.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
use std::fmt::Display;

use serde::{Deserialize, Serialize};

use wasm_pkg_common::{package::PackageRef, registry::Registry};

/// Variable definition
#[derive(Clone, Debug, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct Variable {
    /// `required = true`
    #[serde(default, skip_serializing_if = "is_false")]
    pub required: bool,
    /// `default = "default value"`
    #[serde(default, skip_serializing_if = "Option::is_none")]
    pub default: Option<String>,
    /// `secret = true`
    #[serde(default, skip_serializing_if = "is_false")]
    pub secret: bool,
}

/// Component source
#[derive(Clone, Debug, Serialize, Deserialize)]
#[serde(deny_unknown_fields, untagged)]
pub enum ComponentSource {
    /// `"local.wasm"`
    Local(String),
    /// `{ ... }`
    Remote {
        /// `url = "https://example.test/remote.wasm"`
        url: String,
        /// `digest = `"sha256:abc123..."`
        digest: String,
    },
    /// `{ ... }`
    Registry {
        /// `registry = "example.com"`
        registry: Option<Registry>,
        /// `package = "example:component"`
        package: PackageRef,
        /// `version = "1.2.3"`
        version: String,
    },
}

impl Display for ComponentSource {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        match self {
            ComponentSource::Local(path) => write!(f, "{path:?}"),
            ComponentSource::Remote { url, digest } => write!(f, "{url:?} with digest {digest:?}"),
            ComponentSource::Registry {
                registry,
                package,
                version,
            } => {
                let registry_suffix = match registry {
                    None => "default registry".to_owned(),
                    Some(r) => format!("registry {r:?}"),
                };
                write!(f, "\"{package}@{version}\" from {registry_suffix}")
            }
        }
    }
}

/// WASI files mount
#[derive(Clone, Debug, Serialize, Deserialize)]
#[serde(deny_unknown_fields, untagged)]
pub enum WasiFilesMount {
    /// `"images/*.png"`
    Pattern(String),
    /// `{ ... }`
    Placement {
        /// `source = "content/dir"`
        source: String,
        /// `destination = "/"`
        destination: String,
    },
}

/// Component build configuration
#[derive(Clone, Debug, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct ComponentBuildConfig {
    /// `command = "cargo build"`
    pub command: Commands,
    /// `workdir = "components/main"
    #[serde(default, skip_serializing_if = "Option::is_none")]
    pub workdir: Option<String>,
    /// watch = ["src/**/*.rs"]
    #[serde(default, skip_serializing_if = "Vec::is_empty")]
    pub watch: Vec<String>,
}

impl ComponentBuildConfig {
    /// The commands to execute for the build
    pub fn commands(&self) -> impl Iterator<Item = &String> {
        let as_vec = match &self.command {
            Commands::Single(cmd) => vec![cmd],
            Commands::Multiple(cmds) => cmds.iter().collect(),
        };
        as_vec.into_iter()
    }
}

/// Component build command or commands
#[derive(Clone, Debug, Serialize, Deserialize)]
#[serde(untagged)]
pub enum Commands {
    /// `command = "cargo build"`
    Single(String),
    /// `command = ["cargo build", "wac encode compose-deps.wac -d my:pkg=app.wasm --registry fermyon.com"]`
    Multiple(Vec<String>),
}

fn is_false(v: &bool) -> bool {
    !*v
}