2017-09-11 23:31:00 -03:00
|
|
|
// Package docker provides a Pipe that creates and pushes a Docker image
|
|
|
|
package docker
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2018-08-20 18:58:56 -03:00
|
|
|
"io/ioutil"
|
2017-09-12 07:54:43 -03:00
|
|
|
"os"
|
2017-09-11 23:31:00 -03:00
|
|
|
"os/exec"
|
|
|
|
"path/filepath"
|
2017-12-24 10:12:51 -02:00
|
|
|
"strings"
|
2017-09-11 23:31:00 -03:00
|
|
|
|
2017-12-03 00:42:52 -02:00
|
|
|
"github.com/apex/log"
|
2017-12-17 23:11:17 -02:00
|
|
|
"github.com/goreleaser/goreleaser/internal/artifact"
|
2018-10-20 22:49:14 +08:00
|
|
|
"github.com/goreleaser/goreleaser/internal/deprecate"
|
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"
|
2018-10-20 13:45:31 -03:00
|
|
|
"github.com/pkg/errors"
|
2017-09-11 23:31:00 -03:00
|
|
|
)
|
|
|
|
|
2017-09-12 00:22:02 -03:00
|
|
|
// ErrNoDocker is shown when docker cannot be found in $PATH
|
|
|
|
var ErrNoDocker = errors.New("docker not present in $PATH")
|
|
|
|
|
|
|
|
// Pipe for docker
|
2017-09-11 23:31:00 -03:00
|
|
|
type Pipe struct{}
|
|
|
|
|
2017-12-02 19:53:19 -02:00
|
|
|
func (Pipe) String() string {
|
2018-10-26 14:27:41 -06:00
|
|
|
return "Docker images"
|
2017-09-11 23:31:00 -03:00
|
|
|
}
|
|
|
|
|
2017-12-02 19:53:19 -02:00
|
|
|
// Default sets the pipe defaults
|
|
|
|
func (Pipe) Default(ctx *context.Context) error {
|
2017-12-05 11:19:44 -02:00
|
|
|
for i := range ctx.Config.Dockers {
|
2017-12-17 18:10:38 -02:00
|
|
|
var docker = &ctx.Config.Dockers[i]
|
2018-10-20 21:09:55 +08:00
|
|
|
|
2018-10-20 20:21:52 +08:00
|
|
|
if docker.Image != "" {
|
|
|
|
deprecate.Notice("docker.image")
|
2019-01-11 16:27:39 -02:00
|
|
|
deprecate.Notice("docker.tag_templates")
|
2018-10-20 21:09:55 +08:00
|
|
|
|
2019-01-11 16:27:39 -02:00
|
|
|
if len(docker.TagTemplates) == 0 {
|
2018-10-20 21:09:55 +08:00
|
|
|
docker.TagTemplates = []string{"{{ .Version }}"}
|
|
|
|
}
|
2019-01-11 16:27:39 -02:00
|
|
|
|
|
|
|
for _, tag := range docker.TagTemplates {
|
|
|
|
docker.ImageTemplates = append(
|
|
|
|
docker.ImageTemplates,
|
|
|
|
fmt.Sprintf("%s:%s", docker.Image, tag),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if docker.Binary != "" {
|
|
|
|
deprecate.Notice("docker.binary")
|
|
|
|
docker.Binaries = append(docker.Binaries, docker.Binary)
|
2018-10-20 20:21:52 +08:00
|
|
|
}
|
2018-10-20 21:09:55 +08:00
|
|
|
|
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
|
|
|
}
|
|
|
|
}
|
2018-10-05 11:00:25 -05:00
|
|
|
// only set defaults if there is exactly 1 docker setup in the config file.
|
2017-12-02 19:53:19 -02:00
|
|
|
if len(ctx.Config.Dockers) != 1 {
|
|
|
|
return nil
|
|
|
|
}
|
2019-01-11 16:27:39 -02:00
|
|
|
if len(ctx.Config.Dockers[0].Binaries) == 0 {
|
|
|
|
ctx.Config.Dockers[0].Binaries = []string{
|
|
|
|
ctx.Config.Builds[0].Binary,
|
|
|
|
}
|
2017-12-02 19:53:19 -02:00
|
|
|
}
|
|
|
|
if ctx.Config.Dockers[0].Dockerfile == "" {
|
|
|
|
ctx.Config.Dockers[0].Dockerfile = "Dockerfile"
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-12-17 16:46:45 -02:00
|
|
|
// Run the pipe
|
|
|
|
func (Pipe) Run(ctx *context.Context) error {
|
2018-10-20 20:21:52 +08:00
|
|
|
if len(ctx.Config.Dockers) == 0 || missingImage(ctx) {
|
2018-09-12 14:18:01 -03:00
|
|
|
return pipe.Skip("docker section is not configured")
|
2017-12-17 16:46:45 -02:00
|
|
|
}
|
|
|
|
_, err := exec.LookPath("docker")
|
|
|
|
if err != nil {
|
|
|
|
return ErrNoDocker
|
|
|
|
}
|
|
|
|
return doRun(ctx)
|
|
|
|
}
|
|
|
|
|
2018-10-20 13:45:31 -03:00
|
|
|
// Publish the docker images
|
|
|
|
func (Pipe) Publish(ctx *context.Context) error {
|
|
|
|
var images = ctx.Artifacts.Filter(artifact.ByType(artifact.PublishableDockerImage)).List()
|
|
|
|
for _, image := range images {
|
|
|
|
if err := dockerPush(ctx, image); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-10-20 20:21:52 +08:00
|
|
|
func missingImage(ctx *context.Context) bool {
|
|
|
|
return ctx.Config.Dockers[0].Image == "" && len(ctx.Config.Dockers[0].ImageTemplates) == 0
|
|
|
|
}
|
|
|
|
|
2017-09-14 21:19:56 -03:00
|
|
|
func doRun(ctx *context.Context) error {
|
2018-07-09 21:38:00 -07:00
|
|
|
var g = 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 {
|
|
|
|
log.WithField("docker", docker).Debug("looking for binaries matching")
|
2019-01-17 18:22:12 -02:00
|
|
|
var binaryNames = make([]string, len(docker.Binaries))
|
|
|
|
for i := range docker.Binaries {
|
|
|
|
bin, err := tmpl.New(ctx).Apply(docker.Binaries[i])
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrapf(err, "failed to execute binary template '%s'", docker.Binaries[i])
|
|
|
|
}
|
|
|
|
binaryNames[i] = bin
|
|
|
|
}
|
2017-12-25 23:27:06 -02:00
|
|
|
var binaries = ctx.Artifacts.Filter(
|
|
|
|
artifact.And(
|
|
|
|
artifact.ByGoos(docker.Goos),
|
|
|
|
artifact.ByGoarch(docker.Goarch),
|
|
|
|
artifact.ByGoarm(docker.Goarm),
|
|
|
|
artifact.ByType(artifact.Binary),
|
|
|
|
func(a artifact.Artifact) bool {
|
2019-01-17 18:22:12 -02:00
|
|
|
for _, bin := range binaryNames {
|
2019-01-11 16:27:39 -02:00
|
|
|
if a.ExtraOr("Binary", "").(string) == bin {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
2017-12-25 23:27:06 -02:00
|
|
|
},
|
|
|
|
),
|
|
|
|
).List()
|
2019-01-11 16:27:39 -02:00
|
|
|
// TODO: not so good of a check, if one binary match multiple
|
|
|
|
// binaries and the other match none, this will still pass...
|
|
|
|
if len(binaries) != len(docker.Binaries) {
|
2018-07-17 08:24:58 -03:00
|
|
|
return fmt.Errorf(
|
2019-01-11 16:27:39 -02:00
|
|
|
"%d binaries match docker definition: %v: %s_%s_%s, should be %d",
|
2018-07-17 08:24:58 -03:00
|
|
|
len(binaries),
|
2019-01-11 16:27:39 -02:00
|
|
|
docker.Binaries, docker.Goos, docker.Goarch, docker.Goarm,
|
|
|
|
len(docker.Binaries),
|
2018-07-17 08:24:58 -03:00
|
|
|
)
|
2017-09-11 23:31:00 -03:00
|
|
|
}
|
2019-01-11 16:27:39 -02:00
|
|
|
return process(ctx, docker, binaries)
|
2017-12-25 23:27:06 -02:00
|
|
|
})
|
2017-09-11 23:31:00 -03:00
|
|
|
}
|
2017-12-25 23:27:06 -02:00
|
|
|
return g.Wait()
|
2017-09-11 23:31:00 -03:00
|
|
|
}
|
|
|
|
|
2019-01-11 16:27:39 -02:00
|
|
|
func process(ctx *context.Context, docker config.Docker, bins []artifact.Artifact) error {
|
2018-09-04 15:19:01 +03:00
|
|
|
tmp, err := ioutil.TempDir(ctx.Config.Dist, "goreleaserdocker")
|
2018-08-20 18:58:56 -03:00
|
|
|
if err != nil {
|
2018-10-05 11:00:25 -05:00
|
|
|
return errors.Wrap(err, "failed to create temporary dir")
|
2018-08-20 18:58:56 -03:00
|
|
|
}
|
2018-09-04 09:26:45 -03:00
|
|
|
log.Debug("tempdir: " + tmp)
|
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
|
|
|
|
2018-08-20 18:58:56 -03:00
|
|
|
if err := os.Link(docker.Dockerfile, filepath.Join(tmp, "Dockerfile")); err != nil {
|
2017-09-12 07:54:43 -03:00
|
|
|
return errors.Wrap(err, "failed to link dockerfile")
|
2017-09-11 23:31:00 -03:00
|
|
|
}
|
2017-09-25 19:10:04 -03:00
|
|
|
for _, file := range docker.Files {
|
2018-12-16 11:11:01 -02:00
|
|
|
if err := os.MkdirAll(filepath.Join(tmp, filepath.Dir(file)), 0755); err != nil {
|
|
|
|
return errors.Wrapf(err, "failed to link extra file '%s'", file)
|
|
|
|
}
|
|
|
|
if err := link(file, filepath.Join(tmp, file)); err != nil {
|
2017-09-25 19:10:04 -03:00
|
|
|
return errors.Wrapf(err, "failed to link extra file '%s'", file)
|
|
|
|
}
|
|
|
|
}
|
2019-01-11 16:27:39 -02:00
|
|
|
for _, bin := range bins {
|
|
|
|
if err := os.Link(bin.Path, filepath.Join(tmp, filepath.Base(bin.Path))); err != nil {
|
|
|
|
return errors.Wrap(err, "failed to link binary")
|
|
|
|
}
|
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
|
|
|
|
}
|
|
|
|
|
2018-10-25 14:18:58 -03:00
|
|
|
if err := dockerBuild(ctx, tmp, images, buildFlags); err != nil {
|
2017-09-11 23:31:00 -03:00
|
|
|
return err
|
|
|
|
}
|
2018-10-20 13:45:31 -03:00
|
|
|
if docker.SkipPush {
|
|
|
|
// TODO: this should also be better handled
|
|
|
|
log.Warn(pipe.Skip("skip_push is set").Error())
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
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
|
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 {
|
|
|
|
return nil, errors.Wrapf(err, "failed to execute image template '%s'", imageTemplate)
|
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
|
|
|
return nil, errors.Wrapf(err, "failed to process build flag template '%s'", buildFlagTemplate)
|
|
|
|
}
|
|
|
|
buildFlags = append(buildFlags, buildFlag)
|
|
|
|
}
|
|
|
|
return buildFlags, nil
|
|
|
|
}
|
|
|
|
|
2017-12-24 10:12:51 -02:00
|
|
|
// walks the src, recreating dirs and hard-linking files
|
2017-12-20 22:08:35 +01:00
|
|
|
func link(src, dest string) error {
|
2017-12-24 10:12:51 -02:00
|
|
|
return filepath.Walk(src, func(path string, info os.FileInfo, err error) error {
|
2017-12-26 14:56:20 -02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-12-24 10:12:51 -02:00
|
|
|
// 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.
|
|
|
|
var dst = filepath.Join(dest, strings.Replace(path, src, "", 1))
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"src": path,
|
|
|
|
"dst": dst,
|
2018-10-05 09:49:47 -03:00
|
|
|
}).Debug("extra file")
|
2017-12-20 22:08:35 +01:00
|
|
|
if info.IsDir() {
|
2017-12-24 10:12:51 -02:00
|
|
|
return os.MkdirAll(dst, info.Mode())
|
2017-12-20 22:08:35 +01:00
|
|
|
}
|
2017-12-24 10:12:51 -02:00
|
|
|
return os.Link(path, dst)
|
|
|
|
})
|
2017-12-20 22:08:35 +01:00
|
|
|
}
|
|
|
|
|
2018-10-25 14:18:58 -03:00
|
|
|
func dockerBuild(ctx *context.Context, root string, images, flags []string) error {
|
|
|
|
log.WithField("image", images[0]).Info("building docker image")
|
2017-11-26 12:09:12 -02:00
|
|
|
/* #nosec */
|
2018-10-25 14:18:58 -03:00
|
|
|
var cmd = exec.CommandContext(ctx, "docker", buildCommand(images, flags)...)
|
2018-08-20 18:58:56 -03:00
|
|
|
cmd.Dir = root
|
|
|
|
log.WithField("cmd", cmd.Args).WithField("cwd", cmd.Dir).Debug("running")
|
2017-09-11 23:31:00 -03:00
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrapf(err, "failed to build docker image: \n%s", string(out))
|
|
|
|
}
|
|
|
|
log.Debugf("docker build output: \n%s", string(out))
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-10-25 14:18:58 -03:00
|
|
|
func buildCommand(images, flags []string) []string {
|
|
|
|
base := []string{"build", "."}
|
|
|
|
for _, image := range images {
|
|
|
|
base = append(base, "-t", image)
|
|
|
|
}
|
2018-10-03 00:02:12 +08:00
|
|
|
base = append(base, flags...)
|
|
|
|
return base
|
|
|
|
}
|
|
|
|
|
2018-10-20 13:45:31 -03:00
|
|
|
func dockerPush(ctx *context.Context, image artifact.Artifact) error {
|
2018-10-20 13:59:13 -03:00
|
|
|
log.WithField("image", image.Name).Info("pushing docker image")
|
2017-11-26 12:09:12 -02:00
|
|
|
/* #nosec */
|
2018-10-20 13:45:31 -03:00
|
|
|
var cmd = exec.CommandContext(ctx, "docker", "push", image.Name)
|
2018-02-13 14:09:03 -02:00
|
|
|
log.WithField("cmd", cmd.Args).Debug("running")
|
2017-09-11 23:31:00 -03:00
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrapf(err, "failed to push docker image: \n%s", string(out))
|
|
|
|
}
|
|
|
|
log.Debugf("docker push output: \n%s", string(out))
|
2018-10-20 13:45:31 -03:00
|
|
|
image.Type = artifact.DockerImage
|
|
|
|
ctx.Artifacts.Add(image)
|
2017-09-11 23:31:00 -03:00
|
|
|
return nil
|
|
|
|
}
|