petri_tool/
main.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
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

//! Tool for using petri functionality from the command line.

use anyhow::Context as _;
use clap::Parser;
use petri::ArtifactResolver;
use petri::TestArtifactRequirements;

/// Command line interface for petri-related tasks.
#[derive(Parser)]
struct CliArgs {
    #[clap(subcommand)]
    command: Command,
}

#[derive(clap::Subcommand)]
enum Command {
    /// Builds a cloud-init disk image for a Linux VM image.
    ///
    /// This image creates a petri user with the password "petri".
    CloudInit {
        /// The architecture of the VM image.
        #[clap(long)]
        arch: MachineArch,

        /// Path to the output disk image.
        output: std::path::PathBuf,
    },
}

#[derive(clap::ValueEnum, Clone, Copy, Debug)]
enum MachineArch {
    #[clap(name = "x86_64")]
    X86_64,
    Aarch64,
}

fn main() -> anyhow::Result<()> {
    let args = CliArgs::parse();
    match args.command {
        Command::CloudInit { output, arch } => {
            let image = build_with_artifacts(|resolver: ArtifactResolver<'_>| {
                petri::disk_image::AgentImage::new(
                    &resolver,
                    match arch {
                        MachineArch::X86_64 => petri_artifacts_common::tags::MachineArch::X86_64,
                        MachineArch::Aarch64 => petri_artifacts_common::tags::MachineArch::Aarch64,
                    },
                    petri_artifacts_common::tags::OsFlavor::Linux,
                )
            })?;

            let disk = image.build().context("failed to build disk image")?;
            disk.persist(output)
                .context("failed to persist disk image")?;

            Ok(())
        }
    }
}

fn build_with_artifacts<R>(mut f: impl FnMut(ArtifactResolver<'_>) -> R) -> anyhow::Result<R> {
    let resolver =
        petri_artifact_resolver_openvmm_known_paths::OpenvmmKnownPathsTestArtifactResolver::new("");
    let mut requirements = TestArtifactRequirements::new();
    f(ArtifactResolver::collector(&mut requirements));
    let artifacts = requirements.resolve(&resolver)?;
    Ok(f(ArtifactResolver::resolver(&artifacts)))
}