Move package set chain collation to the distro package and add repositories to the package sets while returning the package sets from their source, i.e., the ImageType.PackageSets() method. This also removes the concept of "base repositories". There are no longer repositories that are added implicitly to all package sets but instead each package set needs to specify *all* the repositories it will be depsolved against. This paves the way for the requirement we have for building RHEL 7 images with a RHEL 8 build root. The build root package set has to be depsolved against RHEL 8 repositories without any "base repos" included. This is now possible since package sets and repositories are explicitly associated from the start and there is no implicit global repository set. The change requires adding a list of PackageSet names to the core rpmmd.RepoConfig. In the cloud API, repositories that are limited to specific package sets already contain the correct package set names and these are now copied to the internal RepoConfig when converting types in genRepoConfig(). The user-specified repositories are only associated with the payload package sets like before.
329 lines
9.2 KiB
Go
329 lines
9.2 KiB
Go
package distro
|
|
|
|
import (
|
|
"bufio"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/osbuild/osbuild-composer/internal/blueprint"
|
|
"github.com/osbuild/osbuild-composer/internal/disk"
|
|
"github.com/osbuild/osbuild-composer/internal/ostree"
|
|
"github.com/osbuild/osbuild-composer/internal/rpmmd"
|
|
)
|
|
|
|
const (
|
|
// architecture names
|
|
|
|
X86_64ArchName = "x86_64"
|
|
Aarch64ArchName = "aarch64"
|
|
Ppc64leArchName = "ppc64le"
|
|
S390xArchName = "s390x"
|
|
)
|
|
|
|
type BootType string
|
|
|
|
const (
|
|
UnsetBootType BootType = ""
|
|
LegacyBootType BootType = "legacy"
|
|
UEFIBootType BootType = "uefi"
|
|
HybridBootType BootType = "hybrid"
|
|
)
|
|
|
|
// A Distro represents composer's notion of what a given distribution is.
|
|
type Distro interface {
|
|
// Returns the name of the distro.
|
|
Name() string
|
|
|
|
// Returns the release version of the distro. This is used in repo
|
|
// files on the host system and required for the subscription support.
|
|
Releasever() string
|
|
|
|
// Returns the module platform id of the distro. This is used by DNF
|
|
// for modularity support.
|
|
ModulePlatformID() string
|
|
|
|
// Returns the ostree reference template
|
|
OSTreeRef() string
|
|
|
|
// Returns a sorted list of the names of the architectures this distro
|
|
// supports.
|
|
ListArches() []string
|
|
|
|
// Returns an object representing the given architecture as support
|
|
// by this distro.
|
|
GetArch(arch string) (Arch, error)
|
|
}
|
|
|
|
// An Arch represents a given distribution's support for a given architecture.
|
|
type Arch interface {
|
|
// Returns the name of the architecture.
|
|
Name() string
|
|
|
|
// Returns a sorted list of the names of the image types this architecture
|
|
// supports.
|
|
ListImageTypes() []string
|
|
|
|
// Returns an object representing a given image format for this architecture,
|
|
// on this distro.
|
|
GetImageType(imageType string) (ImageType, error)
|
|
|
|
// Returns the parent distro
|
|
Distro() Distro
|
|
}
|
|
|
|
// An ImageType represents a given distribution's support for a given Image Type
|
|
// for a given architecture.
|
|
type ImageType interface {
|
|
// Returns the name of the image type.
|
|
Name() string
|
|
|
|
// Returns the parent architecture
|
|
Arch() Arch
|
|
|
|
// Returns the canonical filename for the image type.
|
|
Filename() string
|
|
|
|
// Retrns the MIME-type for the image type.
|
|
MIMEType() string
|
|
|
|
// Returns the default OSTree ref for the image type.
|
|
OSTreeRef() string
|
|
|
|
// Returns the proper image size for a given output format. If the input size
|
|
// is 0 the default value for the format will be returned.
|
|
Size(size uint64) uint64
|
|
|
|
// Returns the corresponding partion type ("gpt", "dos") or "" the image type
|
|
// has no partition table. Only support for RHEL 8.5+
|
|
PartitionType() string
|
|
|
|
// Returns the sets of packages to include and exclude when building the image.
|
|
// Indexed by a string label. How each set is labeled and used depends on the
|
|
// image type.
|
|
PackageSets(bp blueprint.Blueprint, repos []rpmmd.RepoConfig) map[string][]rpmmd.PackageSet
|
|
|
|
// Returns the names of the pipelines that set up the build environment (buildroot).
|
|
BuildPipelines() []string
|
|
|
|
// Returns the names of the pipelines that create the image.
|
|
PayloadPipelines() []string
|
|
|
|
// Returns the package set names safe to install custom packages via custom repositories.
|
|
PayloadPackageSets() []string
|
|
|
|
// Returns named arrays of package set names which should be depsolved in a chain.
|
|
PackageSetsChains() map[string][]string
|
|
|
|
// Returns the names of the stages that will produce the build output.
|
|
Exports() []string
|
|
|
|
// Returns an osbuild manifest, containing the sources and pipeline necessary
|
|
// to build an image, given output format with all packages and customizations
|
|
// specified in the given blueprint. The packageSpecSets must be labelled in
|
|
// the same way as the originating PackageSets.
|
|
Manifest(b *blueprint.Customizations, options ImageOptions, repos []rpmmd.RepoConfig, packageSpecSets map[string][]rpmmd.PackageSpec, seed int64) (Manifest, error)
|
|
}
|
|
|
|
// The ImageOptions specify options for a specific image build
|
|
type ImageOptions struct {
|
|
OSTree ostree.RequestParams
|
|
Size uint64
|
|
Subscription *SubscriptionImageOptions
|
|
}
|
|
|
|
// The SubscriptionImageOptions specify subscription-specific image options
|
|
// ServerUrl denotes the host to register the system with
|
|
// BaseUrl specifies the repository URL for DNF
|
|
type SubscriptionImageOptions struct {
|
|
Organization string
|
|
ActivationKey string
|
|
ServerUrl string
|
|
BaseUrl string
|
|
Insights bool
|
|
}
|
|
|
|
type BasePartitionTableMap map[string]disk.PartitionTable
|
|
|
|
// A Manifest is an opaque JSON object, which is a valid input to osbuild
|
|
type Manifest []byte
|
|
|
|
func (m Manifest) MarshalJSON() ([]byte, error) {
|
|
return json.RawMessage(m).MarshalJSON()
|
|
}
|
|
|
|
func (m *Manifest) UnmarshalJSON(payload []byte) error {
|
|
var raw json.RawMessage
|
|
err := (&raw).UnmarshalJSON(payload)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
*m = Manifest(raw)
|
|
return nil
|
|
}
|
|
|
|
type manifestVersion struct {
|
|
Version string `json:"version"`
|
|
}
|
|
|
|
func (m Manifest) Version() (string, error) {
|
|
mver := new(manifestVersion)
|
|
if err := json.Unmarshal(m, mver); err != nil {
|
|
return "", err
|
|
}
|
|
|
|
switch mver.Version {
|
|
case "":
|
|
return "1", nil
|
|
case "2":
|
|
return "2", nil
|
|
default:
|
|
return "", fmt.Errorf("Unsupported Manifest version %s", mver.Version)
|
|
}
|
|
}
|
|
|
|
func GetHostDistroName() (string, bool, bool, error) {
|
|
f, err := os.Open("/etc/os-release")
|
|
if err != nil {
|
|
return "", false, false, err
|
|
}
|
|
defer f.Close()
|
|
osrelease, err := readOSRelease(f)
|
|
if err != nil {
|
|
return "", false, false, err
|
|
}
|
|
|
|
isStream := osrelease["NAME"] == "CentOS Stream"
|
|
|
|
// NOTE: We only consider major releases up until rhel 8.4
|
|
version := strings.Split(osrelease["VERSION_ID"], ".")
|
|
name := osrelease["ID"] + "-" + version[0]
|
|
if osrelease["ID"] == "rhel" && ((version[0] == "8" && version[1] >= "4") || version[0] == "9") {
|
|
name = name + version[1]
|
|
}
|
|
|
|
// TODO: We should probably index these things by the full CPE
|
|
beta := strings.Contains(osrelease["CPE_NAME"], "beta")
|
|
return name, beta, isStream, nil
|
|
}
|
|
|
|
// GetRedHatRelease returns the content of /etc/redhat-release
|
|
// without the trailing new-line.
|
|
func GetRedHatRelease() (string, error) {
|
|
raw, err := ioutil.ReadFile("/etc/redhat-release")
|
|
if err != nil {
|
|
return "", fmt.Errorf("cannot read /etc/redhat-release: %v", err)
|
|
}
|
|
|
|
//Remove the trailing new-line.
|
|
redHatRelease := strings.TrimSpace(string(raw))
|
|
|
|
return redHatRelease, nil
|
|
}
|
|
|
|
func readOSRelease(r io.Reader) (map[string]string, error) {
|
|
osrelease := make(map[string]string)
|
|
scanner := bufio.NewScanner(r)
|
|
for scanner.Scan() {
|
|
line := strings.TrimSpace(scanner.Text())
|
|
if len(line) == 0 {
|
|
continue
|
|
}
|
|
|
|
parts := strings.SplitN(line, "=", 2)
|
|
if len(parts) != 2 {
|
|
return nil, errors.New("readOSRelease: invalid input")
|
|
}
|
|
|
|
key := strings.TrimSpace(parts[0])
|
|
value := strings.TrimSpace(parts[1])
|
|
if value[0] == '"' {
|
|
if len(value) < 2 || value[len(value)-1] != '"' {
|
|
return nil, errors.New("readOSRelease: invalid input")
|
|
}
|
|
value = value[1 : len(value)-1]
|
|
}
|
|
|
|
osrelease[key] = value
|
|
}
|
|
|
|
return osrelease, nil
|
|
}
|
|
|
|
// Fallbacks: When a new method is added to an interface to provide to provide
|
|
// information that isn't available for older implementations, the older
|
|
// methods should return a fallback/default value by calling the appropriate
|
|
// function from below.
|
|
// Example: Exports() simply returns "assembler" for older image type
|
|
// implementations that didn't produce v1 manifests that have named pipelines.
|
|
func BuildPipelinesFallback() []string {
|
|
return []string{"build"}
|
|
}
|
|
|
|
func PayloadPipelinesFallback() []string {
|
|
return []string{"os", "assembler"}
|
|
}
|
|
|
|
func ExportsFallback() []string {
|
|
return []string{"assembler"}
|
|
}
|
|
|
|
func PayloadPackageSets() []string {
|
|
return []string{}
|
|
}
|
|
|
|
func MakePackageSetChains(t ImageType, packageSets map[string]rpmmd.PackageSet, repos []rpmmd.RepoConfig) map[string][]rpmmd.PackageSet {
|
|
allSetNames := make([]string, len(packageSets))
|
|
idx := 0
|
|
for setName := range packageSets {
|
|
allSetNames[idx] = setName
|
|
idx++
|
|
}
|
|
|
|
// map repository PackageSets to the list of repo configs
|
|
packageSetsRepos := make(map[string][]rpmmd.RepoConfig)
|
|
for idx := range repos {
|
|
repo := repos[idx]
|
|
if len(repo.PackageSets) == 0 {
|
|
// repos that don't specify package sets get used everywhere
|
|
repo.PackageSets = allSetNames
|
|
}
|
|
for _, name := range repo.PackageSets {
|
|
psRepos := packageSetsRepos[name]
|
|
psRepos = append(psRepos, repo)
|
|
packageSetsRepos[name] = psRepos
|
|
}
|
|
}
|
|
|
|
chainedSets := make(map[string][]rpmmd.PackageSet)
|
|
addedSets := make(map[string]bool)
|
|
// first collect package sets that are part of a chain
|
|
for specName, setNames := range t.PackageSetsChains() {
|
|
pkgSets := make([]rpmmd.PackageSet, len(setNames))
|
|
|
|
// add package-set-specific repositories to each set if one is defined
|
|
for idx, pkgSetName := range setNames {
|
|
pkgSets[idx] = packageSets[pkgSetName]
|
|
pkgSets[idx].Repositories = packageSetsRepos[pkgSetName]
|
|
addedSets[pkgSetName] = true
|
|
}
|
|
chainedSets[specName] = pkgSets
|
|
}
|
|
|
|
// add the rest of the package sets
|
|
for name, pkgSet := range packageSets {
|
|
if addedSets[name] {
|
|
// already added
|
|
continue
|
|
}
|
|
pkgSet.Repositories = packageSetsRepos[name]
|
|
chainedSets[name] = []rpmmd.PackageSet{pkgSet}
|
|
addedSets[name] = true // NOTE: not really necessary but good book-keeping in case this function gets expanded
|
|
}
|
|
|
|
return chainedSets
|
|
}
|