Skip to content

Commit

Permalink
Merge pull request #886 from stgraber/main
Browse files Browse the repository at this point in the history
fedora: Add support for OCI images
  • Loading branch information
monstermunchkin authored Nov 4, 2024
2 parents 2c5ab32 + 034cd08 commit b067f83
Showing 1 changed file with 63 additions and 14 deletions.
77 changes: 63 additions & 14 deletions sources/fedora-http.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,8 +9,11 @@ import (
"os"
"path/filepath"
"regexp"
"slices"
"sort"

"github.com/lxc/incus/v6/shared/subprocess"

"github.com/lxc/distrobuilder/shared"
)

Expand All @@ -20,17 +23,22 @@ type fedora struct {

// Run downloads a container base image and unpacks it and its layers.
func (s *fedora) Run() error {
baseURL := fmt.Sprintf("%s/packages/Fedora-Container-Base",
s.definition.Source.URL)
base := "Fedora-Container-Base-Generic"
extension := "oci.tar.xz"
if slices.Contains([]string{"39", "40"}, s.definition.Image.Release) {
base = "Fedora-Container-Base"
extension = "tar.xz"
}

baseURL := fmt.Sprintf("%s/packages/%s", s.definition.Source.URL, base)

// Get latest build
build, err := s.getLatestBuild(baseURL, s.definition.Image.Release)
if err != nil {
return fmt.Errorf("Failed to get latest build: %w", err)
}

fname := fmt.Sprintf("Fedora-Container-Base-%s-%s.%s.tar.xz",
s.definition.Image.Release, build, s.definition.Image.ArchitectureMapped)
fname := fmt.Sprintf("%s-%s-%s.%s.%s", base, s.definition.Image.Release, build, s.definition.Image.ArchitectureMapped, extension)

// Download image
sourceURL := fmt.Sprintf("%s/%s/%s/images/%s", baseURL, s.definition.Image.Release, build, fname)
Expand All @@ -42,18 +50,59 @@ func (s *fedora) Run() error {

s.logger.WithField("file", filepath.Join(fpath, fname)).Info("Unpacking image")

// Unpack the base image
err = shared.Unpack(filepath.Join(fpath, fname), s.rootfsDir)
if err != nil {
return fmt.Errorf("Failed to unpack %q: %w", filepath.Join(fpath, fname), err)
}
if extension == "oci.tar.xz" {
// Unpack the OCI image.
ociDir, err := os.MkdirTemp(s.getTargetDir(), "oci.")
if err != nil {
return fmt.Errorf("Failed to create OCI path: %q: %w", ociDir, err)
}

s.logger.Info("Unpacking layers")
err = os.Mkdir(filepath.Join(ociDir, "image"), 0755)
if err != nil {
return fmt.Errorf("Failed to create OCI path: %q: %w", ociDir, err)
}

// Unpack the rest of the image (/bin, /sbin, /usr, etc.)
err = s.unpackLayers(s.rootfsDir)
if err != nil {
return fmt.Errorf("Failed to unpack: %w", err)
err = shared.Unpack(filepath.Join(fpath, fname), filepath.Join(ociDir, "image"))
if err != nil {
return fmt.Errorf("Failed to unpack %q: %w", filepath.Join(fpath, fname), err)
}

// Extract the image to a temporary path.
err = os.Mkdir(filepath.Join(ociDir, "content"), 0755)
if err != nil {
return fmt.Errorf("Failed to create OCI path: %q: %w", ociDir, err)
}

_, err = subprocess.RunCommand("umoci", "unpack", "--keep-dirlinks", "--image", fmt.Sprintf("%s:fedora:%s", filepath.Join(ociDir, "image"), s.definition.Image.Release), filepath.Join(ociDir, "content"))
if err != nil {
return fmt.Errorf("Failed to run umoci: %w", err)
}

// Transfer the content.
_, err = subprocess.RunCommand("rsync", "-avP", fmt.Sprintf("%s/rootfs/", filepath.Join(ociDir, "content")), s.rootfsDir)
if err != nil {
return fmt.Errorf("Failed to run rsync: %w", err)
}

// Delete the temporary directory.
err = os.RemoveAll(ociDir)
if err != nil {
return fmt.Errorf("Failed to wipe OCI directory: %w", err)
}
} else {
// Handle simple rootfs tarballs.
err = shared.Unpack(filepath.Join(fpath, fname), s.rootfsDir)
if err != nil {
return fmt.Errorf("Failed to unpack %q: %w", filepath.Join(fpath, fname), err)
}

s.logger.Info("Unpacking layers")

// Unpack the rest of the image (/bin, /sbin, /usr, etc.)
err = s.unpackLayers(s.rootfsDir)
if err != nil {
return fmt.Errorf("Failed to unpack: %w", err)
}
}

return nil
Expand Down

0 comments on commit b067f83

Please sign in to comment.