Skip to content

Commit

Permalink
Move platform code to its own directory
Browse files Browse the repository at this point in the history
Move the platform code into its own directory
under libimage so it can be used by farm build in
podman without causing the binary size to increase a lot.

Signed-off-by: Urvashi Mohnani <[email protected]>
  • Loading branch information
umohnani8 committed Oct 12, 2023
1 parent 09776aa commit b154d6a
Show file tree
Hide file tree
Showing 8 changed files with 114 additions and 108 deletions.
3 changes: 2 additions & 1 deletion libimage/copier.go
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ import (
"time"

"github.com/containers/common/libimage/manifests"
"github.com/containers/common/libimage/platform"
"github.com/containers/common/pkg/config"
"github.com/containers/common/pkg/retry"
"github.com/containers/image/v5/copy"
Expand Down Expand Up @@ -239,7 +240,7 @@ func (r *Runtime) newCopier(options *CopyOptions) (*copier, error) {

c.systemContext.DockerArchiveAdditionalTags = options.dockerArchiveAdditionalTags

c.systemContext.OSChoice, c.systemContext.ArchitectureChoice, c.systemContext.VariantChoice = NormalizePlatform(options.OS, options.Architecture, options.Variant)
c.systemContext.OSChoice, c.systemContext.ArchitectureChoice, c.systemContext.VariantChoice = platform.Normalize(options.OS, options.Architecture, options.Variant)

if options.SignaturePolicyPath != "" {
c.systemContext.SignaturePolicyPath = options.SignaturePolicyPath
Expand Down
11 changes: 11 additions & 0 deletions libimage/define/platform.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
package define

// PlatformPolicy controls the behavior of image-platform matching.
type PlatformPolicy int

const (
// Only debug log if an image does not match the expected platform.
PlatformPolicyDefault PlatformPolicy = iota
// Warn if an image does not match the expected platform.
PlatformPolicyWarn
)
34 changes: 34 additions & 0 deletions libimage/image.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,8 @@ import (
"strings"
"time"

"github.com/containerd/containerd/platforms"
"github.com/containers/common/libimage/platform"
"github.com/containers/image/v5/docker/reference"
"github.com/containers/image/v5/manifest"
storageTransport "github.com/containers/image/v5/storage"
Expand Down Expand Up @@ -1021,3 +1023,35 @@ func getImageID(ctx context.Context, src types.ImageReference, sys *types.System
}
return "@" + imageDigest.Encoded(), nil
}

// Checks whether the image matches the specified platform.
// Returns
// - 1) a matching error that can be used for logging (or returning) what does not match
// - 2) a bool indicating whether architecture, os or variant were set (some callers need that to decide whether they need to throw an error)
// - 3) a fatal error that occurred prior to check for matches (e.g., storage errors etc.)
func (i *Image) matchesPlatform(ctx context.Context, os, arch, variant string) (error, bool, error) {
if err := i.isCorrupted(""); err != nil {
return err, false, nil
}
inspectInfo, err := i.inspectInfo(ctx)
if err != nil {
return nil, false, fmt.Errorf("inspecting image: %w", err)
}

customPlatform := len(os)+len(arch)+len(variant) != 0

expected, err := platforms.Parse(platform.ToString(os, arch, variant))
if err != nil {
return nil, false, fmt.Errorf("parsing host platform: %v", err)
}
fromImage, err := platforms.Parse(platform.ToString(inspectInfo.Os, inspectInfo.Architecture, inspectInfo.Variant))
if err != nil {
return nil, false, fmt.Errorf("parsing image platform: %v", err)
}

if platforms.NewMatcher(expected).Match(fromImage) {
return nil, customPlatform, nil
}

return fmt.Errorf("image platform (%s) does not match the expected platform (%s)", platforms.Format(fromImage), platforms.Format(expected)), customPlatform, nil
}
3 changes: 2 additions & 1 deletion libimage/normalize_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ package libimage
import (
"testing"

lplatform "github.com/containers/common/libimage/platform"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
Expand Down Expand Up @@ -83,7 +84,7 @@ func TestNormalizePlatform(t *testing.T) {
platform{"linux", "arm", "v6"},
},
} {
os, arch, variant := NormalizePlatform(test.input.os, test.input.arch, test.input.variant)
os, arch, variant := lplatform.Normalize(test.input.os, test.input.arch, test.input.variant)
assert.Equal(t, test.expected.os, os, test.input)
assert.Equal(t, test.expected.arch, arch, test.input)
assert.Equal(t, test.expected.variant, variant, test.input)
Expand Down
100 changes: 0 additions & 100 deletions libimage/platform.go

This file was deleted.

57 changes: 57 additions & 0 deletions libimage/platform/platform.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
package platform

import (
"fmt"
"runtime"

"github.com/containerd/containerd/platforms"
v1 "github.com/opencontainers/image-spec/specs-go/v1"
"github.com/sirupsen/logrus"
)

// Normalize normalizes (according to the OCI spec) the specified os,
// arch and variant. If left empty, the individual item will be normalized.
func Normalize(rawOS, rawArch, rawVariant string) (os, arch, variant string) {
platformSpec := v1.Platform{
OS: rawOS,
Architecture: rawArch,
Variant: rawVariant,
}
normalizedSpec := platforms.Normalize(platformSpec)
if normalizedSpec.Variant == "" && rawVariant != "" {
normalizedSpec.Variant = rawVariant
}
rawPlatform := ToString(normalizedSpec.OS, normalizedSpec.Architecture, normalizedSpec.Variant)
normalizedPlatform, err := platforms.Parse(rawPlatform)
if err != nil {
logrus.Debugf("Error normalizing platform: %v", err)
return rawOS, rawArch, rawVariant
}
logrus.Debugf("Normalized platform %s to %s", rawPlatform, normalizedPlatform)
os = rawOS
if rawOS != "" {
os = normalizedPlatform.OS
}
arch = rawArch
if rawArch != "" {
arch = normalizedPlatform.Architecture
}
variant = rawVariant
if rawVariant != "" || (rawVariant == "" && normalizedPlatform.Variant != "") {
variant = normalizedPlatform.Variant
}
return os, arch, variant
}

func ToString(os, arch, variant string) string {
if os == "" {
os = runtime.GOOS
}
if arch == "" {
arch = runtime.GOARCH
}
if variant == "" {
return fmt.Sprintf("%s/%s", os, arch)
}
return fmt.Sprintf("%s/%s/%s", os, arch, variant)
}
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
package libimage
package platform

import (
"fmt"
Expand All @@ -19,7 +19,7 @@ func TestToPlatformString(t *testing.T) {
{"", "", "", fmt.Sprintf("%s/%s", runtime.GOOS, runtime.GOARCH)},
{"", "", "c", fmt.Sprintf("%s/%s/c", runtime.GOOS, runtime.GOARCH)},
} {
platform := toPlatformString(test.os, test.arch, test.variant)
platform := ToString(test.os, test.arch, test.variant)
require.Equal(t, test.expected, platform)
}
}
10 changes: 6 additions & 4 deletions libimage/runtime.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@ import (
"os"
"strings"

"github.com/containers/common/libimage/define"
"github.com/containers/common/libimage/platform"
"github.com/containers/common/pkg/config"
"github.com/containers/image/v5/docker/reference"
"github.com/containers/image/v5/pkg/shortnames"
Expand Down Expand Up @@ -184,7 +186,7 @@ type LookupImageOptions struct {
Variant string

// Controls the behavior when checking the platform of an image.
PlatformPolicy PlatformPolicy
PlatformPolicy define.PlatformPolicy

// If set, do not look for items/instances in the manifest list that
// match the current platform but return the manifest list as is.
Expand Down Expand Up @@ -283,7 +285,7 @@ func (r *Runtime) LookupImage(name string, options *LookupImageOptions) (*Image,
options.Variant = r.systemContext.VariantChoice
}
// Normalize platform to be OCI compatible (e.g., "aarch64" -> "arm64").
options.OS, options.Architecture, options.Variant = NormalizePlatform(options.OS, options.Architecture, options.Variant)
options.OS, options.Architecture, options.Variant = platform.Normalize(options.OS, options.Architecture, options.Variant)

// Second, try out the candidates as resolved by shortnames. This takes
// "localhost/" prefixed images into account as well.
Expand Down Expand Up @@ -435,9 +437,9 @@ func (r *Runtime) lookupImageInLocalStorage(name, candidate string, namedCandida
return nil, nil
}
switch options.PlatformPolicy {
case PlatformPolicyDefault:
case define.PlatformPolicyDefault:
logrus.Debugf("%v", matchError)
case PlatformPolicyWarn:
case define.PlatformPolicyWarn:
logrus.Warnf("%v", matchError)
}
}
Expand Down

0 comments on commit b154d6a

Please sign in to comment.