Skip to content

Commit

Permalink
Merge pull request #888 from pothos/guestinfo-metadata-netplan
Browse files Browse the repository at this point in the history
  • Loading branch information
jlebon authored Nov 17, 2023
2 parents b6cda33 + 7e71b88 commit bceb3f4
Show file tree
Hide file tree
Showing 11 changed files with 288 additions and 3 deletions.
43 changes: 43 additions & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 2 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -35,9 +35,11 @@ debug = true

[dependencies]
anyhow = "1.0"
base64 = "0.21"
cfg-if = "1.0"
clap = { version = "4", "default_features" = false, "features" = ["std", "cargo", "derive", "error-context", "help", "suggestions", "usage", "wrap_help"] }
ipnetwork = ">= 0.17, < 0.21"
libflate = "1.3"
libsystemd = ">= 0.2.1, < 0.7.0"
mailparse = ">= 0.13, < 0.15"
maplit = "1.0"
Expand Down
13 changes: 13 additions & 0 deletions docs/development/distro.md
Original file line number Diff line number Diff line change
Expand Up @@ -14,3 +14,16 @@ Alternatively, sshd can be configured to read the fragment file directly:
```
AuthorizedKeysFile .ssh/authorized_keys .ssh/authorized_keys.d/afterburn
```

## VMware Netplan guestinfo metadata

The `guestinfo.metadata` and `guestinfo.metadata.encoding` fields can contain a Netplan configuration provided by the VM provisioning logic.
Netplan is required on the OS to render the Netplan format to either NetworkManager or systemd-networkd configuration files. By default, Netplan generates systemd-networkd units. Since the renderer backend is defined in the Netplan config itself, requiring NetworkManager in the config would rule out support for systems that don't use it (unless they would ship a drop-in file with later lexicographical ordering to force it to `networkd`). As systemd-networkd can work in parallel with NetworkManager, it's expected that the renderer field is left to its default but systems can also add a default drop-in file with early lexicographical ordering to prefer NetworkManager.

The Afterburn invocation is as follows, where `FOLDER` could be `/run/netplan/`:

```
afterburn multi --netplan-configs FOLDER --provider vmware
```

Afterwards, `netplan generate` can be used to render the config files. If that is done before `systemd-networkd` runs, this is enough, but if the network already is up, `netplan apply` should be used instead.
1 change: 1 addition & 0 deletions docs/index.md
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ It comprises several modules which may run at different times during the lifecyc
Depending on the specific platform, the following services may run in the [initramfs](https://github.com/coreos/afterburn/tree/main/dracut/30afterburn) on first boot:
* setting local hostname
* injecting [network command-line arguments](usage/initrd-network-cmdline.md)
* configuring the network with [Netplan guestinfo metadata on VMware](usage/vmware-netplan-guestinfo-metadata.md)

The following features are conditionally available on some platforms as [systemd service units](https://github.com/coreos/afterburn/tree/main/systemd):
* installing public SSH keys for local system users
Expand Down
1 change: 1 addition & 0 deletions docs/release-notes.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ nav_order: 8
Major changes:

- Add support for Scaleway
- Add Netplan guestinfo support on VMware

Minor changes:

Expand Down
4 changes: 4 additions & 0 deletions docs/usage.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,10 @@ has_toc: false

See [Initrd first-boot network arguments](usage/initrd-network-cmdline.md).

## VMware Netplan guestinfo metadata

See [VMware Netplan guestinfo metadata](usage/vmware-netplan-guestinfo-metadata.md).

## Metadata attributes

See [Metadata attributes](usage/attributes.md).
27 changes: 27 additions & 0 deletions docs/usage/vmware-netplan-guestinfo-metadata.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
---
nav_order: 2
parent: Usage
---

# VMware Netplan guestinfo metadata

The network environment can vary between VMware servers and instead of leaking these requirements into userdata snippets, a well-known guestinfo metadata field can be used.
The guestinfo metadata field is OS-independent and supported by cloud-init (spec [here](https://cloudinit.readthedocs.io/en/latest/reference/network-config-format-v2.html), example [here](https://cloudinit.readthedocs.io/en/latest/reference/datasources/vmware.html#walkthrough-of-guestinfo-keys-transport)) and Afterburn. When the OS supports this mechanism the user can provide Netplan configs which the OS renders using the backend of choice.

## Specifying the guestinfo metadata

The guestinfo keys are named `guestinfo.metadata` for the content and `guestinfo.metadata.encoding` to specify the encoding of the content.
The value of the encoding field can be empty to indicate raw string data, or one of `base64` or `b64` to indicate an base64 encoding, or one of `gzip+base64` or `gz+b64` to indicate base64-encoded gzip data.

An example for raw string data is the following:
```
network:
version: 2
ethernets:
nics:
match:
name: ens*
dhcp4: yes
```

The supported config format with examples can be found in the [Netplan specification](https://netplan.readthedocs.io/en/latest/netplan-yaml/).
9 changes: 9 additions & 0 deletions src/cli/multi.rs
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,9 @@ pub struct CliMulti {
/// The directory into which network units are written
#[arg(long = "network-units", value_name = "path")]
network_units_dir: Option<String>,
/// The directory into which a netplan config is written
#[arg(long = "netplan-config", value_name = "path")]
netplan_config_dir: Option<String>,
/// Update SSH keys for the given user
#[arg(long = "ssh-keys", value_name = "username")]
ssh_keys_user: Option<String>,
Expand All @@ -41,6 +44,7 @@ impl CliMulti {

if self.attributes_file.is_none()
&& self.network_units_dir.is_none()
&& self.netplan_config_dir.is_none()
&& !self.check_in
&& self.ssh_keys_user.is_none()
&& self.hostname_file.is_none()
Expand Down Expand Up @@ -72,6 +76,11 @@ impl CliMulti {
.map_or(Ok(()), |x| metadata.write_network_units(x))
.context("writing network units")?;

// write netplan config if configured to do so
self.netplan_config_dir
.map_or(Ok(()), |x| metadata.write_netplan_config(x))
.context("writing netplan config")?;

// perform boot check-in.
if self.check_in {
metadata
Expand Down
20 changes: 20 additions & 0 deletions src/providers/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -200,6 +200,10 @@ pub trait MetadataProvider {
Ok(vec![])
}

fn netplan_config(&self) -> Result<Option<String>> {
Ok(None)
}

fn boot_checkin(&self) -> Result<()> {
warn!("boot check-in requested, but not supported on this platform");
Ok(())
Expand Down Expand Up @@ -295,6 +299,22 @@ pub trait MetadataProvider {
}
Ok(())
}

fn write_netplan_config(&self, netplan_config_dir: String) -> Result<()> {
let dir_path = Path::new(&netplan_config_dir);
fs::create_dir_all(dir_path)
.with_context(|| format!("failed to create directory {dir_path:?}"))?;

// Write a single afterburn `.yaml` netplan config.
if let Some(netplan_config) = &self.netplan_config()? {
let file_path = dir_path.join("50-afterburn.yaml");
let mut config_file = File::create(&file_path)
.with_context(|| format!("failed to create file {file_path:?}"))?;
write!(&mut config_file, "{netplan_config}")
.with_context(|| format!("failed to write netplan config file {config_file:?}"))?;
}
Ok(())
}
}

#[cfg(test)]
Expand Down
Loading

0 comments on commit bceb3f4

Please sign in to comment.