mirror of
https://gitea.com/Lydanne/buildx.git
synced 2025-05-18 00:47:48 +08:00

Removes gogo/protobuf from buildx and updates to a version of moby/buildkit where gogo is removed. This also changes how the proto files are generated. This is because newer versions of protobuf are more strict about name conflicts. If two files have the same name (even if they are relative paths) and are used in different protoc commands, they'll conflict in the registry. Since protobuf file generation doesn't work very well with `paths=source_relative`, this removes the `go:generate` expression and just relies on the dockerfile to perform the generation. Signed-off-by: Jonathan A. Sternberg <jonathan.sternberg@docker.com>
224 lines
5.0 KiB
Go
224 lines
5.0 KiB
Go
package bake
|
|
|
|
import (
|
|
"archive/tar"
|
|
"bytes"
|
|
"context"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/docker/buildx/builder"
|
|
controllerapi "github.com/docker/buildx/controller/pb"
|
|
"github.com/docker/buildx/driver"
|
|
"github.com/docker/buildx/util/progress"
|
|
"github.com/moby/buildkit/client"
|
|
"github.com/moby/buildkit/client/llb"
|
|
"github.com/moby/buildkit/frontend/dockerui"
|
|
gwclient "github.com/moby/buildkit/frontend/gateway/client"
|
|
"github.com/moby/buildkit/session"
|
|
"github.com/pkg/errors"
|
|
"google.golang.org/protobuf/proto"
|
|
)
|
|
|
|
type Input struct {
|
|
State *llb.State
|
|
URL string
|
|
}
|
|
|
|
func ReadRemoteFiles(ctx context.Context, nodes []builder.Node, url string, names []string, pw progress.Writer) ([]File, *Input, error) {
|
|
var sessions []session.Attachable
|
|
var filename string
|
|
|
|
st, ok := dockerui.DetectGitContext(url, false)
|
|
if ok {
|
|
if ssh, err := controllerapi.CreateSSH([]*controllerapi.SSH{{
|
|
ID: "default",
|
|
Paths: strings.Split(os.Getenv("BUILDX_BAKE_GIT_SSH"), ","),
|
|
}}); err == nil {
|
|
sessions = append(sessions, ssh)
|
|
}
|
|
var gitAuthSecrets []*controllerapi.Secret
|
|
if _, ok := os.LookupEnv("BUILDX_BAKE_GIT_AUTH_TOKEN"); ok {
|
|
gitAuthSecrets = append(gitAuthSecrets, &controllerapi.Secret{
|
|
ID: llb.GitAuthTokenKey,
|
|
Env: "BUILDX_BAKE_GIT_AUTH_TOKEN",
|
|
})
|
|
}
|
|
if _, ok := os.LookupEnv("BUILDX_BAKE_GIT_AUTH_HEADER"); ok {
|
|
gitAuthSecrets = append(gitAuthSecrets, &controllerapi.Secret{
|
|
ID: llb.GitAuthHeaderKey,
|
|
Env: "BUILDX_BAKE_GIT_AUTH_HEADER",
|
|
})
|
|
}
|
|
if len(gitAuthSecrets) > 0 {
|
|
if secrets, err := controllerapi.CreateSecrets(gitAuthSecrets); err == nil {
|
|
sessions = append(sessions, secrets)
|
|
}
|
|
}
|
|
} else {
|
|
st, filename, ok = dockerui.DetectHTTPContext(url)
|
|
if !ok {
|
|
return nil, nil, errors.Errorf("not url context")
|
|
}
|
|
}
|
|
|
|
inp := &Input{State: st, URL: url}
|
|
var files []File
|
|
|
|
var node *builder.Node
|
|
for i, n := range nodes {
|
|
if n.Err == nil {
|
|
node = &nodes[i]
|
|
continue
|
|
}
|
|
}
|
|
if node == nil {
|
|
return nil, nil, nil
|
|
}
|
|
|
|
c, err := driver.Boot(ctx, ctx, node.Driver, pw)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
ch, done := progress.NewChannel(pw)
|
|
defer func() { <-done }()
|
|
_, err = c.Build(ctx, client.SolveOpt{Session: sessions, Internal: true}, "buildx", func(ctx context.Context, c gwclient.Client) (*gwclient.Result, error) {
|
|
def, err := st.Marshal(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
res, err := c.Solve(ctx, gwclient.SolveRequest{
|
|
Definition: def.ToPB(),
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ref, err := res.SingleRef()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if filename != "" {
|
|
files, err = filesFromURLRef(ctx, c, ref, inp, filename, names)
|
|
} else {
|
|
files, err = filesFromRef(ctx, ref, names)
|
|
}
|
|
return nil, err
|
|
}, ch)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
return files, inp, nil
|
|
}
|
|
|
|
func isArchive(header []byte) bool {
|
|
for _, m := range [][]byte{
|
|
{0x42, 0x5A, 0x68}, // bzip2
|
|
{0x1F, 0x8B, 0x08}, // gzip
|
|
{0xFD, 0x37, 0x7A, 0x58, 0x5A, 0x00}, // xz
|
|
} {
|
|
if len(header) < len(m) {
|
|
continue
|
|
}
|
|
if bytes.Equal(m, header[:len(m)]) {
|
|
return true
|
|
}
|
|
}
|
|
|
|
r := tar.NewReader(bytes.NewBuffer(header))
|
|
_, err := r.Next()
|
|
return err == nil
|
|
}
|
|
|
|
func filesFromURLRef(ctx context.Context, c gwclient.Client, ref gwclient.Reference, inp *Input, filename string, names []string) ([]File, error) {
|
|
stat, err := ref.StatFile(ctx, gwclient.StatRequest{Path: filename})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
dt, err := ref.ReadFile(ctx, gwclient.ReadRequest{
|
|
Filename: filename,
|
|
Range: &gwclient.FileRange{
|
|
Length: 1024,
|
|
},
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if isArchive(dt) {
|
|
bc := llb.Scratch().File(llb.Copy(inp.State, filename, "/", &llb.CopyInfo{
|
|
AttemptUnpack: true,
|
|
}))
|
|
inp.State = &bc
|
|
inp.URL = ""
|
|
def, err := bc.Marshal(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
res, err := c.Solve(ctx, gwclient.SolveRequest{
|
|
Definition: def.ToPB(),
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ref, err := res.SingleRef()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return filesFromRef(ctx, ref, names)
|
|
}
|
|
|
|
inp.State = nil
|
|
name := inp.URL
|
|
inp.URL = ""
|
|
|
|
if len(dt) > proto.Size(stat) {
|
|
if proto.Size(stat) > 1024*512 {
|
|
return nil, errors.Errorf("non-archive definition URL bigger than maximum allowed size")
|
|
}
|
|
|
|
dt, err = ref.ReadFile(ctx, gwclient.ReadRequest{
|
|
Filename: filename,
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
return []File{{Name: name, Data: dt}}, nil
|
|
}
|
|
|
|
func filesFromRef(ctx context.Context, ref gwclient.Reference, names []string) ([]File, error) {
|
|
// TODO: auto-remove parent dir in needed
|
|
var files []File
|
|
|
|
isDefault := false
|
|
if len(names) == 0 {
|
|
isDefault = true
|
|
names = defaultFilenames()
|
|
}
|
|
|
|
for _, name := range names {
|
|
_, err := ref.StatFile(ctx, gwclient.StatRequest{Path: name})
|
|
if err != nil {
|
|
if isDefault {
|
|
continue
|
|
}
|
|
return nil, err
|
|
}
|
|
dt, err := ref.ReadFile(ctx, gwclient.ReadRequest{Filename: name})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
files = append(files, File{Name: name, Data: dt})
|
|
}
|
|
|
|
return files, nil
|
|
}
|