-
Notifications
You must be signed in to change notification settings - Fork 86
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
WIP: Retrieve bound images when staging new image
Signed-off-by: Chris Kyrouac <[email protected]>
- Loading branch information
Showing
5 changed files
with
167 additions
and
14 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,129 @@ | ||
use anyhow::Result; | ||
|
||
use ostree_ext::ostree::Deployment; | ||
use ostree_ext::sysroot::SysrootLock; | ||
|
||
use std::fs; | ||
use std::path::Path; | ||
|
||
use crate::task::Task; | ||
|
||
|
||
const BOOTC_SYSTEMD_DIR: &'static str = "/etc/systemd/system/bootc"; | ||
const BOOTC_QUADLET_DIR: &'static str = "/etc/containers/systemd/bootc"; | ||
const QUADLET_BINARY: &'static str = "/usr/lib/systemd/system-generators/podman-system-generator"; | ||
const SYSTEMD_DIR: &'static str = "/etc/systemd/system"; | ||
|
||
pub(crate) struct BoundImageManager { | ||
quadlet_unit_dir: String, | ||
units: Vec<String>, | ||
} | ||
|
||
impl BoundImageManager { | ||
pub(crate) fn new(deployment: Deployment, sysroot: &SysrootLock) -> BoundImageManager { | ||
let deployment_dir = sysroot.deployment_dirpath(&deployment); | ||
let quadlet_unit_dir = "/".to_string() + deployment_dir.as_str() + BOOTC_QUADLET_DIR.to_string().as_str(); | ||
|
||
BoundImageManager { | ||
quadlet_unit_dir, | ||
units: Vec::new(), | ||
} | ||
} | ||
|
||
pub(crate) fn run(&mut self) -> Result<()> { | ||
match self.sync_images() { | ||
Ok(_) => println!("Successfully synced images"), | ||
Err(e) => { | ||
self.clean_up()?; | ||
drop(e); | ||
} | ||
} | ||
|
||
Ok(()) | ||
} | ||
|
||
fn sync_images(&mut self) -> Result<()> { | ||
if Path::new(&self.quadlet_unit_dir).exists() { | ||
self.run_quadlet()?; | ||
self.move_units()?; | ||
self.restart_systemd()?; | ||
self.start_new_services()?; | ||
self.clean_up()?; | ||
} | ||
|
||
Ok(()) | ||
} | ||
|
||
// Run podman-system-generator to generate the systemd units | ||
// the output is written to /etc/systemd/system/bootc | ||
// in order to track the generated units. | ||
// The generated units need to be moved to /etc/systemd/system | ||
// to be started by systemd. | ||
fn run_quadlet(&self) -> Result<()> { | ||
fs::create_dir_all(BOOTC_SYSTEMD_DIR)?; | ||
|
||
Task::new( | ||
format!("Running quadlet on {:#}", self.quadlet_unit_dir), | ||
QUADLET_BINARY, | ||
) | ||
.arg(BOOTC_SYSTEMD_DIR) | ||
.env(&"QUADLET_UNIT_DIRS".to_string(), &self.quadlet_unit_dir) | ||
.run()?; | ||
|
||
Ok(()) | ||
} | ||
|
||
fn move_units(&mut self) -> Result<()> { | ||
let entries = fs::read_dir(BOOTC_SYSTEMD_DIR)?; | ||
for bound_image in entries { | ||
let bound_image = bound_image?; | ||
let bound_image_path = bound_image.path(); | ||
let unit_name = bound_image_path.file_name().unwrap().to_str().unwrap(); | ||
|
||
//move the unit file from the bootc subdirectory to the root systemd directory | ||
let systemd_dst = format!( | ||
"{}/{}", | ||
SYSTEMD_DIR, | ||
unit_name, | ||
); | ||
if !Path::new(systemd_dst.as_str()).exists() { | ||
fs::rename(bound_image_path.clone(), systemd_dst.clone())?; | ||
} | ||
|
||
self.units.push(unit_name.to_string()); | ||
} | ||
|
||
Ok(()) | ||
} | ||
|
||
fn restart_systemd(&self) -> Result<()> { | ||
Task::new_and_run("Reloading systemd", "/usr/bin/systemctl", ["daemon-reload"])?; | ||
Ok(()) | ||
} | ||
|
||
fn start_new_services(&self) -> Result<()> { | ||
//TODO: do this in parallel | ||
for unit in &self.units { | ||
Task::new_and_run( | ||
format!("Starting target: {:#}", unit), | ||
"/usr/bin/systemctl", | ||
["start", unit], | ||
)?; | ||
} | ||
Ok(()) | ||
} | ||
|
||
fn clean_up(&self) -> Result<()> { | ||
//remove the temp directory used for the generated units | ||
if Path::new(BOOTC_SYSTEMD_DIR).exists() { | ||
fs::remove_dir_all(BOOTC_SYSTEMD_DIR)? | ||
} | ||
|
||
//remove the generated units from the root systemd directory | ||
for unit in &self.units { | ||
fs::remove_file(format!("{}/{}", SYSTEMD_DIR, unit))?; | ||
} | ||
|
||
Ok(()) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters