use std::collections::HashSet;
use super::store::LayeredImageState;
use super::{ImageReference, OstreeImageReference};
use crate::container::store::PrepareResult;
use crate::keyfileext::KeyFileExt;
use crate::sysroot::SysrootLock;
use anyhow::Result;
use fn_error_context::context;
use ostree::glib;
pub const ORIGIN_CONTAINER: &str = "container-image-reference";
pub const STATEROOT_DEFAULT: &str = "default";
#[derive(Debug, Default)]
pub struct DeployOpts<'a> {
pub kargs: Option<&'a [&'a str]>,
pub target_imgref: Option<&'a OstreeImageReference>,
pub proxy_cfg: Option<super::store::ImageProxyConfig>,
pub no_imgref: bool,
}
#[context("Performing deployment")]
pub async fn deploy(
sysroot: &ostree::Sysroot,
stateroot: &str,
imgref: &OstreeImageReference,
options: Option<DeployOpts<'_>>,
) -> Result<Box<LayeredImageState>> {
let cancellable = ostree::gio::Cancellable::NONE;
let options = options.unwrap_or_default();
let repo = &sysroot.repo();
let merge_deployment = sysroot.merge_deployment(Some(stateroot));
let mut imp =
super::store::ImageImporter::new(repo, imgref, options.proxy_cfg.unwrap_or_default())
.await?;
if let Some(target) = options.target_imgref {
imp.set_target(target);
}
if options.no_imgref {
imp.set_no_imgref();
}
let state = match imp.prepare().await? {
PrepareResult::AlreadyPresent(r) => r,
PrepareResult::Ready(prep) => {
if let Some(warning) = prep.deprecated_warning() {
crate::cli::print_deprecated_warning(warning).await;
}
imp.import(prep).await?
}
};
let commit = state.get_commit();
let origin = glib::KeyFile::new();
let target_imgref = options.target_imgref.unwrap_or(imgref);
origin.set_string("origin", ORIGIN_CONTAINER, &target_imgref.to_string());
if sysroot.booted_deployment().is_some() {
let opts = ostree::SysrootDeployTreeOpts {
override_kernel_argv: options.kargs,
..Default::default()
};
sysroot.stage_tree_with_options(
Some(stateroot),
commit,
Some(&origin),
merge_deployment.as_ref(),
&opts,
cancellable,
)?;
} else {
let deployment = &sysroot.deploy_tree(
Some(stateroot),
commit,
Some(&origin),
merge_deployment.as_ref(),
options.kargs.unwrap_or_default(),
cancellable,
)?;
let flags = ostree::SysrootSimpleWriteDeploymentFlags::NONE;
sysroot.simple_write_deployment(
Some(stateroot),
deployment,
merge_deployment.as_ref(),
flags,
cancellable,
)?;
sysroot.cleanup(cancellable)?;
}
Ok(state)
}
fn deployment_origin_container(
deploy: &ostree::Deployment,
) -> Result<Option<OstreeImageReference>> {
let origin = deploy
.origin()
.map(|o| o.optional_string("origin", ORIGIN_CONTAINER))
.transpose()?
.flatten();
let r = origin
.map(|v| OstreeImageReference::try_from(v.as_str()))
.transpose()?;
Ok(r)
}
pub fn remove_undeployed_images(sysroot: &SysrootLock) -> Result<Vec<ImageReference>> {
let repo = &sysroot.repo();
let deployment_origins: Result<HashSet<_>> = sysroot
.deployments()
.into_iter()
.filter_map(|deploy| {
deployment_origin_container(&deploy)
.map(|v| v.map(|v| v.imgref))
.transpose()
})
.collect();
let deployment_origins = deployment_origins?;
let all_images = super::store::list_images(&sysroot.repo())?
.into_iter()
.filter_map(|img| ImageReference::try_from(img.as_str()).ok());
let mut removed = Vec::new();
for image in all_images {
if !deployment_origins.contains(&image) {
super::store::remove_image(repo, &image)?;
removed.push(image);
}
}
Ok(removed)
}