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

351 lines
9.6 KiB
Go
Raw Normal View History

// Package nfpm implements the Pipe interface providing nFPM bindings.
package nfpm
import (
"fmt"
"os"
"path/filepath"
"strings"
"github.com/apex/log"
"github.com/goreleaser/nfpm/v2"
_ "github.com/goreleaser/nfpm/v2/apk" // blank import to register the format
_ "github.com/goreleaser/nfpm/v2/deb" // blank import to register the format
"github.com/goreleaser/nfpm/v2/files"
_ "github.com/goreleaser/nfpm/v2/rpm" // blank import to register the format
"github.com/imdario/mergo"
"github.com/goreleaser/goreleaser/internal/artifact"
2019-05-07 07:18:35 -03:00
"github.com/goreleaser/goreleaser/internal/ids"
"github.com/goreleaser/goreleaser/internal/linux"
2018-09-12 14:18:01 -03:00
"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"
)
const defaultNameTemplate = "{{ .PackageName }}_{{ .Version }}_{{ .Os }}_{{ .Arch }}{{ if .Arm }}v{{ .Arm }}{{ end }}{{ if .Mips }}_{{ .Mips }}{{ end }}"
// Pipe for nfpm packaging.
type Pipe struct{}
feat: improve output and pipe skipping (#2480) * refactor: improve middleware Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: upload tests Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: twitter tests Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: source tests Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: snapshot tests Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * test: improved some tests Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: snapcraft skip Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip slack Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip sign Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip scoop Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip reddit Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip discord Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip publish Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip nfpm Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip milestone Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip custompublishers Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip checksums Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip changelog Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip brew Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip blob Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip before Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip artifactory Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip announce Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip defaults Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: cmds Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip docker Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * chore: todo Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: go.mod Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip release Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: remove old skip pipe errors Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip teams Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip brew Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix/test: skip smtp Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: lint issues Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip docker Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip brew and scoop Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip docker Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip http/artifactory Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * test: increase coverage Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * test: fix Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com>
2021-09-18 10:21:29 -03:00
func (Pipe) String() string { return "linux packages" }
func (Pipe) Skip(ctx *context.Context) bool { return len(ctx.Config.NFPMs) == 0 }
// Default sets the pipe defaults.
func (Pipe) Default(ctx *context.Context) error {
ids := ids.New("nfpms")
for i := range ctx.Config.NFPMs {
fpm := &ctx.Config.NFPMs[i]
2019-05-07 07:18:35 -03:00
if fpm.ID == "" {
fpm.ID = "default"
}
if fpm.Bindir == "" {
fpm.Bindir = "/usr/local/bin"
}
if fpm.PackageName == "" {
fpm.PackageName = ctx.Config.ProjectName
}
if fpm.FileNameTemplate == "" {
fpm.FileNameTemplate = defaultNameTemplate
}
if len(fpm.Builds) == 0 { // TODO: change this to empty by default and deal with it in the filtering code
for _, b := range ctx.Config.Builds {
fpm.Builds = append(fpm.Builds, b.ID)
}
}
2019-05-07 07:18:35 -03:00
ids.Inc(fpm.ID)
}
2019-05-07 07:18:35 -03:00
return ids.Validate()
}
// Run the pipe.
func (Pipe) Run(ctx *context.Context) error {
for _, nfpm := range ctx.Config.NFPMs {
if len(nfpm.Formats) == 0 {
// FIXME: this assumes other nfpm configs will fail too...
return pipe.Skip("no output formats configured")
}
if err := doRun(ctx, nfpm); err != nil {
return err
}
}
return nil
}
func doRun(ctx *context.Context, fpm config.NFPM) error {
linuxBinaries := ctx.Artifacts.Filter(artifact.And(
2018-02-17 14:28:48 -02:00
artifact.ByType(artifact.Binary),
artifact.ByGoos("linux"),
artifact.ByIDs(fpm.Builds...),
2018-02-17 14:28:48 -02:00
)).GroupByPlatform()
if len(linuxBinaries) == 0 {
return fmt.Errorf("no linux binaries found for builds %v", fpm.Builds)
}
g := semerrgroup.New(ctx.Parallelism)
for _, format := range fpm.Formats {
2018-02-17 14:28:48 -02:00
for platform, artifacts := range linuxBinaries {
format := format
arch := linux.Arch(platform)
artifacts := artifacts
g.Go(func() error {
return create(ctx, fpm, format, arch, artifacts)
})
}
}
return g.Wait()
}
func mergeOverrides(fpm config.NFPM, format string) (*config.NFPMOverridables, error) {
var overridden config.NFPMOverridables
if err := mergo.Merge(&overridden, fpm.NFPMOverridables); err != nil {
return nil, err
}
perFormat, ok := fpm.Overrides[format]
if ok {
err := mergo.Merge(&overridden, perFormat, mergo.WithOverride)
if err != nil {
return nil, err
}
}
return &overridden, nil
}
func create(ctx *context.Context, fpm config.NFPM, format, arch string, binaries []*artifact.Artifact) error {
overridden, err := mergeOverrides(fpm, format)
if err != nil {
return err
}
tmpl := tmpl.New(ctx).
WithArtifact(binaries[0], overridden.Replacements).
WithExtraFields(tmpl.Fields{
"Release": fpm.Release,
"Epoch": fpm.Epoch,
"PackageName": fpm.PackageName,
})
name, err := tmpl.Apply(overridden.FileNameTemplate)
if err != nil {
return err
}
binDir, err := tmpl.Apply(fpm.Bindir)
if err != nil {
return err
}
homepage, err := tmpl.Apply(fpm.Homepage)
if err != nil {
return err
}
description, err := tmpl.Apply(fpm.Description)
if err != nil {
return err
}
debKeyFile, err := tmpl.Apply(overridden.Deb.Signature.KeyFile)
if err != nil {
return err
}
rpmKeyFile, err := tmpl.Apply(overridden.RPM.Signature.KeyFile)
if err != nil {
return err
}
apkKeyFile, err := tmpl.Apply(overridden.APK.Signature.KeyFile)
if err != nil {
return err
}
contents := files.Contents{}
for _, content := range overridden.Contents {
src, err := tmpl.Apply(content.Source)
if err != nil {
return err
}
dst, err := tmpl.Apply(content.Destination)
if err != nil {
return err
}
contents = append(contents, &files.Content{
Source: src,
Destination: dst,
Type: content.Type,
Packager: content.Packager,
FileInfo: content.FileInfo,
})
}
// FPM meta package should not contain binaries at all
if !fpm.Meta {
log := log.WithField("package", name+"."+format).WithField("arch", arch)
for _, binary := range binaries {
src := binary.Path
dst := filepath.Join(binDir, binary.Name)
log.WithField("src", src).WithField("dst", dst).Debug("adding binary to package")
contents = append(contents, &files.Content{
2021-03-09 18:16:54 -03:00
Source: filepath.ToSlash(src),
Destination: filepath.ToSlash(dst),
})
}
}
log.WithField("files", destinations(contents)).Debug("all archive files")
info := &nfpm.Info{
Arch: arch,
Platform: "linux",
Name: fpm.PackageName,
Version: ctx.Version,
Section: fpm.Section,
Priority: fpm.Priority,
Epoch: fpm.Epoch,
Release: fpm.Release,
Prerelease: fpm.Prerelease,
VersionMetadata: fpm.VersionMetadata,
Maintainer: fpm.Maintainer,
Description: description,
Vendor: fpm.Vendor,
Homepage: homepage,
License: fpm.License,
2018-04-19 00:01:51 -03:00
Overridables: nfpm.Overridables{
Conflicts: overridden.Conflicts,
Depends: overridden.Dependencies,
Recommends: overridden.Recommends,
Suggests: overridden.Suggests,
Replaces: overridden.Replaces,
EmptyFolders: overridden.EmptyFolders,
Contents: contents,
2018-04-19 00:01:51 -03:00
Scripts: nfpm.Scripts{
PreInstall: overridden.Scripts.PreInstall,
PostInstall: overridden.Scripts.PostInstall,
PreRemove: overridden.Scripts.PreRemove,
PostRemove: overridden.Scripts.PostRemove,
2018-04-19 00:01:51 -03:00
},
Deb: nfpm.Deb{
Scripts: nfpm.DebScripts{
Rules: overridden.Deb.Scripts.Rules,
Templates: overridden.Deb.Scripts.Templates,
},
Triggers: nfpm.DebTriggers{
Interest: overridden.Deb.Triggers.Interest,
InterestAwait: overridden.Deb.Triggers.InterestAwait,
InterestNoAwait: overridden.Deb.Triggers.InterestNoAwait,
Activate: overridden.Deb.Triggers.Activate,
ActivateAwait: overridden.Deb.Triggers.ActivateAwait,
ActivateNoAwait: overridden.Deb.Triggers.ActivateNoAwait,
},
Breaks: overridden.Deb.Breaks,
Signature: nfpm.DebSignature{
PackageSignature: nfpm.PackageSignature{
KeyFile: debKeyFile,
KeyPassphrase: getPassphraseFromEnv(ctx, "DEB", fpm.ID),
},
Type: overridden.Deb.Signature.Type,
},
},
RPM: nfpm.RPM{
Summary: overridden.RPM.Summary,
Group: overridden.RPM.Group,
Compression: overridden.RPM.Compression,
Signature: nfpm.RPMSignature{
PackageSignature: nfpm.PackageSignature{
KeyFile: rpmKeyFile,
KeyPassphrase: getPassphraseFromEnv(ctx, "RPM", fpm.ID),
},
},
Scripts: nfpm.RPMScripts{
PreTrans: overridden.RPM.Scripts.PreTrans,
PostTrans: overridden.RPM.Scripts.PostTrans,
},
},
APK: nfpm.APK{
Signature: nfpm.APKSignature{
PackageSignature: nfpm.PackageSignature{
KeyFile: apkKeyFile,
KeyPassphrase: getPassphraseFromEnv(ctx, "APK", fpm.ID),
},
KeyName: overridden.APK.Signature.KeyName,
},
Scripts: nfpm.APKScripts{
PreUpgrade: overridden.APK.Scripts.PreUpgrade,
PostUpgrade: overridden.APK.Scripts.PostUpgrade,
},
},
},
}
2018-02-17 14:28:48 -02:00
if ctx.SkipSign {
info.APK.Signature = nfpm.APKSignature{}
info.RPM.Signature = nfpm.RPMSignature{}
info.Deb.Signature = nfpm.DebSignature{}
}
2018-04-05 21:34:25 -03:00
if err = nfpm.Validate(info); err != nil {
return fmt.Errorf("invalid nfpm config: %w", err)
2018-04-05 20:41:12 -03:00
}
packager, err := nfpm.Get(format)
if err != nil {
return err
}
path := filepath.Join(ctx.Config.Dist, name+"."+format)
2018-10-27 13:38:24 -03:00
log.WithField("file", path).Info("creating")
w, err := os.Create(path)
if err != nil {
return err
}
defer w.Close()
if err := packager.Package(nfpm.WithDefaults(info), w); err != nil {
return fmt.Errorf("nfpm failed: %w", err)
}
2018-02-17 14:28:48 -02:00
if err := w.Close(); err != nil {
return fmt.Errorf("could not close package file: %w", err)
2018-02-17 14:28:48 -02:00
}
ctx.Artifacts.Add(&artifact.Artifact{
Type: artifact.LinuxPackage,
Name: name + "." + format,
Path: path,
Goos: binaries[0].Goos,
Goarch: binaries[0].Goarch,
Goarm: binaries[0].Goarm,
Extra: map[string]interface{}{
"Builds": binaries,
"ID": fpm.ID,
"Format": format,
"Files": contents,
},
})
return nil
}
func destinations(contents files.Contents) []string {
result := make([]string, 0, len(contents))
for _, f := range contents {
result = append(result, f.Destination)
}
return result
}
func getPassphraseFromEnv(ctx *context.Context, packager string, nfpmID string) string {
var passphrase string
nfpmID = strings.ToUpper(nfpmID)
packagerSpecificPassphrase := ctx.Env[fmt.Sprintf(
"NFPM_%s_%s_PASSPHRASE",
nfpmID,
packager,
)]
if packagerSpecificPassphrase != "" {
passphrase = packagerSpecificPassphrase
} else {
generalPassphrase := ctx.Env[fmt.Sprintf("NFPM_%s_PASSPHRASE", nfpmID)]
passphrase = generalPassphrase
}
return passphrase
}