2017-09-11 23:31:00 -03:00
|
|
|
package docker
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2017-09-12 07:54:43 -03:00
|
|
|
"os"
|
2017-09-11 23:31:00 -03:00
|
|
|
"path/filepath"
|
2021-06-27 17:55:00 -03:00
|
|
|
"sort"
|
2017-12-24 10:12:51 -02:00
|
|
|
"strings"
|
2017-09-11 23:31:00 -03:00
|
|
|
|
2022-06-21 21:11:15 -03:00
|
|
|
"github.com/caarlos0/log"
|
2017-12-17 23:11:17 -02:00
|
|
|
"github.com/goreleaser/goreleaser/internal/artifact"
|
2022-03-17 07:55:17 -03:00
|
|
|
"github.com/goreleaser/goreleaser/internal/deprecate"
|
2021-07-24 19:59:43 -03:00
|
|
|
"github.com/goreleaser/goreleaser/internal/gio"
|
2021-08-16 22:11:54 -03:00
|
|
|
"github.com/goreleaser/goreleaser/internal/ids"
|
2018-09-12 14:18:01 -03:00
|
|
|
"github.com/goreleaser/goreleaser/internal/pipe"
|
2018-07-09 21:38:00 -07:00
|
|
|
"github.com/goreleaser/goreleaser/internal/semerrgroup"
|
2018-07-08 21:05:18 -07:00
|
|
|
"github.com/goreleaser/goreleaser/internal/tmpl"
|
2018-08-14 23:50:20 -03:00
|
|
|
"github.com/goreleaser/goreleaser/pkg/config"
|
|
|
|
"github.com/goreleaser/goreleaser/pkg/context"
|
2017-09-11 23:31:00 -03:00
|
|
|
)
|
|
|
|
|
2021-06-26 16:36:31 -03:00
|
|
|
const (
|
|
|
|
dockerConfigExtra = "DockerConfig"
|
|
|
|
|
2021-09-17 01:18:44 +03:00
|
|
|
useBuildx = "buildx"
|
|
|
|
useDocker = "docker"
|
2022-03-17 07:55:17 -03:00
|
|
|
useBuildPacks = "buildpacks" // deprecated: should not be used anymore
|
2021-06-26 16:36:31 -03:00
|
|
|
)
|
|
|
|
|
2020-05-26 00:48:10 -03:00
|
|
|
// Pipe for docker.
|
2017-09-11 23:31:00 -03:00
|
|
|
type Pipe struct{}
|
|
|
|
|
2021-09-18 10:21:29 -03:00
|
|
|
func (Pipe) String() string { return "docker images" }
|
2022-06-22 21:39:20 -03:00
|
|
|
func (Pipe) Skip(ctx *context.Context) bool { return len(ctx.Config.Dockers) == 0 || ctx.SkipDocker }
|
2017-09-11 23:31:00 -03:00
|
|
|
|
2020-05-26 00:48:10 -03:00
|
|
|
// Default sets the pipe defaults.
|
2017-12-02 19:53:19 -02:00
|
|
|
func (Pipe) Default(ctx *context.Context) error {
|
2021-08-16 22:11:54 -03:00
|
|
|
ids := ids.New("dockers")
|
2017-12-05 11:19:44 -02:00
|
|
|
for i := range ctx.Config.Dockers {
|
2021-04-19 09:31:57 -03:00
|
|
|
docker := &ctx.Config.Dockers[i]
|
2018-10-20 21:09:55 +08:00
|
|
|
|
2021-08-16 22:11:54 -03:00
|
|
|
if docker.ID != "" {
|
|
|
|
ids.Inc(docker.ID)
|
|
|
|
}
|
2017-12-17 18:10:38 -02:00
|
|
|
if docker.Goos == "" {
|
|
|
|
docker.Goos = "linux"
|
|
|
|
}
|
|
|
|
if docker.Goarch == "" {
|
|
|
|
docker.Goarch = "amd64"
|
2017-12-05 11:19:44 -02:00
|
|
|
}
|
2022-04-11 22:43:22 -03:00
|
|
|
if docker.Goamd64 == "" {
|
2022-04-13 21:29:39 -03:00
|
|
|
docker.Goamd64 = "v1"
|
2022-04-11 22:43:22 -03:00
|
|
|
}
|
2020-11-24 23:41:40 +01:00
|
|
|
if docker.Dockerfile == "" {
|
|
|
|
docker.Dockerfile = "Dockerfile"
|
|
|
|
}
|
2022-03-17 07:55:17 -03:00
|
|
|
if docker.Use == useBuildPacks {
|
|
|
|
deprecate.Notice(ctx, "dockers.use: buildpacks")
|
|
|
|
}
|
2021-06-26 16:36:31 -03:00
|
|
|
if docker.Use == "" {
|
|
|
|
docker.Use = useDocker
|
|
|
|
}
|
2021-06-27 17:55:00 -03:00
|
|
|
if err := validateImager(docker.Use); err != nil {
|
|
|
|
return err
|
2021-06-26 16:36:31 -03:00
|
|
|
}
|
2019-10-06 17:09:51 -03:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
}
|
2017-12-05 11:19:44 -02:00
|
|
|
}
|
2021-08-16 22:11:54 -03:00
|
|
|
return ids.Validate()
|
2017-12-02 19:53:19 -02:00
|
|
|
}
|
|
|
|
|
2021-06-27 17:55:00 -03:00
|
|
|
func validateImager(use string) error {
|
|
|
|
valid := make([]string, 0, len(imagers))
|
|
|
|
for k := range imagers {
|
|
|
|
valid = append(valid, k)
|
|
|
|
}
|
|
|
|
for _, s := range valid {
|
2021-06-26 16:36:31 -03:00
|
|
|
if s == use {
|
2021-06-27 17:55:00 -03:00
|
|
|
return nil
|
2021-06-26 16:36:31 -03:00
|
|
|
}
|
|
|
|
}
|
2021-06-27 17:55:00 -03:00
|
|
|
sort.Strings(valid)
|
|
|
|
return fmt.Errorf("docker: invalid use: %s, valid options are %v", use, valid)
|
2021-06-26 16:36:31 -03:00
|
|
|
}
|
|
|
|
|
2020-05-26 00:48:10 -03:00
|
|
|
// Publish the docker images.
|
2018-10-20 13:45:31 -03:00
|
|
|
func (Pipe) Publish(ctx *context.Context) error {
|
2021-04-19 09:31:57 -03:00
|
|
|
images := ctx.Artifacts.Filter(artifact.ByType(artifact.PublishableDockerImage)).List()
|
2018-10-20 13:45:31 -03:00
|
|
|
for _, image := range images {
|
|
|
|
if err := dockerPush(ctx, image); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-09-18 10:21:29 -03:00
|
|
|
// Run the pipe.
|
|
|
|
func (Pipe) Run(ctx *context.Context) error {
|
2021-04-19 09:31:57 -03:00
|
|
|
g := semerrgroup.NewSkipAware(semerrgroup.New(ctx.Parallelism))
|
2018-08-20 18:58:56 -03:00
|
|
|
for _, docker := range ctx.Config.Dockers {
|
2017-12-25 23:27:06 -02:00
|
|
|
docker := docker
|
|
|
|
g.Go(func() error {
|
2021-01-07 16:21:12 -03:00
|
|
|
log.WithField("docker", docker).Debug("looking for artifacts matching")
|
2021-04-19 09:31:57 -03:00
|
|
|
filters := []artifact.Filter{
|
2019-12-27 11:55:03 -03:00
|
|
|
artifact.ByGoos(docker.Goos),
|
|
|
|
artifact.ByGoarch(docker.Goarch),
|
2021-01-07 16:21:12 -03:00
|
|
|
artifact.Or(
|
|
|
|
artifact.ByType(artifact.Binary),
|
|
|
|
artifact.ByType(artifact.LinuxPackage),
|
|
|
|
),
|
2019-12-27 11:55:03 -03:00
|
|
|
}
|
2022-04-11 22:43:22 -03:00
|
|
|
// TODO: properly test this
|
|
|
|
switch docker.Goarch {
|
|
|
|
case "amd64":
|
|
|
|
filters = append(filters, artifact.ByGoamd64(docker.Goamd64))
|
|
|
|
case "arm":
|
|
|
|
filters = append(filters, artifact.ByGoarm(docker.Goarm))
|
|
|
|
}
|
2021-01-07 16:21:12 -03:00
|
|
|
if len(docker.IDs) > 0 {
|
|
|
|
filters = append(filters, artifact.ByIDs(docker.IDs...))
|
2017-09-11 23:31:00 -03:00
|
|
|
}
|
2021-04-19 09:31:57 -03:00
|
|
|
artifacts := ctx.Artifacts.Filter(artifact.And(filters...))
|
2021-01-07 16:21:12 -03:00
|
|
|
log.WithField("artifacts", artifacts.Paths()).Debug("found artifacts")
|
|
|
|
return process(ctx, docker, artifacts.List())
|
2017-12-25 23:27:06 -02:00
|
|
|
})
|
2017-09-11 23:31:00 -03:00
|
|
|
}
|
2022-05-02 22:05:42 -03:00
|
|
|
if err := g.Wait(); err != nil {
|
2022-05-12 14:44:09 -03:00
|
|
|
if pipe.IsSkip(err) {
|
|
|
|
return err
|
|
|
|
}
|
2022-05-02 22:05:42 -03:00
|
|
|
return fmt.Errorf("docker build failed: %w\nLearn more at https://goreleaser.com/errors/docker-build\n", err) // nolint:revive
|
|
|
|
}
|
|
|
|
return nil
|
2017-09-11 23:31:00 -03:00
|
|
|
}
|
|
|
|
|
2021-01-07 16:21:12 -03:00
|
|
|
func process(ctx *context.Context, docker config.Docker, artifacts []*artifact.Artifact) error {
|
2021-11-21 15:10:08 +01:00
|
|
|
tmp, err := os.MkdirTemp(ctx.Config.Dist, "goreleaserdocker")
|
2018-08-20 18:58:56 -03:00
|
|
|
if err != nil {
|
2020-09-21 14:47:51 -03:00
|
|
|
return fmt.Errorf("failed to create temporary dir: %w", err)
|
2018-08-20 18:58:56 -03:00
|
|
|
}
|
2018-10-03 20:58:02 +08:00
|
|
|
|
2019-01-11 16:27:39 -02:00
|
|
|
images, err := processImageTemplates(ctx, docker)
|
2018-10-03 20:58:02 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
2017-12-05 11:19:44 -02:00
|
|
|
}
|
2018-10-03 20:58:02 +08:00
|
|
|
|
2021-09-18 10:21:29 -03:00
|
|
|
if len(images) == 0 {
|
|
|
|
return pipe.Skip("no image templates found")
|
|
|
|
}
|
|
|
|
|
2021-07-04 01:52:59 +00:00
|
|
|
log := log.WithField("image", images[0])
|
|
|
|
log.Debug("tempdir: " + tmp)
|
|
|
|
|
2021-09-17 01:18:44 +03:00
|
|
|
if docker.Use != useBuildPacks {
|
2022-01-06 15:34:55 -03:00
|
|
|
dockerfile, err := tmpl.New(ctx).Apply(docker.Dockerfile)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := gio.Copy(dockerfile, filepath.Join(tmp, "Dockerfile")); err != nil {
|
2021-09-17 01:18:44 +03:00
|
|
|
return fmt.Errorf("failed to copy dockerfile: %w", err)
|
|
|
|
}
|
2017-09-11 23:31:00 -03:00
|
|
|
}
|
2017-09-25 19:10:04 -03:00
|
|
|
for _, file := range docker.Files {
|
2021-04-19 09:31:57 -03:00
|
|
|
if err := os.MkdirAll(filepath.Join(tmp, filepath.Dir(file)), 0o755); err != nil {
|
2021-07-24 19:59:43 -03:00
|
|
|
return fmt.Errorf("failed to copy extra file '%s': %w", file, err)
|
2018-12-16 11:11:01 -02:00
|
|
|
}
|
2021-07-24 19:59:43 -03:00
|
|
|
if err := gio.Copy(file, filepath.Join(tmp, file)); err != nil {
|
|
|
|
return fmt.Errorf("failed to copy extra file '%s': %w", file, err)
|
2017-09-25 19:10:04 -03:00
|
|
|
}
|
|
|
|
}
|
2021-01-07 16:21:12 -03:00
|
|
|
for _, art := range artifacts {
|
2021-07-24 19:59:43 -03:00
|
|
|
if err := gio.Copy(art.Path, filepath.Join(tmp, filepath.Base(art.Path))); err != nil {
|
|
|
|
return fmt.Errorf("failed to copy artifact: %w", err)
|
2019-01-11 16:27:39 -02:00
|
|
|
}
|
2018-08-20 18:58:56 -03:00
|
|
|
}
|
2018-10-03 20:58:02 +08:00
|
|
|
|
2019-01-11 16:27:39 -02:00
|
|
|
buildFlags, err := processBuildFlagTemplates(ctx, docker)
|
2018-10-03 20:58:02 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-07-04 01:52:59 +00:00
|
|
|
log.Info("building docker image")
|
2021-06-27 17:55:00 -03:00
|
|
|
if err := imagers[docker.Use].Build(ctx, tmp, images, buildFlags); err != nil {
|
2017-09-11 23:31:00 -03:00
|
|
|
return err
|
|
|
|
}
|
2019-03-06 17:17:53 +01:00
|
|
|
|
|
|
|
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")
|
2018-10-20 13:45:31 -03:00
|
|
|
}
|
|
|
|
for _, img := range images {
|
2019-08-12 17:44:48 -03:00
|
|
|
ctx.Artifacts.Add(&artifact.Artifact{
|
2018-10-20 13:45:31 -03:00
|
|
|
Type: artifact.PublishableDockerImage,
|
|
|
|
Name: img,
|
|
|
|
Path: img,
|
|
|
|
Goarch: docker.Goarch,
|
|
|
|
Goos: docker.Goos,
|
|
|
|
Goarm: docker.Goarm,
|
2021-06-26 16:36:31 -03:00
|
|
|
Extra: map[string]interface{}{
|
|
|
|
dockerConfigExtra: docker,
|
|
|
|
},
|
2018-10-20 13:45:31 -03:00
|
|
|
})
|
|
|
|
}
|
|
|
|
return nil
|
2017-09-26 18:50:00 -03:00
|
|
|
}
|
|
|
|
|
2019-01-11 16:27:39 -02:00
|
|
|
func processImageTemplates(ctx *context.Context, docker config.Docker) ([]string, error) {
|
2018-10-03 20:58:02 +08:00
|
|
|
// nolint:prealloc
|
|
|
|
var images []string
|
2018-10-20 20:21:52 +08:00
|
|
|
for _, imageTemplate := range docker.ImageTemplates {
|
2019-01-11 16:27:39 -02:00
|
|
|
image, err := tmpl.New(ctx).Apply(imageTemplate)
|
2018-10-20 20:21:52 +08:00
|
|
|
if err != nil {
|
2020-09-21 14:47:51 -03:00
|
|
|
return nil, fmt.Errorf("failed to execute image template '%s': %w", imageTemplate, err)
|
2018-10-20 20:21:52 +08:00
|
|
|
}
|
2020-12-28 17:36:46 -03:00
|
|
|
if image == "" {
|
|
|
|
continue
|
|
|
|
}
|
2018-10-20 20:21:52 +08:00
|
|
|
|
|
|
|
images = append(images, image)
|
|
|
|
}
|
|
|
|
|
2018-10-03 20:58:02 +08:00
|
|
|
return images, nil
|
|
|
|
}
|
|
|
|
|
2019-01-11 16:27:39 -02:00
|
|
|
func processBuildFlagTemplates(ctx *context.Context, docker config.Docker) ([]string, error) {
|
2018-10-03 20:58:02 +08:00
|
|
|
// nolint:prealloc
|
|
|
|
var buildFlags []string
|
|
|
|
for _, buildFlagTemplate := range docker.BuildFlagTemplates {
|
2019-01-11 16:27:39 -02:00
|
|
|
buildFlag, err := tmpl.New(ctx).Apply(buildFlagTemplate)
|
2018-10-03 20:58:02 +08:00
|
|
|
if err != nil {
|
2020-09-21 14:47:51 -03:00
|
|
|
return nil, fmt.Errorf("failed to process build flag template '%s': %w", buildFlagTemplate, err)
|
2018-10-03 20:58:02 +08:00
|
|
|
}
|
|
|
|
buildFlags = append(buildFlags, buildFlag)
|
|
|
|
}
|
|
|
|
return buildFlags, nil
|
|
|
|
}
|
|
|
|
|
2019-08-12 17:44:48 -03:00
|
|
|
func dockerPush(ctx *context.Context, image *artifact.Artifact) error {
|
2021-12-20 23:43:35 -03:00
|
|
|
log.WithField("image", image.Name).Info("pushing")
|
2021-06-26 16:36:31 -03:00
|
|
|
docker := image.Extra[dockerConfigExtra].(config.Docker)
|
2021-06-27 17:55:00 -03:00
|
|
|
if err := imagers[docker.Use].Push(ctx, image.Name, docker.PushFlags); err != nil {
|
2021-06-26 16:36:31 -03:00
|
|
|
return err
|
2017-09-11 23:31:00 -03:00
|
|
|
}
|
2021-08-16 22:11:54 -03:00
|
|
|
art := &artifact.Artifact{
|
2019-09-03 18:27:16 +02:00
|
|
|
Type: artifact.DockerImage,
|
|
|
|
Name: image.Name,
|
|
|
|
Path: image.Path,
|
|
|
|
Goarch: image.Goarch,
|
|
|
|
Goos: image.Goos,
|
|
|
|
Goarm: image.Goarm,
|
2021-08-16 22:11:54 -03:00
|
|
|
Extra: map[string]interface{}{},
|
|
|
|
}
|
|
|
|
if docker.ID != "" {
|
2021-10-16 22:46:11 -03:00
|
|
|
art.Extra[artifact.ExtraID] = docker.ID
|
2021-08-16 22:11:54 -03:00
|
|
|
}
|
|
|
|
ctx.Artifacts.Add(art)
|
2017-09-11 23:31:00 -03:00
|
|
|
return nil
|
|
|
|
}
|