1
0
mirror of https://github.com/goreleaser/goreleaser.git synced 2025-01-22 04:08:49 +02:00

192 lines
4.2 KiB
Go
Raw Normal View History

// Package artifact provides the core artifact storage for goreleaser
package artifact
2017-12-17 16:31:06 -02:00
import (
2018-08-21 15:55:35 -03:00
"crypto/sha256"
"encoding/hex"
"io"
"os"
2017-12-17 16:31:06 -02:00
"sync"
2017-12-17 16:59:54 -02:00
"github.com/apex/log"
"github.com/pkg/errors"
2017-12-17 16:31:06 -02:00
)
// Type defines the type of an artifact
type Type int
const (
2017-12-17 16:10:40 -02:00
// UploadableArchive a tar.gz/zip archive to be uploaded
UploadableArchive Type = iota
// UploadableBinary is a binary file to be uploaded
UploadableBinary
// Binary is a binary (output of a gobuild)
Binary
// LinuxPackage is a linux package generated by nfpm or snapcraft
2017-12-17 17:11:08 -02:00
LinuxPackage
// DockerImage is a docker image
DockerImage
2017-12-17 17:25:04 -02:00
// Checksum is a checksums file
Checksum
2017-12-17 17:25:04 -02:00
// Signature is a signature file
Signature
)
2018-09-15 18:53:59 -03:00
func (t Type) String() string {
switch t {
case UploadableArchive:
return "Archive"
case UploadableBinary:
return "Binary"
case Binary:
return "Binary"
case LinuxPackage:
return "LinuxPackage"
case DockerImage:
return "DockerImage"
case Checksum:
return "Checksum"
case Signature:
return "Signature"
}
return "unknown"
}
// Artifact represents an artifact and its relevant info
type Artifact struct {
Name string
2017-12-17 15:24:49 -02:00
Path string
Goos string
Goarch string
Goarm string
Type Type
2017-12-17 18:01:58 -02:00
Extra map[string]string
}
2018-08-21 15:55:35 -03:00
// Checksum calculates the SHA256 checksum of the artifact.
func (a Artifact) Checksum() (string, error) {
log.Debugf("calculating sha256sum for %s", a.Path)
file, err := os.Open(a.Path)
if err != nil {
return "", errors.Wrap(err, "failed to checksum")
2018-08-21 15:55:35 -03:00
}
defer file.Close() // nolint: errcheck
var hash = sha256.New()
_, err = io.Copy(hash, file)
if err != nil {
return "", errors.Wrap(err, "failed to checksum")
2018-08-21 15:55:35 -03:00
}
return hex.EncodeToString(hash.Sum(nil)), nil
}
// Artifacts is a list of artifacts
type Artifacts struct {
items []Artifact
lock *sync.Mutex
}
// New return a new list of artifacts
func New() Artifacts {
return Artifacts{
items: []Artifact{},
lock: &sync.Mutex{},
}
}
2017-12-17 15:50:09 -02:00
// List return the actual list of artifacts
2017-12-17 16:10:40 -02:00
func (artifacts Artifacts) List() []Artifact {
2017-12-17 15:50:09 -02:00
return artifacts.items
}
// GroupByPlatform groups the artifacts by their platform
2017-12-17 17:11:08 -02:00
func (artifacts Artifacts) GroupByPlatform() map[string][]Artifact {
2017-12-17 15:50:09 -02:00
var result = map[string][]Artifact{}
for _, a := range artifacts.items {
2017-12-18 09:19:02 -02:00
plat := a.Goos + a.Goarch + a.Goarm
result[plat] = append(result[plat], a)
2017-12-17 15:50:09 -02:00
}
return result
}
// Add safely adds a new artifact to an artifact list
func (artifacts *Artifacts) Add(a Artifact) {
artifacts.lock.Lock()
defer artifacts.lock.Unlock()
2017-12-17 18:01:58 -02:00
log.WithFields(log.Fields{
2017-12-18 09:19:02 -02:00
"name": a.Name,
"path": a.Path,
"type": a.Type,
2018-10-05 09:48:00 -03:00
}).Debug("added new artifact")
artifacts.items = append(artifacts.items, a)
}
// Filter defines an artifact filter which can be used within the Filter
// function
type Filter func(a Artifact) bool
// ByGoos is a predefined filter that filters by the given goos
func ByGoos(s string) Filter {
return func(a Artifact) bool {
return a.Goos == s
}
}
// ByGoarch is a predefined filter that filters by the given goarch
func ByGoarch(s string) Filter {
return func(a Artifact) bool {
return a.Goarch == s
}
}
// ByGoarm is a predefined filter that filters by the given goarm
func ByGoarm(s string) Filter {
return func(a Artifact) bool {
return a.Goarm == s
}
}
// ByType is a predefined filter that filters by the given type
func ByType(t Type) Filter {
return func(a Artifact) bool {
return a.Type == t
}
}
2017-12-17 16:59:54 -02:00
// Or performs an OR between all given filters
func Or(filters ...Filter) Filter {
return func(a Artifact) bool {
for _, f := range filters {
if f(a) {
return true
}
}
return false
}
}
// And performs an AND between all given filters
func And(filters ...Filter) Filter {
return func(a Artifact) bool {
for _, f := range filters {
if !f(a) {
return false
}
}
return true
}
}
// Filter filters the artifact list, returning a new instance.
// There are some pre-defined filters but anything of the Type Filter
// is accepted.
2017-12-17 16:59:54 -02:00
// You can compose filters by using the And and Or filters.
func (artifacts *Artifacts) Filter(filter Filter) Artifacts {
var result = New()
for _, a := range artifacts.items {
2017-12-17 16:59:54 -02:00
if filter(a) {
2017-12-17 18:01:58 -02:00
result.items = append(result.items, a)
}
}
return result
}