use std::collections::HashSet;
use anyhow::Result;
use fn_error_context::context;
use ostree::glib;
use super::store::{gc_image_layers, LayeredImageState};
use super::{ImageReference, OstreeImageReference};
use crate::container::store::PrepareResult;
use crate::keyfileext::KeyFileExt;
use crate::sysroot::SysrootLock;
pub const ORIGIN_CONTAINER: &str = "container-image-reference";
pub const STATEROOT_DEFAULT: &str = "default";
#[derive(Debug, Default)]
#[non_exhaustive]
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,
pub no_clean: 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?;
imp.require_bootable();
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.merge_commit.as_str();
let origin = glib::KeyFile::new();
let target_imgref = options.target_imgref.unwrap_or(imgref);
origin.set_string("origin", ORIGIN_CONTAINER, &target_imgref.to_string());
let opts = ostree::SysrootDeployTreeOpts {
override_kernel_argv: options.kargs,
..Default::default()
};
if sysroot.booted_deployment().is_some() {
sysroot.stage_tree_with_options(
Some(stateroot),
commit,
Some(&origin),
merge_deployment.as_ref(),
&opts,
cancellable,
)?;
} else {
let deployment = &sysroot.deploy_tree_with_options(
Some(stateroot),
commit,
Some(&origin),
merge_deployment.as_ref(),
Some(&opts),
cancellable,
)?;
let flags = if options.no_clean {
ostree::SysrootSimpleWriteDeploymentFlags::NO_CLEAN
} else {
ostree::SysrootSimpleWriteDeploymentFlags::NONE
};
sysroot.simple_write_deployment(
Some(stateroot),
deployment,
merge_deployment.as_ref(),
flags,
cancellable,
)?;
if !options.no_clean {
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)
}
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct Pruned {
pub n_images: u32,
pub n_layers: u32,
pub n_objects_pruned: u32,
pub objsize: u64,
}
impl Pruned {
pub fn is_empty(&self) -> bool {
self.n_images == 0 && self.n_layers == 0 && self.n_objects_pruned == 0
}
}
pub fn prune(sysroot: &SysrootLock) -> Result<Pruned> {
let repo = &sysroot.repo();
let n_images = remove_undeployed_images(sysroot)?.len().try_into().unwrap();
let n_layers = gc_image_layers(repo)?;
let (_, n_objects_pruned, objsize) = repo.prune(
ostree::RepoPruneFlags::REFS_ONLY,
0,
ostree::gio::Cancellable::NONE,
)?;
let n_objects_pruned = u32::try_from(n_objects_pruned).unwrap();
Ok(Pruned {
n_images,
n_layers,
n_objects_pruned,
objsize,
})
}