diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 47aaa00ea..84b48634d 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -116,22 +116,45 @@ jobs: run: sudo tar -C / -xvf bootc.tar.zst - name: Integration tests run: bootc internal-tests run-container-integration + build-skopeo-ubuntu: + name: "Build skopeo git main for ubuntu" + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v3 + with: + repository: containers/skopeo + path: skopeo + - name: Install build deps + run: | + sudo sed -ie s,'# deb-src,deb-src,' /etc/apt/sources.list + sudo apt update + sudo apt build-dep -y skopeo + - uses: actions/setup-go@v4 + with: + go-version: '>=1.20' + - name: Build skopeo + run: cd skopeo && make bin/skopeo PREFIX=/usr + - name: Upload binary + uses: actions/upload-artifact@v4 + with: + name: skopeo-ubuntu + path: skopeo/bin/skopeo privtest-alongside: name: "Test install-alongside" - needs: build-fedora + needs: [build-fedora, build-skopeo-ubuntu] runs-on: ubuntu-latest steps: + - name: Download + uses: actions/download-artifact@v4 + with: + name: skopeo-ubuntu + - run: chmod a+x skopeo && sudo mv skopeo /usr/bin - name: Download uses: actions/download-artifact@v3 with: name: bootc.tar.zst - name: Install run: tar -xvf bootc.tar.zst - - name: Update host skopeo - run: | - echo 'deb http://cz.archive.ubuntu.com/ubuntu lunar main universe' | sudo tee -a /etc/apt/sources.list - sudo apt update - sudo apt upgrade skopeo - name: Integration tests run: | set -xeuo pipefail diff --git a/lib/src/install.rs b/lib/src/install.rs index 3a26db230..b79f699cc 100644 --- a/lib/src/install.rs +++ b/lib/src/install.rs @@ -245,14 +245,7 @@ pub(crate) struct SourceInfo { /// Whether or not SELinux appears to be enabled in the source commit pub(crate) selinux: bool, /// Whether the source is available in the host mount namespace - pub(crate) in_host_mountns: Option, -} - -/// Information about the host mount namespace -#[derive(Debug, Clone)] -pub(crate) struct HostMountnsInfo { - /// True if the skoepo on host supports containers-storage: - pub(crate) skopeo_supports_containers_storage: bool, + pub(crate) in_host_mountns: bool, } // Shared read-only global state @@ -395,27 +388,23 @@ impl SourceInfo { }; let digest = crate::podman::imageid_to_digest(&container_info.imageid)?; - let skopeo_supports_containers_storage = skopeo_supports_containers_storage() - .context("Failed to run skopeo (it currently must be installed in the host root)")?; - Self::from( - imageref, - Some(digest), - Some(HostMountnsInfo { - skopeo_supports_containers_storage, - }), - ) + // Verify up front we can do the fetch + require_skopeo_with_containers_storage()?; + + Self::new(imageref, Some(digest), true) } #[context("Creating source info from a given imageref")] pub(crate) fn from_imageref(imageref: &str) -> Result { let imageref = ostree_container::ImageReference::try_from(imageref)?; - Self::from(imageref, None, None) + Self::new(imageref, None, false) } - fn from( + /// Construct a new source information structure + fn new( imageref: ostree_container::ImageReference, digest: Option, - in_host_mountns: Option, + in_host_mountns: bool, ) -> Result { let cancellable = ostree::gio::Cancellable::NONE; let commit = Task::new("Reading ostree commit", "ostree") @@ -612,39 +601,31 @@ async fn initialize_ostree_root_from_self( let sysroot = ostree::Sysroot::new(Some(&gio::File::for_path(rootfs))); sysroot.load(cancellable)?; - let mut temporary_dir = None; - let (src_imageref, proxy_cfg) = match &state.source.in_host_mountns { - None => (state.source.imageref.clone(), None), - Some(host_mountns_info) => { - let src_imageref = if host_mountns_info.skopeo_supports_containers_storage { - // We always use exactly the digest of the running image to ensure predictability. - let digest = state - .source - .digest - .as_ref() - .ok_or_else(|| anyhow::anyhow!("Missing container image digest"))?; - let spec = crate::utils::digested_pullspec(&state.source.imageref.name, digest); - ostree_container::ImageReference { - transport: ostree_container::Transport::ContainerStorage, - name: spec, - } - } else { - let td = tempfile::tempdir_in("/var/tmp")?; - let path: &Utf8Path = td.path().try_into().unwrap(); - let r = copy_to_oci(&state.source.imageref, path)?; - temporary_dir = Some(td); - r - }; + let (src_imageref, proxy_cfg) = if !state.source.in_host_mountns { + (state.source.imageref.clone(), None) + } else { + let src_imageref = { + // We always use exactly the digest of the running image to ensure predictability. + let digest = state + .source + .digest + .as_ref() + .ok_or_else(|| anyhow::anyhow!("Missing container image digest"))?; + let spec = crate::utils::digested_pullspec(&state.source.imageref.name, digest); + ostree_container::ImageReference { + transport: ostree_container::Transport::ContainerStorage, + name: spec, + } + }; - // We need to fetch the container image from the root mount namespace - let skopeo_cmd = run_in_host_mountns("skopeo"); - let proxy_cfg = ostree_container::store::ImageProxyConfig { - skopeo_cmd: Some(skopeo_cmd), - ..Default::default() - }; + // We need to fetch the container image from the root mount namespace + let skopeo_cmd = run_in_host_mountns("skopeo"); + let proxy_cfg = ostree_container::store::ImageProxyConfig { + skopeo_cmd: Some(skopeo_cmd), + ..Default::default() + }; - (src_imageref, Some(proxy_cfg)) - } + (src_imageref, Some(proxy_cfg)) }; let src_imageref = ostree_container::OstreeImageReference { // There are no signatures to verify since we're fetching the already @@ -671,8 +652,6 @@ async fn initialize_ostree_root_from_self( println!("Installed: {target_image}"); println!(" Digest: {digest}"); - drop(temporary_dir); - // Write the entry for /boot to /etc/fstab. TODO: Encourage OSes to use the karg? // Or better bind this with the grub data. sysroot.load(cancellable)?; @@ -717,32 +696,6 @@ async fn initialize_ostree_root_from_self( Ok(aleph) } -#[context("Copying to oci")] -fn copy_to_oci( - src_imageref: &ostree_container::ImageReference, - dir: &Utf8Path, -) -> Result { - tracing::debug!("Copying {src_imageref}"); - let src_imageref = src_imageref.to_string(); - let dest_imageref = ostree_container::ImageReference { - transport: ostree_container::Transport::OciDir, - name: dir.to_string(), - }; - let dest_imageref_str = dest_imageref.to_string(); - Task::new_cmd( - "Copying to temporary OCI (skopeo is too old)", - run_in_host_mountns("skopeo"), - ) - .args([ - "copy", - // TODO: enable this once ostree is fixed "--dest-oci-accept-uncompressed-layers", - src_imageref.as_str(), - dest_imageref_str.as_str(), - ]) - .run()?; - Ok(dest_imageref) -} - /// Run a command in the host mount namespace pub(crate) fn run_in_host_mountns(cmd: &str) -> Command { let mut c = Command::new("/proc/self/exe"); @@ -769,11 +722,12 @@ pub(crate) fn exec_in_host_mountns(args: &[std::ffi::OsString]) -> Result<()> { } #[context("Querying skopeo version")] -fn skopeo_supports_containers_storage() -> Result { +fn require_skopeo_with_containers_storage() -> Result<()> { let out = Task::new_cmd("skopeo --version", run_in_host_mountns("skopeo")) .args(["--version"]) .quiet() - .read()?; + .read() + .context("Failed to run skopeo (it currently must be installed in the host root)")?; let mut v = out .strip_prefix("skopeo version ") .map(|v| v.split('.')) @@ -785,7 +739,12 @@ fn skopeo_supports_containers_storage() -> Result { .next() .ok_or_else(|| anyhow::anyhow!("Missing minor version"))?; let (major, minor) = (major.parse::()?, minor.parse::()?); - Ok(major > 1 || minor > 10) + let supported = major > 1 || minor > 10; + if supported { + Ok(()) + } else { + anyhow::bail!("skopeo >= 1.11 is required on host") + } } pub(crate) struct RootSetup {