spin_trigger/cli/
launch_metadata.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
use clap::CommandFactory;
use serde::{Deserialize, Serialize};
use std::ffi::OsString;

use crate::{cli::FactorsTriggerCommand, Trigger};

use super::RuntimeFactorsBuilder;

/// Contains information about the trigger flags (and potentially
/// in future configuration) that a consumer (such as `spin up`)
/// can query using `--launch-metadata-only`.
#[derive(Clone, Debug, Deserialize, Serialize)]
pub struct LaunchMetadata {
    all_flags: Vec<LaunchFlag>,
}

// This assumes no triggers that want to participate in multi-trigger
// use positional arguments. This is a restriction we'll have to make
// anyway: suppose triggers A and B both take one positional arg, and
// the user writes `spin up 123 456` - which value would go to which trigger?
#[derive(Clone, Debug, PartialEq, Deserialize, Serialize)]
struct LaunchFlag {
    #[serde(skip_serializing_if = "Option::is_none")]
    #[serde(default)]
    short: Option<String>,
    #[serde(skip_serializing_if = "Option::is_none")]
    #[serde(default)]
    long: Option<String>,
}

impl LaunchMetadata {
    pub fn infer<T: Trigger<B::Factors>, B: RuntimeFactorsBuilder>() -> Self {
        let all_flags: Vec<_> = FactorsTriggerCommand::<T, B>::command()
            .get_arguments()
            .map(LaunchFlag::infer)
            .collect();

        LaunchMetadata { all_flags }
    }

    pub fn matches<'a>(&self, groups: &[Vec<&'a OsString>]) -> Vec<&'a OsString> {
        let mut matches = vec![];

        for group in groups {
            if group.is_empty() {
                continue;
            }
            if self.is_match(group[0]) {
                matches.extend(group);
            }
        }

        matches
    }

    fn is_match(&self, arg: &OsString) -> bool {
        self.all_flags.iter().any(|f| f.is_match(arg))
    }

    pub fn is_group_match(&self, group: &[&OsString]) -> bool {
        if group.is_empty() {
            false
        } else {
            self.all_flags.iter().any(|f| f.is_match(group[0]))
        }
    }
}

impl LaunchFlag {
    fn infer(arg: &clap::Arg) -> Self {
        Self {
            long: arg.get_long().map(|s| format!("--{s}")),
            short: arg.get_short().map(|ch| format!("-{ch}")),
        }
    }

    fn is_match(&self, candidate: &OsString) -> bool {
        let Some(s) = candidate.to_str() else {
            return false;
        };
        let candidate = Some(s.to_owned());

        candidate == self.long || candidate == self.short
    }
}