Skip to content

Commit

Permalink
feat: support platform override.
Browse files Browse the repository at this point in the history
Support setting `platform` for dependencies and also allow overriding
`platform` for a specific package.

Signed-off-by: Noel Georgi <[email protected]>
  • Loading branch information
frezbo committed Mar 5, 2024
1 parent 6db0478 commit 5052222
Show file tree
Hide file tree
Showing 13 changed files with 143 additions and 23 deletions.
4 changes: 4 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -225,6 +225,8 @@ Additionally, hermetic text functions from [Sprig](http://masterminds.github.io/
On the root level, following properties are available:

- `name` (*str*, *required*): name of the package, also used to reference this package from other packages as dependency.
- `platform` (*str*, *optional*): platform override for the build.
If not set, defaults to the platform of the build.
- `variant` (*str*, *optional*): variant of the base image of the build.
Two variants are available:
- `alpine`: Alpine Linux 3.16 image with `bash` package pre-installed
Expand Down Expand Up @@ -268,6 +270,8 @@ Properties:
Contents of the stage are poured into the build at the location specified with `to:` parameter.
- `image` (*str*, *external dependency*): reference to the registry container image this package depends on.
Contents of the image are poured into the build at the location specified with `to:` parameter.
- `platform` (*str*, *optional*): platform to pull the image for.
If not set, defaults to the platform of the build.
- `runtime` (*bool*, *optional*): if set, marks dependency as runtime.
This means that when this package is pulled in into the build, all the runtime dependencies are pulled in automatically as well.
This also applies to transitive runtime dependencies.
Expand Down
2 changes: 1 addition & 1 deletion go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ require (
github.com/Masterminds/semver v1.5.0
github.com/Masterminds/sprig/v3 v3.2.3
github.com/alessio/shellescape v1.4.2
github.com/containerd/containerd v1.7.11
github.com/containerd/containerd v1.7.13
github.com/emicklei/dot v1.6.1
github.com/google/go-github/v60 v60.0.0
github.com/hashicorp/go-multierror v1.1.1
Expand Down
4 changes: 2 additions & 2 deletions go.sum
Original file line number Diff line number Diff line change
Expand Up @@ -12,8 +12,8 @@ github.com/Microsoft/hcsshim v0.11.4 h1:68vKo2VN8DE9AdN4tnkWnmdhqdbpUFM8OF3Airm7
github.com/Microsoft/hcsshim v0.11.4/go.mod h1:smjE4dvqPX9Zldna+t5FG3rnoHhaB7QYxPRqGcpAD9w=
github.com/alessio/shellescape v1.4.2 h1:MHPfaU+ddJ0/bYWpgIeUnQUqKrlJ1S7BfEYPM4uEoM0=
github.com/alessio/shellescape v1.4.2/go.mod h1:PZAiSCk0LJaZkiCSkPv8qIobYglO3FPpyFjDCtHLS30=
github.com/containerd/containerd v1.7.11 h1:lfGKw3eU35sjV0aG2eYZTiwFEY1pCzxdzicHP3SZILw=
github.com/containerd/containerd v1.7.11/go.mod h1:5UluHxHTX2rdvYuZ5OJTC5m/KJNs0Zs9wVoJm9zf5ZE=
github.com/containerd/containerd v1.7.13 h1:wPYKIeGMN8vaggSKuV1X0wZulpMz4CrgEsZdaCyB6Is=
github.com/containerd/containerd v1.7.13/go.mod h1:zT3up6yTRfEUa6+GsITYIJNgSVL9NQ4x4h1RPzk0Wu4=
github.com/containerd/continuity v0.4.2 h1:v3y/4Yz5jwnvqPKJJ+7Wf93fyWoCB3F5EclWG023MDM=
github.com/containerd/continuity v0.4.2/go.mod h1:F6PTNCKepoxEaXLQp3wDAjygEnImnZ/7o4JzpodfroQ=
github.com/containerd/log v0.1.0 h1:TCJt7ioM2cr/tfR8GPbGf9/VRAX8D2B4PjzCpfX540I=
Expand Down
58 changes: 50 additions & 8 deletions internal/pkg/convert/graph.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import (
"sort"

"github.com/moby/buildkit/client/llb"
v1 "github.com/opencontainers/image-spec/specs-go/v1"

"github.com/siderolabs/bldr/internal/pkg/constants"
"github.com/siderolabs/bldr/internal/pkg/environment"
Expand All @@ -29,19 +30,24 @@ type GraphLLB struct {
LocalContext llb.State

baseImageProcessor llbProcessor
cache map[*solver.PackageNode]llb.State
cache map[cacheState]llb.State

commonRunOptions []llb.RunOption
}

type cacheState struct {
*solver.PackageNode
Platform string
}

type llbProcessor func(llb.State) llb.State

// NewGraphLLB creates new GraphLLB and initializes shared images.
func NewGraphLLB(graph *solver.PackageGraph, options *environment.Options) *GraphLLB {
result := &GraphLLB{
PackageGraph: graph,
Options: options,
cache: make(map[*solver.PackageNode]llb.State),
cache: make(map[cacheState]llb.State),
}

if options.ProxyEnv != nil {
Expand All @@ -55,6 +61,23 @@ func NewGraphLLB(graph *solver.PackageGraph, options *environment.Options) *Grap
return result
}

func (graph *GraphLLB) convertPlatform(platform string) *v1.Platform {
switch platform {
case "linux/amd64":
return &v1.Platform{
OS: "linux",
Architecture: "amd64",
}
case "linux/arm64":
return &v1.Platform{
OS: "linux",
Architecture: "arm64",
}
default:
return nil
}
}

func (graph *GraphLLB) buildBaseImages() {
graph.BaseImages = make(map[v1alpha2.Variant]llb.State)

Expand Down Expand Up @@ -86,10 +109,20 @@ func (graph *GraphLLB) buildBaseImages() {
return addEnv(addPkg(root))
}

graph.BaseImages[v1alpha2.Alpine] = graph.baseImageProcessor(llb.Image(
alpineBase := llb.Image(
constants.DefaultBaseImage,
llb.WithCustomName(graph.Options.CommonPrefix+"base"),
).Run(
)

if platform := graph.convertPlatform(graph.Root.Pkg.Platform); platform != nil {
alpineBase = llb.Image(
constants.DefaultBaseImage,
llb.WithCustomName(graph.Options.CommonPrefix+"base"),
llb.Platform(*platform),
)
}

graph.BaseImages[v1alpha2.Alpine] = graph.baseImageProcessor(alpineBase).Run(
append(graph.commonRunOptions,
llb.Shlex("apk --no-cache --update add bash"),
llb.WithCustomName(graph.Options.CommonPrefix+"base-apkinstall"),
Expand All @@ -99,16 +132,25 @@ func (graph *GraphLLB) buildBaseImages() {
llb.Args([]string{"ln", "-svf", "/bin/bash", "/bin/sh"}),
llb.WithCustomName(graph.Options.CommonPrefix+"base-symlink"),
)...,
).Root())
).Root()

graph.BaseImages[v1alpha2.Scratch] = graph.baseImageProcessor(llb.Scratch())
}

func (graph *GraphLLB) buildChecksummer() {
graph.Checksummer = llb.Image(
alpineBase := llb.Image(
constants.DefaultBaseImage,
llb.WithCustomName(graph.Options.CommonPrefix+"cksum"),
).Run(
)

if platform := graph.convertPlatform(graph.Root.Pkg.Platform); platform != nil {
alpineBase = llb.Image(
constants.DefaultBaseImage,
llb.WithCustomName(graph.Options.CommonPrefix+"cksum"),
llb.Platform(*platform))
}

graph.Checksummer = alpineBase.Run(
append(graph.commonRunOptions,
llb.Shlex("apk --no-cache --update add coreutils"),
llb.WithCustomName(graph.Options.CommonPrefix+"cksum-apkinstall"),
Expand All @@ -132,7 +174,7 @@ func (graph *GraphLLB) buildLocalContext() {

// Build converts package graph to LLB.
func (graph *GraphLLB) Build() (llb.State, error) {
return NewNodeLLB(graph.Root, graph).Build()
return NewNodeLLB(graph.Root, graph).Build(graph.Root.Pkg.Platform)
}

// Marshal returns marshaled LLB.
Expand Down
47 changes: 41 additions & 6 deletions internal/pkg/convert/node.go
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,20 @@ type NodeLLB struct {

Graph *GraphLLB
Prefix string
Platform
}

// Platform wraps v1.Platform to provide Stringer interface.
type Platform struct {
v1.Platform
}

func (p Platform) String() string {
if p.OS == "" && p.Architecture == "" {
return ""
}

return p.OS + "/" + p.Architecture
}

// NewNodeLLB wraps PackageNode for LLB conversion.
Expand Down Expand Up @@ -114,7 +128,7 @@ func (node *NodeLLB) convertPlatform(platform string) (v1.Platform, error) {

func (node *NodeLLB) convertDependency(dep solver.PackageDependency) (depState llb.State, srcName string, err error) {
if dep.IsInternal() {
depState, err = NewNodeLLB(dep.Node, node.Graph).Build()
depState, err = NewNodeLLB(dep.Node, node.Graph).Build(node.Pkg.Platform)
if err != nil {
return llb.Scratch(), "", err
}
Expand Down Expand Up @@ -159,11 +173,18 @@ func (node *NodeLLB) dependencies(root llb.State) (llb.State, error) {
stages := []llb.State{root}

for _, dep := range deps {
if _, alreadyProcessed := seen[dep.ID()]; alreadyProcessed {
depID := dep.ID() + node.Platform.String()

// set dep platform to node platform if not set
if dep.Platform == "" {
dep.Platform = node.Platform.String()
}

if _, alreadyProcessed := seen[depID]; alreadyProcessed {
continue
}

seen[dep.ID()] = struct{}{}
seen[depID] = struct{}{}

depState, srcName, err := node.convertDependency(dep)
if err != nil {
Expand Down Expand Up @@ -328,8 +349,22 @@ func (node *NodeLLB) finalize(root llb.State) llb.State {
}

// Build converts PackageNode to buildkit LLB.
func (node *NodeLLB) Build() (llb.State, error) {
if state, ok := node.Graph.cache[node.PackageNode]; ok {
func (node *NodeLLB) Build(platform string) (llb.State, error) {
cacheSt := cacheState{
PackageNode: node.PackageNode,
}

if platform != "" {
v1Platform, err := node.convertPlatform(platform)
if err != nil {
return llb.Scratch(), err
}

node.Platform = Platform{v1Platform}
cacheSt.Platform = node.Platform.String()
}

if state, ok := node.Graph.cache[cacheSt]; ok {
return state, nil
}

Expand All @@ -349,7 +384,7 @@ func (node *NodeLLB) Build() (llb.State, error) {

root = node.finalize(root)

node.Graph.cache[node.PackageNode] = root
node.Graph.cache[cacheSt] = root

return root, nil
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
# syntax = SHEBANG

format: v1alpha2
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
---
name: final
platform: linux/arm64
steps:
- test:
- test `uname -m` = "aarch64"
finalize:
- from: /
to: /
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
---
run:
- name: docker-amd64
runner: docker
platform: linux/amd64
target: final
expect: success
- name: llb-amd64
runner: llb
platform: linux/amd64
target: final
expect: success
- name: buildkit-amd64
runner: buildkit
platform: linux/amd64
target: final
expect: success
7 changes: 7 additions & 0 deletions internal/pkg/integration/testdata/arch-amd64/final/pkg.yaml
Original file line number Diff line number Diff line change
@@ -1,5 +1,12 @@
---
name: final
dependencies:
# this arm64 variant of the image only contains the u-boot binary,
# so this is a good test to ensure that we can copy the binary from
# the arm64 image to the x86_64 image
- image: ghcr.io/siderolabs/u-boot:v1.7.0-alpha.0-31-gcb39126
platform: linux/arm64
from: /rpi_generic
steps:
- test:
- test "${BUILD:-x}" = "x86_64-linux-musl"
Expand Down
5 changes: 3 additions & 2 deletions internal/pkg/solver/graph.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,8 +26,9 @@ func (dep PackageDependency) ID() string {

// PackageNode is a Pkg with associated dependencies.
type PackageNode struct {
Pkg *v1alpha2.Pkg
Name string
Pkg *v1alpha2.Pkg
Name string
// Platform string
Dependencies []PackageDependency
}

Expand Down
6 changes: 4 additions & 2 deletions internal/pkg/solver/packages.go
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,7 @@ func (pkgs *Packages) resolve(name string, path []string, cache map[string]*Pack
node := &PackageNode{
Pkg: pkg,
Name: name,
// Platform: pkg.Platform,
}

for _, dep := range pkg.Dependencies {
Expand Down Expand Up @@ -105,8 +106,9 @@ func (pkgs *Packages) ToSet() (set PackageSet) {
}

set = append(set, &PackageNode{
Name: name,
Pkg: pkg,
Name: name,
Pkg: pkg,
// Platform: pkg.Platform,
Dependencies: dependencies,
})
}
Expand Down
1 change: 1 addition & 0 deletions internal/pkg/types/v1alpha2/pkg.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ type Pkg struct {
Shell Shell `yaml:"shell,omitempty"`
BaseDir string `yaml:"-"`
FileName string `yaml:"-"`
Platform string `yaml:"platform,omitempty"`
Install Install `yaml:"install,omitempty"`
Dependencies Dependencies `yaml:"dependencies,omitempty"`
Steps Steps `yaml:"steps,omitempty"`
Expand Down
3 changes: 1 addition & 2 deletions internal/pkg/util/testutil/integration.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,6 @@ package testutil

import (
"bytes"
"fmt"
"io"
"os"
"testing"
Expand Down Expand Up @@ -75,7 +74,7 @@ func (test IntegrationTest) patch(t *testing.T) {
t.Fatalf("error reading %q: %v", constants.Pkgfile, err)
}

contents = bytes.ReplaceAll(contents, []byte("SHEBANG"), []byte(fmt.Sprintf("%s/%s/bldr:%s", constants.DefaultRegistry, constants.DefaultOrganization, constants.Version)))
contents = bytes.ReplaceAll(contents, []byte("SHEBANG"), []byte(`127.0.0.1:5010/frezbo/bldr:v0.2.3-2-gda04a0d-dirty@sha256:2d15a4285d17832dfab103b760af61ade3023e565eae1a1308b55aa7687aa64d`))

_, err = pkgfile.Seek(0, io.SeekStart)
if err != nil {
Expand Down

0 comments on commit 5052222

Please sign in to comment.