|
|
|
package llb
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/moby/buildkit/solver/pb"
|
|
|
|
digest "github.com/opencontainers/go-digest"
|
|
|
|
ocispecs "github.com/opencontainers/image-spec/specs-go/v1"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
)
|
|
|
|
|
|
|
|
// DefinitionOp implements llb.Vertex using a marshalled definition.
|
|
|
|
//
|
|
|
|
// For example, after marshalling a LLB state and sending over the wire, the
|
|
|
|
// LLB state can be reconstructed from the definition.
|
|
|
|
type DefinitionOp struct {
|
|
|
|
MarshalCache
|
|
|
|
mu sync.Mutex
|
|
|
|
ops map[digest.Digest]*pb.Op
|
|
|
|
defs map[digest.Digest][]byte
|
|
|
|
metas map[digest.Digest]pb.OpMetadata
|
|
|
|
sources map[digest.Digest][]*SourceLocation
|
|
|
|
platforms map[digest.Digest]*ocispecs.Platform
|
|
|
|
dgst digest.Digest
|
|
|
|
index pb.OutputIndex
|
|
|
|
inputCache map[digest.Digest][]*DefinitionOp
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewDefinitionOp returns a new operation from a marshalled definition.
|
|
|
|
func NewDefinitionOp(def *pb.Definition) (*DefinitionOp, error) {
|
|
|
|
if def == nil {
|
|
|
|
return nil, errors.New("invalid nil input definition to definition op")
|
|
|
|
}
|
|
|
|
|
|
|
|
ops := make(map[digest.Digest]*pb.Op)
|
|
|
|
defs := make(map[digest.Digest][]byte)
|
|
|
|
platforms := make(map[digest.Digest]*ocispecs.Platform)
|
|
|
|
|
|
|
|
var dgst digest.Digest
|
|
|
|
for _, dt := range def.Def {
|
|
|
|
var op pb.Op
|
|
|
|
if err := (&op).Unmarshal(dt); err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to parse llb proto op")
|
|
|
|
}
|
|
|
|
dgst = digest.FromBytes(dt)
|
|
|
|
ops[dgst] = &op
|
|
|
|
defs[dgst] = dt
|
|
|
|
|
|
|
|
var platform *ocispecs.Platform
|
|
|
|
if op.Platform != nil {
|
|
|
|
spec := op.Platform.Spec()
|
|
|
|
platform = &spec
|
|
|
|
}
|
|
|
|
platforms[dgst] = platform
|
|
|
|
}
|
|
|
|
|
|
|
|
srcs := map[digest.Digest][]*SourceLocation{}
|
|
|
|
|
|
|
|
if def.Source != nil {
|
|
|
|
sourceMaps := make([]*SourceMap, len(def.Source.Infos))
|
|
|
|
for i, info := range def.Source.Infos {
|
|
|
|
var st *State
|
|
|
|
sdef := info.Definition
|
|
|
|
if sdef != nil {
|
|
|
|
op, err := NewDefinitionOp(sdef)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
state := NewState(op)
|
|
|
|
st = &state
|
|
|
|
}
|
|
|
|
sourceMaps[i] = NewSourceMap(st, info.Filename, info.Data)
|
|
|
|
}
|
|
|
|
|
|
|
|
for dgst, locs := range def.Source.Locations {
|
|
|
|
for _, loc := range locs.Locations {
|
|
|
|
if loc.SourceIndex < 0 || int(loc.SourceIndex) >= len(sourceMaps) {
|
|
|
|
return nil, errors.Errorf("failed to find source map with index %d", loc.SourceIndex)
|
|
|
|
}
|
|
|
|
|
|
|
|
srcs[digest.Digest(dgst)] = append(srcs[digest.Digest(dgst)], &SourceLocation{
|
|
|
|
SourceMap: sourceMaps[int(loc.SourceIndex)],
|
|
|
|
Ranges: loc.Ranges,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var index pb.OutputIndex
|
|
|
|
if dgst != "" {
|
|
|
|
index = ops[dgst].Inputs[0].Index
|
|
|
|
dgst = ops[dgst].Inputs[0].Digest
|
|
|
|
}
|
|
|
|
|
|
|
|
return &DefinitionOp{
|
|
|
|
ops: ops,
|
|
|
|
defs: defs,
|
|
|
|
metas: def.Metadata,
|
|
|
|
sources: srcs,
|
|
|
|
platforms: platforms,
|
|
|
|
dgst: dgst,
|
|
|
|
index: index,
|
|
|
|
inputCache: make(map[digest.Digest][]*DefinitionOp),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DefinitionOp) ToInput(ctx context.Context, c *Constraints) (*pb.Input, error) {
|
|
|
|
return d.Output().ToInput(ctx, c)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DefinitionOp) Vertex(context.Context, *Constraints) Vertex {
|
|
|
|
return d
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DefinitionOp) Validate(context.Context, *Constraints) error {
|
|
|
|
// Scratch state has no digest, ops or metas.
|
|
|
|
if d.dgst == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
d.mu.Lock()
|
|
|
|
defer d.mu.Unlock()
|
|
|
|
|
|
|
|
if len(d.ops) == 0 || len(d.defs) == 0 || len(d.metas) == 0 {
|
|
|
|
return errors.Errorf("invalid definition op with no ops %d %d", len(d.ops), len(d.metas))
|
|
|
|
}
|
|
|
|
|
|
|
|
_, ok := d.ops[d.dgst]
|
|
|
|
if !ok {
|
|
|
|
return errors.Errorf("invalid definition op with unknown op %q", d.dgst)
|
|
|
|
}
|
|
|
|
|
|
|
|
_, ok = d.defs[d.dgst]
|
|
|
|
if !ok {
|
|
|
|
return errors.Errorf("invalid definition op with unknown def %q", d.dgst)
|
|
|
|
}
|
|
|
|
|
|
|
|
_, ok = d.metas[d.dgst]
|
|
|
|
if !ok {
|
|
|
|
return errors.Errorf("invalid definition op with unknown metas %q", d.dgst)
|
|
|
|
}
|
|
|
|
|
|
|
|
// It is possible for d.index >= len(d.ops[d.dgst]) when depending on scratch
|
|
|
|
// images.
|
|
|
|
if d.index < 0 {
|
|
|
|
return errors.Errorf("invalid definition op with invalid index")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DefinitionOp) Marshal(ctx context.Context, c *Constraints) (digest.Digest, []byte, *pb.OpMetadata, []*SourceLocation, error) {
|
|
|
|
if d.dgst == "" {
|
|
|
|
return "", nil, nil, nil, errors.Errorf("cannot marshal empty definition op")
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := d.Validate(ctx, c); err != nil {
|
|
|
|
return "", nil, nil, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
d.mu.Lock()
|
|
|
|
defer d.mu.Unlock()
|
|
|
|
|
|
|
|
meta := d.metas[d.dgst]
|
|
|
|
return d.dgst, d.defs[d.dgst], &meta, d.sources[d.dgst], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DefinitionOp) Output() Output {
|
|
|
|
if d.dgst == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
d.mu.Lock()
|
|
|
|
platform := d.platforms[d.dgst]
|
|
|
|
d.mu.Unlock()
|
|
|
|
|
|
|
|
return &output{vertex: d, platform: platform, getIndex: func() (pb.OutputIndex, error) {
|
|
|
|
return d.index, nil
|
|
|
|
}}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DefinitionOp) Inputs() []Output {
|
|
|
|
if d.dgst == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var inputs []Output
|
|
|
|
|
|
|
|
d.mu.Lock()
|
|
|
|
op := d.ops[d.dgst]
|
|
|
|
platform := d.platforms[d.dgst]
|
|
|
|
d.mu.Unlock()
|
|
|
|
|
|
|
|
for _, input := range op.Inputs {
|
|
|
|
var vtx *DefinitionOp
|
|
|
|
d.mu.Lock()
|
|
|
|
if existingIndexes, ok := d.inputCache[input.Digest]; ok {
|
|
|
|
if int(input.Index) < len(existingIndexes) && existingIndexes[input.Index] != nil {
|
|
|
|
vtx = existingIndexes[input.Index]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if vtx == nil {
|
|
|
|
vtx = &DefinitionOp{
|
|
|
|
ops: d.ops,
|
|
|
|
defs: d.defs,
|
|
|
|
metas: d.metas,
|
|
|
|
platforms: d.platforms,
|
|
|
|
dgst: input.Digest,
|
|
|
|
index: input.Index,
|
|
|
|
inputCache: d.inputCache,
|
|
|
|
sources: d.sources,
|
|
|
|
}
|
|
|
|
existingIndexes := d.inputCache[input.Digest]
|
|
|
|
indexDiff := int(input.Index) - len(existingIndexes)
|
|
|
|
if indexDiff >= 0 {
|
|
|
|
// make room in the slice for the new index being set
|
|
|
|
existingIndexes = append(existingIndexes, make([]*DefinitionOp, indexDiff+1)...)
|
|
|
|
}
|
|
|
|
existingIndexes[input.Index] = vtx
|
|
|
|
d.inputCache[input.Digest] = existingIndexes
|
|
|
|
}
|
|
|
|
d.mu.Unlock()
|
|
|
|
|
|
|
|
inputs = append(inputs, &output{vertex: vtx, platform: platform, getIndex: func() (pb.OutputIndex, error) {
|
|
|
|
return pb.OutputIndex(vtx.index), nil
|
|
|
|
}})
|
|
|
|
}
|
|
|
|
|
|
|
|
return inputs
|
|
|
|
}
|