mirror of
https://github.com/goreleaser/goreleaser.git
synced 2025-01-22 04:08:49 +02:00
76dc0b559e
Following #3540, output artifacts' checksums to the artifact info. This addition makes it easier to consume the checksums, especially when running from e.g. GitHub Actions. New tests: 1. Add a check for the checksum in the extra field. 2. Add a test for every checksum algorithm (to see that it doesn't break for any algo's output). 3. Add a case of a binary and an extra file (to see that the logic doesn't break when there's a mix). p.s. While working on that, I noticed that the convention for extra fields is actually to use UpperCamelCase rather than lower. I was mistaken because I looked at the subfields of the "DockerConfig" extra field. I think it's a good idea to fix it quickly, before the next release rolls and it becomes a compatibility issue. I took the liberty to fix it here as an extra commit. Please let me know if you want it to be in a separate PR. --- Tests: ``` go test • refreshing checksums file=binary_bar_checksums.txt • refreshing checksums file=binary_bar_checksums.txt • refreshing checksums file=binary_bar_checksums.txt PASS ok github.com/goreleaser/goreleaser/internal/pipe/checksums 0.184s ``` Co-authored-by: Carlos Alexandro Becker <caarlos0@users.noreply.github.com>
158 lines
3.6 KiB
Go
158 lines
3.6 KiB
Go
// Package checksums provides a Pipe that creates .checksums files for
|
|
// each artifact.
|
|
package checksums
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"sort"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/caarlos0/log"
|
|
"github.com/goreleaser/goreleaser/internal/artifact"
|
|
"github.com/goreleaser/goreleaser/internal/extrafiles"
|
|
"github.com/goreleaser/goreleaser/internal/semerrgroup"
|
|
"github.com/goreleaser/goreleaser/internal/tmpl"
|
|
"github.com/goreleaser/goreleaser/pkg/context"
|
|
)
|
|
|
|
const (
|
|
artifactChecksumExtra = "Checksum"
|
|
)
|
|
|
|
var (
|
|
errNoArtifacts = errors.New("there are no artifacts to sign")
|
|
lock sync.Mutex
|
|
)
|
|
|
|
// Pipe for checksums.
|
|
type Pipe struct{}
|
|
|
|
func (Pipe) String() string { return "calculating checksums" }
|
|
func (Pipe) Skip(ctx *context.Context) bool { return ctx.Config.Checksum.Disable }
|
|
|
|
// Default sets the pipe defaults.
|
|
func (Pipe) Default(ctx *context.Context) error {
|
|
if ctx.Config.Checksum.NameTemplate == "" {
|
|
ctx.Config.Checksum.NameTemplate = "{{ .ProjectName }}_{{ .Version }}_checksums.txt"
|
|
}
|
|
if ctx.Config.Checksum.Algorithm == "" {
|
|
ctx.Config.Checksum.Algorithm = "sha256"
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Run the pipe.
|
|
func (Pipe) Run(ctx *context.Context) error {
|
|
filename, err := tmpl.New(ctx).Apply(ctx.Config.Checksum.NameTemplate)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
filepath := filepath.Join(ctx.Config.Dist, filename)
|
|
if err := refresh(ctx, filepath); err != nil {
|
|
if errors.Is(err, errNoArtifacts) {
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
ctx.Artifacts.Add(&artifact.Artifact{
|
|
Type: artifact.Checksum,
|
|
Path: filepath,
|
|
Name: filename,
|
|
Extra: map[string]interface{}{
|
|
artifact.ExtraRefresh: func() error {
|
|
log.WithField("file", filename).Info("refreshing checksums")
|
|
return refresh(ctx, filepath)
|
|
},
|
|
},
|
|
})
|
|
return nil
|
|
}
|
|
|
|
func refresh(ctx *context.Context, filepath string) error {
|
|
lock.Lock()
|
|
defer lock.Unlock()
|
|
filter := artifact.Or(
|
|
artifact.ByType(artifact.UploadableArchive),
|
|
artifact.ByType(artifact.UploadableBinary),
|
|
artifact.ByType(artifact.UploadableSourceArchive),
|
|
artifact.ByType(artifact.LinuxPackage),
|
|
artifact.ByType(artifact.SBOM),
|
|
)
|
|
if len(ctx.Config.Checksum.IDs) > 0 {
|
|
filter = artifact.And(filter, artifact.ByIDs(ctx.Config.Checksum.IDs...))
|
|
}
|
|
|
|
artifactList := ctx.Artifacts.Filter(filter).List()
|
|
|
|
extraFiles, err := extrafiles.Find(ctx, ctx.Config.Checksum.ExtraFiles)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for name, path := range extraFiles {
|
|
artifactList = append(artifactList, &artifact.Artifact{
|
|
Name: name,
|
|
Path: path,
|
|
Type: artifact.UploadableFile,
|
|
})
|
|
}
|
|
|
|
if len(artifactList) == 0 {
|
|
return errNoArtifacts
|
|
}
|
|
|
|
g := semerrgroup.New(ctx.Parallelism)
|
|
sumLines := make([]string, len(artifactList))
|
|
for i, artifact := range artifactList {
|
|
i := i
|
|
artifact := artifact
|
|
g.Go(func() error {
|
|
sumLine, err := checksums(ctx.Config.Checksum.Algorithm, artifact)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
sumLines[i] = sumLine
|
|
return nil
|
|
})
|
|
}
|
|
|
|
err = g.Wait()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
file, err := os.OpenFile(
|
|
filepath,
|
|
os.O_APPEND|os.O_WRONLY|os.O_CREATE|os.O_TRUNC,
|
|
0o644,
|
|
)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer file.Close()
|
|
|
|
// sort to ensure the signature is deterministic downstream
|
|
sort.Strings(sumLines)
|
|
_, err = file.WriteString(strings.Join(sumLines, ""))
|
|
return err
|
|
}
|
|
|
|
func checksums(algorithm string, a *artifact.Artifact) (string, error) {
|
|
log.WithField("file", a.Name).Debug("checksumming")
|
|
sha, err := a.Checksum(algorithm)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
if a.Extra == nil {
|
|
a.Extra = make(artifact.Extras)
|
|
}
|
|
a.Extra[artifactChecksumExtra] = fmt.Sprintf("%s:%s", algorithm, sha)
|
|
|
|
return fmt.Sprintf("%v %v\n", sha, a.Name), nil
|
|
}
|