mirror of
https://github.com/goreleaser/goreleaser.git
synced 2025-01-10 03:47:03 +02:00
f61d8c820c
* fix: improve output a bit Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: improve output a bit Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: revert unwanted changes Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip err Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: tests Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: test Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * chore: build Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
278 lines
7.7 KiB
Go
278 lines
7.7 KiB
Go
package docker
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/apex/log"
|
|
"github.com/goreleaser/goreleaser/internal/artifact"
|
|
"github.com/goreleaser/goreleaser/internal/deprecate"
|
|
"github.com/goreleaser/goreleaser/internal/pipe"
|
|
"github.com/goreleaser/goreleaser/internal/semerrgroup"
|
|
"github.com/goreleaser/goreleaser/internal/tmpl"
|
|
"github.com/goreleaser/goreleaser/pkg/config"
|
|
"github.com/goreleaser/goreleaser/pkg/context"
|
|
)
|
|
|
|
// ErrNoDocker is shown when docker cannot be found in $PATH.
|
|
var ErrNoDocker = errors.New("docker not present in $PATH")
|
|
|
|
// Pipe for docker.
|
|
type Pipe struct{}
|
|
|
|
func (Pipe) String() string {
|
|
return "docker images"
|
|
}
|
|
|
|
// Default sets the pipe defaults.
|
|
func (Pipe) Default(ctx *context.Context) error {
|
|
for i := range ctx.Config.Dockers {
|
|
docker := &ctx.Config.Dockers[i]
|
|
|
|
if docker.Goos == "" {
|
|
docker.Goos = "linux"
|
|
}
|
|
if docker.Goarch == "" {
|
|
docker.Goarch = "amd64"
|
|
}
|
|
if docker.Dockerfile == "" {
|
|
docker.Dockerfile = "Dockerfile"
|
|
}
|
|
if len(docker.Binaries) > 0 {
|
|
deprecate.Notice(ctx, "docker.binaries")
|
|
}
|
|
if len(docker.Builds) > 0 {
|
|
deprecate.Notice(ctx, "docker.builds")
|
|
docker.IDs = append(docker.IDs, docker.Builds...)
|
|
}
|
|
for _, f := range docker.Files {
|
|
if f == "." || strings.HasPrefix(f, ctx.Config.Dist) {
|
|
return fmt.Errorf("invalid docker.files: can't be . or inside dist folder: %s", f)
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Run the pipe.
|
|
func (Pipe) Run(ctx *context.Context) error {
|
|
if len(ctx.Config.Dockers) == 0 || len(ctx.Config.Dockers[0].ImageTemplates) == 0 {
|
|
return pipe.ErrSkipDisabledPipe
|
|
}
|
|
_, err := exec.LookPath("docker")
|
|
if err != nil {
|
|
return ErrNoDocker
|
|
}
|
|
return doRun(ctx)
|
|
}
|
|
|
|
// Publish the docker images.
|
|
func (Pipe) Publish(ctx *context.Context) error {
|
|
if ctx.SkipPublish {
|
|
return pipe.ErrSkipPublishEnabled
|
|
}
|
|
images := ctx.Artifacts.Filter(artifact.ByType(artifact.PublishableDockerImage)).List()
|
|
for _, image := range images {
|
|
if err := dockerPush(ctx, image); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func doRun(ctx *context.Context) error {
|
|
g := semerrgroup.NewSkipAware(semerrgroup.New(ctx.Parallelism))
|
|
for _, docker := range ctx.Config.Dockers {
|
|
docker := docker
|
|
g.Go(func() error {
|
|
log.WithField("docker", docker).Debug("looking for artifacts matching")
|
|
filters := []artifact.Filter{
|
|
artifact.ByGoos(docker.Goos),
|
|
artifact.ByGoarch(docker.Goarch),
|
|
artifact.ByGoarm(docker.Goarm),
|
|
artifact.Or(
|
|
artifact.ByType(artifact.Binary),
|
|
artifact.ByType(artifact.LinuxPackage),
|
|
),
|
|
}
|
|
if len(docker.IDs) > 0 {
|
|
filters = append(filters, artifact.ByIDs(docker.IDs...))
|
|
}
|
|
artifacts := ctx.Artifacts.Filter(artifact.And(filters...))
|
|
log.WithField("artifacts", artifacts.Paths()).Debug("found artifacts")
|
|
return process(ctx, docker, artifacts.List())
|
|
})
|
|
}
|
|
return g.Wait()
|
|
}
|
|
|
|
func process(ctx *context.Context, docker config.Docker, artifacts []*artifact.Artifact) error {
|
|
tmp, err := ioutil.TempDir(ctx.Config.Dist, "goreleaserdocker")
|
|
if err != nil {
|
|
return fmt.Errorf("failed to create temporary dir: %w", err)
|
|
}
|
|
log.Debug("tempdir: " + tmp)
|
|
|
|
images, err := processImageTemplates(ctx, docker)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := os.Link(docker.Dockerfile, filepath.Join(tmp, "Dockerfile")); err != nil {
|
|
return fmt.Errorf("failed to link dockerfile: %w", err)
|
|
}
|
|
for _, file := range docker.Files {
|
|
if err := os.MkdirAll(filepath.Join(tmp, filepath.Dir(file)), 0o755); err != nil {
|
|
return fmt.Errorf("failed to link extra file '%s': %w", file, err)
|
|
}
|
|
if err := link(file, filepath.Join(tmp, file)); err != nil {
|
|
return fmt.Errorf("failed to link extra file '%s': %w", file, err)
|
|
}
|
|
}
|
|
for _, art := range artifacts {
|
|
if err := os.Link(art.Path, filepath.Join(tmp, filepath.Base(art.Path))); err != nil {
|
|
return fmt.Errorf("failed to link artifact: %w", err)
|
|
}
|
|
}
|
|
|
|
buildFlags, err := processBuildFlagTemplates(ctx, docker)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := dockerBuild(ctx, tmp, images, buildFlags, docker.Buildx); err != nil {
|
|
return err
|
|
}
|
|
|
|
if strings.TrimSpace(docker.SkipPush) == "true" {
|
|
return pipe.Skip("docker.skip_push is set")
|
|
}
|
|
if ctx.SkipPublish {
|
|
return pipe.ErrSkipPublishEnabled
|
|
}
|
|
if strings.TrimSpace(docker.SkipPush) == "auto" && ctx.Semver.Prerelease != "" {
|
|
return pipe.Skip("prerelease detected with 'auto' push, skipping docker publish")
|
|
}
|
|
for _, img := range images {
|
|
ctx.Artifacts.Add(&artifact.Artifact{
|
|
Type: artifact.PublishableDockerImage,
|
|
Name: img,
|
|
Path: img,
|
|
Goarch: docker.Goarch,
|
|
Goos: docker.Goos,
|
|
Goarm: docker.Goarm,
|
|
})
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func processImageTemplates(ctx *context.Context, docker config.Docker) ([]string, error) {
|
|
// nolint:prealloc
|
|
var images []string
|
|
for _, imageTemplate := range docker.ImageTemplates {
|
|
image, err := tmpl.New(ctx).Apply(imageTemplate)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to execute image template '%s': %w", imageTemplate, err)
|
|
}
|
|
if image == "" {
|
|
continue
|
|
}
|
|
|
|
images = append(images, image)
|
|
}
|
|
|
|
if len(images) == 0 {
|
|
return images, errors.New("no image templates found")
|
|
}
|
|
|
|
return images, nil
|
|
}
|
|
|
|
func processBuildFlagTemplates(ctx *context.Context, docker config.Docker) ([]string, error) {
|
|
// nolint:prealloc
|
|
var buildFlags []string
|
|
for _, buildFlagTemplate := range docker.BuildFlagTemplates {
|
|
buildFlag, err := tmpl.New(ctx).Apply(buildFlagTemplate)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to process build flag template '%s': %w", buildFlagTemplate, err)
|
|
}
|
|
buildFlags = append(buildFlags, buildFlag)
|
|
}
|
|
return buildFlags, nil
|
|
}
|
|
|
|
// walks the src, recreating dirs and hard-linking files.
|
|
func link(src, dest string) error {
|
|
return filepath.Walk(src, func(path string, info os.FileInfo, err error) error {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// We have the following:
|
|
// - src = "a/b"
|
|
// - dest = "dist/linuxamd64/b"
|
|
// - path = "a/b/c.txt"
|
|
// So we join "a/b" with "c.txt" and use it as the destination.
|
|
dst := filepath.Join(dest, strings.Replace(path, src, "", 1))
|
|
log.WithFields(log.Fields{
|
|
"src": path,
|
|
"dst": dst,
|
|
}).Debug("extra file")
|
|
if info.IsDir() {
|
|
return os.MkdirAll(dst, info.Mode())
|
|
}
|
|
return os.Link(path, dst)
|
|
})
|
|
}
|
|
|
|
func dockerBuild(ctx *context.Context, root string, images, flags []string, buildx bool) error {
|
|
log.WithField("image", images[0]).WithField("buildx", buildx).Info("building docker image")
|
|
/* #nosec */
|
|
cmd := exec.CommandContext(ctx, "docker", buildCommand(buildx, images, flags)...)
|
|
cmd.Dir = root
|
|
log.WithField("cmd", cmd.Args).WithField("cwd", cmd.Dir).Debug("running")
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to build docker image: %s: \n%s: %w", images[0], string(out), err)
|
|
}
|
|
log.Debugf("docker build output: \n%s", string(out))
|
|
return nil
|
|
}
|
|
|
|
func buildCommand(buildx bool, images, flags []string) []string {
|
|
base := []string{"build", "."}
|
|
if buildx {
|
|
base = []string{"buildx", "build", ".", "--load"}
|
|
}
|
|
for _, image := range images {
|
|
base = append(base, "-t", image)
|
|
}
|
|
base = append(base, flags...)
|
|
return base
|
|
}
|
|
|
|
func dockerPush(ctx *context.Context, image *artifact.Artifact) error {
|
|
log.WithField("image", image.Name).Info("pushing docker image")
|
|
/* #nosec */
|
|
cmd := exec.CommandContext(ctx, "docker", "push", image.Name)
|
|
log.WithField("cmd", cmd.Args).Debug("running")
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to push docker image: \n%s: %w", string(out), err)
|
|
}
|
|
log.Debugf("docker push output: \n%s", string(out))
|
|
ctx.Artifacts.Add(&artifact.Artifact{
|
|
Type: artifact.DockerImage,
|
|
Name: image.Name,
|
|
Path: image.Path,
|
|
Goarch: image.Goarch,
|
|
Goos: image.Goos,
|
|
Goarm: image.Goarm,
|
|
})
|
|
return nil
|
|
}
|