1
0
mirror of https://github.com/goreleaser/goreleaser.git synced 2025-01-24 04:16:27 +02:00

171 lines
4.6 KiB
Go
Raw Normal View History

2017-04-14 15:39:32 -03:00
// Package archive implements the pipe interface with the intent of
// archiving and compressing the binaries, readme, and other artifacts. It
// also provides an Archive interface which represents an archiving format.
package archive
2016-12-28 22:23:39 -02:00
2016-12-28 22:53:56 -02:00
import (
"fmt"
2016-12-29 18:10:11 -02:00
"os"
2017-07-13 20:22:10 -03:00
"path/filepath"
2017-12-17 18:01:58 -02:00
"strings"
2016-12-29 18:10:11 -02:00
2017-06-22 00:09:14 -03:00
"github.com/apex/log"
"github.com/campoy/unique"
2017-12-17 15:50:09 -02:00
"github.com/mattn/go-zglob"
"golang.org/x/sync/errgroup"
2017-06-18 18:30:39 -03:00
"github.com/goreleaser/archive"
2017-01-14 20:01:32 -02:00
"github.com/goreleaser/goreleaser/context"
2017-12-17 15:50:09 -02:00
"github.com/goreleaser/goreleaser/internal/artifact"
"github.com/goreleaser/goreleaser/internal/tmpl"
2016-12-28 22:53:56 -02:00
)
2016-12-28 22:23:39 -02:00
const (
defaultNameTemplate = "{{ .ProjectName }}_{{ .Version }}_{{ .Os }}_{{ .Arch }}{{ if .Arm }}v{{ .Arm }}{{ end }}"
defaultBinaryNameTemplate = "{{ .Binary }}_{{ .Version }}_{{ .Os }}_{{ .Arch }}{{ if .Arm }}v{{ .Arm }}{{ end }}"
)
2017-12-18 21:32:31 -02:00
// Pipe for archive
2016-12-30 09:27:35 -02:00
type Pipe struct{}
func (Pipe) String() string {
return "creating archives"
2016-12-30 09:27:35 -02:00
}
// Default sets the pipe defaults
func (Pipe) Default(ctx *context.Context) error {
var archive = &ctx.Config.Archive
if archive.Format == "" {
archive.Format = "tar.gz"
}
if len(archive.Files) == 0 {
archive.Files = []string{
"licence*",
"LICENCE*",
"license*",
"LICENSE*",
"readme*",
"README*",
"changelog*",
"CHANGELOG*",
}
}
if archive.NameTemplate == "" {
archive.NameTemplate = defaultNameTemplate
if archive.Format == "binary" {
archive.NameTemplate = defaultBinaryNameTemplate
}
}
return nil
}
// Run the pipe
func (Pipe) Run(ctx *context.Context) error {
var g errgroup.Group
var filtered = ctx.Artifacts.Filter(artifact.ByType(artifact.Binary))
for _, artifacts := range filtered.GroupByPlatform() {
artifacts := artifacts
g.Go(func() error {
if packageFormat(ctx, artifacts[0].Goos) == "binary" {
return skip(ctx, artifacts)
}
return create(ctx, artifacts)
})
}
return g.Wait()
}
func create(ctx *context.Context, binaries []artifact.Artifact) error {
var format = packageFormat(ctx, binaries[0].Goos)
folder, err := tmpl.New(ctx).
WithArtifact(binaries[0], ctx.Config.Archive.Replacements).
Apply(ctx.Config.Archive.NameTemplate)
2017-12-17 15:50:09 -02:00
if err != nil {
return err
}
archivePath := filepath.Join(ctx.Config.Dist, folder+"."+format)
archiveFile, err := os.Create(archivePath)
if err != nil {
return fmt.Errorf("failed to create directory %s: %s", archivePath, err.Error())
}
2017-12-17 23:04:29 -02:00
defer archiveFile.Close() // nolint: errcheck
2017-12-17 15:50:09 -02:00
log.WithField("archive", archivePath).Info("creating")
var a = archive.New(archiveFile)
2017-12-17 23:04:29 -02:00
defer a.Close() // nolint: errcheck
2017-12-17 15:50:09 -02:00
files, err := findFiles(ctx)
if err != nil {
return fmt.Errorf("failed to find files to archive: %s", err.Error())
}
for _, f := range files {
log.Debugf("adding %s", f)
2017-12-17 15:50:09 -02:00
if err = a.Add(wrap(ctx, f, folder), f); err != nil {
return fmt.Errorf("failed to add %s to the archive: %s", f, err.Error())
}
2017-12-17 15:50:09 -02:00
}
for _, binary := range binaries {
var bin = wrap(ctx, binary.Name, folder)
log.Debugf("adding %s", bin)
if err := a.Add(bin, binary.Path); err != nil {
2017-12-17 15:50:09 -02:00
return fmt.Errorf("failed to add %s -> %s to the archive: %s", binary.Path, binary.Name, err.Error())
}
2017-05-11 00:05:51 -03:00
}
2017-12-17 15:50:09 -02:00
ctx.Artifacts.Add(artifact.Artifact{
Type: artifact.UploadableArchive,
2017-12-17 15:50:09 -02:00
Name: folder + "." + format,
Path: archivePath,
Goos: binaries[0].Goos,
Goarch: binaries[0].Goarch,
Goarm: binaries[0].Goarm,
2017-12-17 15:50:09 -02:00
})
2017-01-14 12:51:09 -02:00
return nil
2016-12-28 22:53:56 -02:00
}
func skip(ctx *context.Context, binaries []artifact.Artifact) error {
for _, binary := range binaries {
log.WithField("binary", binary.Name).Info("skip archiving")
name, err := tmpl.New(ctx).
WithArtifact(binary, ctx.Config.Archive.Replacements).
Apply(ctx.Config.Archive.NameTemplate)
2017-12-17 18:01:58 -02:00
if err != nil {
return err
}
binary.Type = artifact.UploadableBinary
binary.Name = name + binary.Extra["Ext"]
ctx.Artifacts.Add(binary)
2017-07-03 00:57:39 -03:00
}
return nil
}
2017-05-11 00:05:51 -03:00
func findFiles(ctx *context.Context) (result []string, err error) {
for _, glob := range ctx.Config.Archive.Files {
2017-05-11 12:43:04 -03:00
files, err := zglob.Glob(glob)
2017-05-11 00:05:51 -03:00
if err != nil {
return result, fmt.Errorf("globbing failed for pattern %s: %s", glob, err.Error())
2017-05-11 00:05:51 -03:00
}
result = append(result, files...)
}
// remove duplicates
unique.Slice(&result, func(i, j int) bool {
return strings.Compare(result[i], result[j]) < 0
})
2017-05-11 00:05:51 -03:00
return
}
// Wrap archive files with folder if set in config.
func wrap(ctx *context.Context, name, folder string) string {
if ctx.Config.Archive.WrapInDirectory {
return filepath.Join(folder, name)
}
return name
}
func packageFormat(ctx *context.Context, platform string) string {
for _, override := range ctx.Config.Archive.FormatOverrides {
if strings.HasPrefix(platform, override.Goos) {
return override.Format
}
}
return ctx.Config.Archive.Format
}