Skip to content

Commit

Permalink
llbsolver: support pinning sources
Browse files Browse the repository at this point in the history
Alternative to PR 2816 ("dockerfile: support Dockerfile.pin for pinning sources")

This version is implemented on the llbsolver side and agnostic to the LLB frontends.
See `solver/llbsolver/ops/source.go`.

See `docs/build-repro.md` for the usage.

Signed-off-by: Akihiro Suda <akihiro.suda.cz@hco.ntt.co.jp>
  • Loading branch information
AkihiroSuda committed Jul 5, 2022
1 parent b57ada1 commit 748c5e5
Show file tree
Hide file tree
Showing 14 changed files with 1,098 additions and 148 deletions.
879 changes: 735 additions & 144 deletions api/services/control/control.pb.go

Large diffs are not rendered by default.

12 changes: 12 additions & 0 deletions api/services/control/control.proto
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,7 @@ message SolveRequest {
CacheOptions Cache = 8 [(gogoproto.nullable) = false];
repeated string Entitlements = 9 [(gogoproto.customtype) = "github.com/moby/buildkit/util/entitlements.Entitlement" ];
map<string, pb.Definition> FrontendInputs = 10;
Pin Pin = 11;
}

message CacheOptions {
Expand Down Expand Up @@ -163,3 +164,14 @@ message InfoRequest {}
message InfoResponse {
moby.buildkit.v1.types.BuildkitVersion buildkitVersion = 1;
}

message Pin {
repeated BuildInfoSource Sources = 1;
}

message BuildInfoSource {
string Type = 1;
string Ref = 2;
string Alias = 3;
string Pin = 4;
}
82 changes: 82 additions & 0 deletions client/client_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ import (
binfotypes "github.com/moby/buildkit/util/buildinfo/types"
"github.com/moby/buildkit/util/contentutil"
"github.com/moby/buildkit/util/entitlements"
pintypes "github.com/moby/buildkit/util/pin/types"
"github.com/moby/buildkit/util/testutil"
"github.com/moby/buildkit/util/testutil/echoserver"
"github.com/moby/buildkit/util/testutil/httpserver"
Expand Down Expand Up @@ -166,6 +167,7 @@ func TestIntegration(t *testing.T) {
testCallInfo,
testPullWithLayerLimit,
testExportAnnotations,
testPin,
)
tests = append(tests, diffOpTestCases()...)
integration.Run(t, tests, mirrors)
Expand Down Expand Up @@ -6287,3 +6289,83 @@ func fixedWriteCloser(wc io.WriteCloser) func(map[string]string) (io.WriteCloser
return wc, nil
}
}

func testPin(t *testing.T, sb integration.Sandbox) {
requiresLinux(t)
c, err := New(sb.Context(), sb.Address())
require.NoError(t, err)
defer c.Close()

frontend := func(ctx context.Context, c gateway.Client) (*gateway.Result, error) {
st := llb.Image("busybox:1.34.1-uclibc").File(
llb.Copy(llb.HTTP("https://raw.githubusercontent.com/moby/buildkit/v0.10.1/README.md"),
"README.md", "README.md"))
def, err := st.Marshal(sb.Context())
if err != nil {
return nil, err
}
return c.Solve(ctx, gateway.SolveRequest{
Definition: def.ToPB(),
})
}

type testCase struct {
pin *pintypes.Pin
expectedErr string
}
testCases := []testCase{
{
// Valid
pin: &pintypes.Pin{
Sources: []pintypes.Source{
{
Type: "docker-image",
Ref: "docker.io/library/busybox:1.34.1-uclibc",
Pin: "sha256:3614ca5eacf0a3a1bcc361c939202a974b4902b9334ff36eb29ffe9011aaad83",
},
{
Type: "http",
Ref: "https://raw.githubusercontent.com/moby/buildkit/v0.10.1/README.md",
Pin: "sha256:6e4b94fc270e708e1068be28bd3551dc6917a4fc5a61293d51bb36e6b75c4b53",
},
},
},
expectedErr: "",
},
{
// Invalid docker-image
pin: &pintypes.Pin{
Sources: []pintypes.Source{
{
Type: "docker-image",
Ref: "docker.io/library/busybox:1.34.1-uclibc",
Pin: "sha256:aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa", // invalid
},
},
},
expectedErr: "docker.io/library/busybox:1.34.1-uclibc@sha256:aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa: not found",
},
{
// Invalid docker-image
pin: &pintypes.Pin{
Sources: []pintypes.Source{
{
Type: "http",
Ref: "https://raw.githubusercontent.com/moby/buildkit/v0.10.1/README.md",
Pin: "sha256:bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb", // invalid
},
},
},
expectedErr: "digest mismatch sha256:6e4b94fc270e708e1068be28bd3551dc6917a4fc5a61293d51bb36e6b75c4b53: sha256:bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb",
},
}
for _, tc := range testCases {
_, err = c.Build(sb.Context(), SolveOpt{Pin: tc.pin}, "", frontend, nil)
if tc.expectedErr == "" {
require.NoError(t, err)
} else {
require.Error(t, err)
require.Contains(t, err.Error(), tc.expectedErr)
}
}
}
16 changes: 16 additions & 0 deletions client/solve.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ import (
"github.com/moby/buildkit/solver/pb"
"github.com/moby/buildkit/util/bklog"
"github.com/moby/buildkit/util/entitlements"
pintypes "github.com/moby/buildkit/util/pin/types"
ocispecs "github.com/opencontainers/image-spec/specs-go/v1"
"github.com/pkg/errors"
"github.com/tonistiigi/fsutil"
Expand All @@ -44,6 +45,7 @@ type SolveOpt struct {
AllowedEntitlements []entitlements.Entitlement
SharedSession *session.Session // TODO: refactor to better session syncing
SessionPreInitialized bool // TODO: refactor to better session syncing
Pin *pintypes.Pin
}

type ExportEntry struct {
Expand Down Expand Up @@ -198,6 +200,19 @@ func (c *Client) solve(ctx context.Context, def *llb.Definition, runGateway runG
opt.FrontendAttrs[k] = v
}

var pin *controlapi.Pin
if opt.Pin != nil && len(opt.Pin.Sources) > 0 {
pin = &controlapi.Pin{}
for _, f := range opt.Pin.Sources {
pin.Sources = append(pin.Sources, &controlapi.BuildInfoSource{
Type: string(f.Type),
Alias: f.Alias,
Ref: f.Ref,
Pin: f.Pin,
})
}
}

solveCtx, cancelSolve := context.WithCancel(ctx)
var res *SolveResponse
eg.Go(func() error {
Expand Down Expand Up @@ -237,6 +252,7 @@ func (c *Client) solve(ctx context.Context, def *llb.Definition, runGateway runG
FrontendInputs: frontendInputs,
Cache: cacheOpt.options,
Entitlements: opt.AllowedEntitlements,
Pin: pin,
})
if err != nil {
return errors.Wrap(err, "failed to solve")
Expand Down
19 changes: 19 additions & 0 deletions cmd/buildctl/build.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ import (
"github.com/moby/buildkit/session/auth/authprovider"
"github.com/moby/buildkit/session/sshforward/sshprovider"
"github.com/moby/buildkit/solver/pb"
pintypes "github.com/moby/buildkit/util/pin/types"
"github.com/moby/buildkit/util/progress/progresswriter"
digest "github.com/opencontainers/go-digest"
"github.com/pkg/errors"
Expand Down Expand Up @@ -88,6 +89,10 @@ var buildCommand = cli.Command{
Name: "metadata-file",
Usage: "Output build metadata (e.g., image digest) to a file as JSON",
},
cli.StringFlag{
Name: "pin-file",
Usage: "Read pinning file from a JSON file",
},
},
}

Expand Down Expand Up @@ -182,6 +187,19 @@ func buildAction(clicontext *cli.Context) error {
return err
}

var pin *pintypes.Pin
if pinFile := clicontext.String("pin-file"); pinFile != "" {
b, err := os.ReadFile(pinFile)
if err != nil {
return err
}
var pinStruct pintypes.Pin
if err := json.Unmarshal(b, &pinStruct); err != nil {
return errors.Wrapf(err, "failed to unmarshal pin-file %q", pinFile)
}
pin = &pinStruct
}

eg, ctx := errgroup.WithContext(bccommon.CommandContext(clicontext))

solveOpt := client.SolveOpt{
Expand All @@ -194,6 +212,7 @@ func buildAction(clicontext *cli.Context) error {
CacheImports: cacheImports,
Session: attachable,
AllowedEntitlements: allowed,
Pin: pin,
}

solveOpt.FrontendAttrs, err = build.ParseOpt(clicontext.StringSlice("opt"))
Expand Down
17 changes: 16 additions & 1 deletion control/control.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,9 @@ import (
"github.com/moby/buildkit/solver/llbsolver"
"github.com/moby/buildkit/solver/pb"
"github.com/moby/buildkit/util/bklog"
binfotypes "github.com/moby/buildkit/util/buildinfo/types"
"github.com/moby/buildkit/util/imageutil"
pintypes "github.com/moby/buildkit/util/pin/types"
"github.com/moby/buildkit/util/throttle"
"github.com/moby/buildkit/util/tracing/transform"
"github.com/moby/buildkit/version"
Expand Down Expand Up @@ -306,6 +308,19 @@ func (c *Controller) Solve(ctx context.Context, req *controlapi.SolveRequest) (*
Attrs: im.Attrs,
})
}
var pin *pintypes.Pin
if req.Pin != nil {
pin = &pintypes.Pin{}
for _, f := range req.Pin.Sources {
s := pintypes.Source{
Type: binfotypes.SourceType(f.Type),
Ref: f.Ref,
Alias: f.Alias,
Pin: f.Pin,
}
pin.Sources = append(pin.Sources, s)
}
}

resp, err := c.solver.Solve(ctx, req.Ref, req.Session, frontend.SolveRequest{
Frontend: req.Frontend,
Expand All @@ -317,7 +332,7 @@ func (c *Controller) Solve(ctx context.Context, req *controlapi.SolveRequest) (*
Exporter: expi,
CacheExporter: cacheExporter,
CacheExportMode: cacheExportMode,
}, req.Entitlements)
}, req.Entitlements, pin)
if err != nil {
return nil, err
}
Expand Down
33 changes: 33 additions & 0 deletions docs/build-repro.md
Original file line number Diff line number Diff line change
Expand Up @@ -127,3 +127,36 @@ jq '.' metadata.json
"containerimage.digest": "sha256:..."
}
```

### Reproducing the pinned dependencies

<!-- TODO: s/master/v0.11/ after the release -->
Reproducing the pinned dependencies is supported in the master branch of BuildKit.

e.g.,
```bash
buildctl build --frontend dockerfile.v0 --local dockerfile=. --local context=. --pin-file Dockerfile.pin
```

The content of the `Dockerfile.pin` is a subset of the `."containerimage.buildinfo".sources` property of the `metadata.json`:
```json
{
"sources": [
{
"type": "docker-image",
"ref": "docker.io/library/alpine:latest",
"pin": "sha256:4edbd2beb5f78b1014028f4fbb99f3237d9561100b6881aabbf5acce2c4f9454"
},
{
"type": "http",
"ref": "https://raw.githubusercontent.com/moby/buildkit/v0.10.1/README.md",
"pin": "sha256:6e4b94fc270e708e1068be28bd3551dc6917a4fc5a61293d51bb36e6b75c4b53"
}
]
}
```

Reproduction is currently supported for the following source types:
* `docker-image`
* `http`
<!-- TODO: git (waiting for https://github.com/moby/buildkit/pull/2799 to be merged for git ref parser) >
8 changes: 7 additions & 1 deletion solver/llbsolver/bridge.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ import (
"github.com/moby/buildkit/util/bklog"
"github.com/moby/buildkit/util/buildinfo"
"github.com/moby/buildkit/util/flightcontrol"
"github.com/moby/buildkit/util/pin/pinner"
"github.com/moby/buildkit/util/progress"
"github.com/moby/buildkit/worker"
digest "github.com/opencontainers/go-digest"
Expand Down Expand Up @@ -73,6 +74,11 @@ func (b *llbBridge) loadResult(ctx context.Context, def *pb.Definition, cacheImp
if err != nil {
return nil, nil, err
}
pin, err := loadPin(b.builder)
if err != nil {
return nil, nil, err
}
pinner := pinner.New(pin)
var cms []solver.CacheManager
for _, im := range cacheImports {
cmID, err := cmKey(im)
Expand Down Expand Up @@ -112,7 +118,7 @@ func (b *llbBridge) loadResult(ctx context.Context, def *pb.Definition, cacheImp
}
dpc := &detectPrunedCacheID{}

edge, err := Load(def, dpc.Load, ValidateEntitlements(ent), WithCacheSources(cms), NormalizeRuntimePlatforms(), WithValidateCaps())
edge, err := Load(def, dpc.Load, ValidateEntitlements(ent), WithCacheSources(cms), NormalizeRuntimePlatforms(), WithValidateCaps(), WithPinner(pinner))
if err != nil {
return nil, nil, errors.Wrap(err, "failed to load LLB")
}
Expand Down
5 changes: 5 additions & 0 deletions solver/llbsolver/ops/source.go
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,11 @@ func (s *sourceOp) instance(ctx context.Context) (source.SourceInstance, error)
if err != nil {
return nil, err
}
if pinFunc := s.vtx.Options().PinFunc; pinFunc != nil {
if err = pinFunc(ctx, id); err != nil {
return nil, err
}
}
src, err := s.sm.Resolve(ctx, id, s.sessM, s.vtx)
if err != nil {
return nil, err
Expand Down
37 changes: 35 additions & 2 deletions solver/llbsolver/solver.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,14 +22,18 @@ import (
"github.com/moby/buildkit/util/buildinfo"
"github.com/moby/buildkit/util/compression"
"github.com/moby/buildkit/util/entitlements"
pintypes "github.com/moby/buildkit/util/pin/types"
"github.com/moby/buildkit/util/progress"
"github.com/moby/buildkit/worker"
digest "github.com/opencontainers/go-digest"
"github.com/pkg/errors"
"golang.org/x/sync/errgroup"
)

const keyEntitlements = "llb.entitlements"
const (
keyEntitlements = "llb.entitlements"
keyPin = "llb.pin"
)

type ExporterRequest struct {
Exporter exporter.ExporterInstance
Expand Down Expand Up @@ -104,7 +108,7 @@ func (s *Solver) Bridge(b solver.Builder) frontend.FrontendLLBBridge {
}
}

func (s *Solver) Solve(ctx context.Context, id string, sessionID string, req frontend.SolveRequest, exp ExporterRequest, ent []entitlements.Entitlement) (*client.SolveResponse, error) {
func (s *Solver) Solve(ctx context.Context, id string, sessionID string, req frontend.SolveRequest, exp ExporterRequest, ent []entitlements.Entitlement, pin *pintypes.Pin) (*client.SolveResponse, error) {
j, err := s.solver.NewJob(id)
if err != nil {
return nil, err
Expand All @@ -118,6 +122,10 @@ func (s *Solver) Solve(ctx context.Context, id string, sessionID string, req fro
}
j.SetValue(keyEntitlements, set)

if pin != nil {
j.SetValue(keyPin, *pin)
}

j.SessionID = sessionID

var res *frontend.Result
Expand Down Expand Up @@ -508,3 +516,28 @@ func loadEntitlements(b solver.Builder) (entitlements.Set, error) {
}
return ent, nil
}

func loadPin(b solver.Builder) (*pintypes.Pin, error) {
set := make(map[pintypes.Source]struct{}, 0)
err := b.EachValue(context.TODO(), keyPin, func(v interface{}) error {
x, ok := v.(pintypes.Pin)
if !ok {
return errors.Errorf("invalid pin %T", v)
}
for _, f := range x.Sources {
set[f] = struct{}{}
}
return nil
})
if err != nil {
return nil, err
}
var pin *pintypes.Pin
if len(set) > 0 {
pin = &pintypes.Pin{}
for k := range set {
pin.Sources = append(pin.Sources, k)
}
}
return pin, nil
}
Loading

0 comments on commit 748c5e5

Please sign in to comment.