1
0
mirror of https://github.com/goreleaser/goreleaser.git synced 2025-01-10 03:47:03 +02:00
goreleaser/internal/http/http.go

359 lines
9.4 KiB
Go
Raw Normal View History

2018-06-14 05:37:48 +02:00
// Package http implements functionality common to HTTP uploading pipelines.
package http
import (
"bytes"
"crypto/tls"
"crypto/x509"
2018-06-14 05:37:48 +02:00
"fmt"
"html/template"
"io"
h "net/http"
"os"
2018-09-14 01:07:13 +02:00
"runtime"
2018-06-14 05:37:48 +02:00
"strings"
"github.com/apex/log"
"github.com/pkg/errors"
"github.com/goreleaser/goreleaser/internal/artifact"
2018-09-12 19:18:01 +02:00
"github.com/goreleaser/goreleaser/internal/pipe"
"github.com/goreleaser/goreleaser/internal/semerrgroup"
"github.com/goreleaser/goreleaser/pkg/config"
"github.com/goreleaser/goreleaser/pkg/context"
2018-06-14 05:37:48 +02:00
)
const (
// ModeBinary uploads only compiled binaries
ModeBinary = "binary"
// ModeArchive uploads release archives
ModeArchive = "archive"
)
2018-06-25 06:38:11 +02:00
type asset struct {
ReadCloser io.ReadCloser
Size int64
}
type assetOpenFunc func(string, *artifact.Artifact) (*asset, error)
2018-11-08 02:04:49 +02:00
// nolint: gochecknoglobals
2018-06-25 06:38:11 +02:00
var assetOpen assetOpenFunc
2018-11-08 02:04:49 +02:00
// TODO: fix this.
// nolint: gochecknoinits
2018-06-25 06:38:11 +02:00
func init() {
assetOpenReset()
}
func assetOpenReset() {
assetOpen = assetOpenDefault
}
func assetOpenDefault(kind string, a *artifact.Artifact) (*asset, error) {
f, err := os.Open(a.Path)
if err != nil {
return nil, err
}
s, err := f.Stat()
if err != nil {
return nil, err
}
if s.IsDir() {
return nil, errors.Errorf("%s: upload failed: the asset to upload can't be a directory", kind)
}
return &asset{
ReadCloser: f,
Size: s.Size(),
}, nil
}
2018-06-14 05:37:48 +02:00
// Defaults sets default configuration options on Put structs
func Defaults(puts []config.Put) error {
for i := range puts {
defaults(&puts[i])
}
return nil
}
func defaults(put *config.Put) {
if put.Mode == "" {
put.Mode = ModeArchive
}
}
2018-06-25 06:38:11 +02:00
// CheckConfig validates a Put configuration returning a descriptive error when appropriate
func CheckConfig(ctx *context.Context, put *config.Put, kind string) error {
if put.Target == "" {
return misconfigured(kind, put, "missing target")
}
2018-06-14 05:37:48 +02:00
2018-06-25 06:38:11 +02:00
if put.Name == "" {
return misconfigured(kind, put, "missing name")
2018-06-14 05:37:48 +02:00
}
2018-06-25 06:38:11 +02:00
if put.Mode != ModeArchive && put.Mode != ModeBinary {
return misconfigured(kind, put, "mode must be 'binary' or 'archive'")
2018-06-14 05:37:48 +02:00
}
2018-06-25 06:38:11 +02:00
envName := fmt.Sprintf("%s_%s_SECRET", strings.ToUpper(kind), strings.ToUpper(put.Name))
2018-06-14 05:37:48 +02:00
if _, ok := ctx.Env[envName]; !ok {
2018-06-25 06:38:11 +02:00
return misconfigured(kind, put, fmt.Sprintf("missing %s environment variable", envName))
2018-06-14 05:37:48 +02:00
}
if put.TrustedCerts != "" && !x509.NewCertPool().AppendCertsFromPEM([]byte(put.TrustedCerts)) {
return misconfigured(kind, put, "no certificate could be added from the specified trusted_certificates configuration")
}
2018-06-14 05:37:48 +02:00
return nil
}
func misconfigured(kind string, upload *config.Put, reason string) error {
2018-09-12 19:18:01 +02:00
return pipe.Skip(fmt.Sprintf("%s section '%s' is not configured properly (%s)", kind, upload.Name, reason))
2018-06-14 05:37:48 +02:00
}
// ResponseChecker is a function capable of validating an http server response.
// It must return and error when the response must be considered a failure.
type ResponseChecker func(*h.Response) error
2018-06-14 05:37:48 +02:00
// Upload does the actual uploading work
func Upload(ctx *context.Context, puts []config.Put, kind string, check ResponseChecker) error {
2018-06-14 05:37:48 +02:00
if ctx.SkipPublish {
2018-09-12 19:18:01 +02:00
return pipe.ErrSkipPublishEnabled
2018-06-14 05:37:48 +02:00
}
// Handle every configured put
for _, put := range puts {
2018-11-08 02:04:49 +02:00
put := put
filters := []artifact.Filter{}
if put.Checksum {
filters = append(filters, artifact.ByType(artifact.Checksum))
}
if put.Signature {
filters = append(filters, artifact.ByType(artifact.Signature))
}
2018-06-14 05:37:48 +02:00
// We support two different modes
// - "archive": Upload all artifacts
// - "binary": Upload only the raw binaries
switch v := strings.ToLower(put.Mode); v {
2018-06-14 05:37:48 +02:00
case ModeArchive:
filters = append(filters,
2018-06-14 05:37:48 +02:00
artifact.ByType(artifact.UploadableArchive),
2018-09-18 15:14:30 +02:00
artifact.ByType(artifact.LinuxPackage),
)
2018-06-14 05:37:48 +02:00
case ModeBinary:
2018-09-18 15:14:30 +02:00
filters = append(filters, artifact.ByType(artifact.UploadableBinary))
2018-06-14 05:37:48 +02:00
default:
err := fmt.Errorf("%s: mode \"%s\" not supported", kind, v)
log.WithFields(log.Fields{
kind: put.Name,
"mode": v,
2018-06-14 05:37:48 +02:00
}).Error(err.Error())
return err
}
if err := uploadWithFilter(ctx, &put, artifact.Or(filters...), kind, check); err != nil {
2018-06-14 05:37:48 +02:00
return err
}
}
return nil
}
func uploadWithFilter(ctx *context.Context, put *config.Put, filter artifact.Filter, kind string, check ResponseChecker) error {
2018-09-18 15:14:30 +02:00
var artifacts = ctx.Artifacts.Filter(filter).List()
log.Debugf("will upload %d artifacts", len(artifacts))
var g = semerrgroup.New(ctx.Parallelism)
2018-09-18 15:14:30 +02:00
for _, artifact := range artifacts {
2018-06-14 05:37:48 +02:00
artifact := artifact
g.Go(func() error {
2018-06-25 06:38:11 +02:00
return uploadAsset(ctx, put, artifact, kind, check)
2018-06-14 05:37:48 +02:00
})
}
return g.Wait()
}
// uploadAsset uploads file to target and logs all actions
func uploadAsset(ctx *context.Context, put *config.Put, artifact artifact.Artifact, kind string, check ResponseChecker) error {
envBase := fmt.Sprintf("%s_%s_", strings.ToUpper(kind), strings.ToUpper(put.Name))
username := put.Username
if username == "" {
// username not configured: using env
username = ctx.Env[envBase+"USERNAME"]
}
secret := ctx.Env[envBase+"SECRET"]
2018-06-14 05:37:48 +02:00
// Generate the target url
2018-06-25 06:38:11 +02:00
targetURL, err := resolveTargetTemplate(ctx, put, artifact)
2018-06-14 05:37:48 +02:00
if err != nil {
msg := fmt.Sprintf("%s: error while building the target url", kind)
2018-06-25 06:38:11 +02:00
log.WithField("instance", put.Name).WithError(err).Error(msg)
2018-06-14 05:37:48 +02:00
return errors.Wrap(err, msg)
}
// Handle the artifact
2018-06-25 06:38:11 +02:00
asset, err := assetOpen(kind, &artifact)
2018-06-14 05:37:48 +02:00
if err != nil {
return err
}
2018-06-25 06:38:11 +02:00
defer asset.ReadCloser.Close() // nolint: errcheck
2018-06-14 05:37:48 +02:00
// The target url needs to contain the artifact name
if !strings.HasSuffix(targetURL, "/") {
targetURL += "/"
}
targetURL += artifact.Name
var headers = map[string]string{}
if put.ChecksumHeader != "" {
sum, err := artifact.Checksum()
if err != nil {
return err
}
headers[put.ChecksumHeader] = sum
}
_, err = uploadAssetToServer(ctx, put, targetURL, username, secret, headers, asset, check)
2018-06-14 05:37:48 +02:00
if err != nil {
msg := fmt.Sprintf("%s: upload failed", kind)
log.WithError(err).WithFields(log.Fields{
2018-06-25 06:38:11 +02:00
"instance": put.Name,
"username": username,
2018-06-14 05:37:48 +02:00
}).Error(msg)
return errors.Wrap(err, msg)
}
log.WithFields(log.Fields{
2018-06-25 06:38:11 +02:00
"instance": put.Name,
"mode": put.Mode,
2018-06-14 05:37:48 +02:00
}).Info("uploaded successful")
return nil
}
// uploadAssetToServer uploads the asset file to target
func uploadAssetToServer(ctx *context.Context, put *config.Put, target, username, secret string, headers map[string]string, a *asset, check ResponseChecker) (*h.Response, error) {
req, err := newUploadRequest(target, username, secret, headers, a)
2018-06-14 05:37:48 +02:00
if err != nil {
return nil, err
2018-06-14 05:37:48 +02:00
}
return executeHTTPRequest(ctx, put, req, check)
2018-06-14 05:37:48 +02:00
}
// newUploadRequest creates a new h.Request for uploading
func newUploadRequest(target, username, secret string, headers map[string]string, a *asset) (*h.Request, error) {
req, err := h.NewRequest("PUT", target, a.ReadCloser)
2018-06-14 05:37:48 +02:00
if err != nil {
return nil, err
}
2018-06-25 06:38:11 +02:00
req.ContentLength = a.Size
2018-06-14 05:37:48 +02:00
req.SetBasicAuth(username, secret)
for k, v := range headers {
req.Header.Add(k, v)
}
2018-06-14 05:37:48 +02:00
return req, err
}
func getHTTPClient(put *config.Put) (*h.Client, error) {
if put.TrustedCerts == "" {
return h.DefaultClient, nil
}
2018-09-14 01:07:13 +02:00
pool, err := x509.SystemCertPool()
if err != nil {
2018-09-14 01:07:13 +02:00
if runtime.GOOS == "windows" {
// on windows ignore errors until golang issues #16736 & #18609 get fixed
pool = x509.NewCertPool()
} else {
return nil, err
}
}
pool.AppendCertsFromPEM([]byte(put.TrustedCerts)) // already validated certs checked by CheckConfig
return &h.Client{
Transport: &h.Transport{
TLSClientConfig: &tls.Config{
RootCAs: pool,
},
},
}, nil
}
2018-06-14 05:37:48 +02:00
// executeHTTPRequest processes the http call with respect of context ctx
func executeHTTPRequest(ctx *context.Context, put *config.Put, req *h.Request, check ResponseChecker) (*h.Response, error) {
client, err := getHTTPClient(put)
if err != nil {
return nil, err
}
log.Debugf("executing request: %s %s (headers: %v)", req.Method, req.URL, req.Header)
resp, err := client.Do(req)
2018-06-14 05:37:48 +02:00
if err != nil {
// If we got an error, and the context has been canceled,
// the context's error is probably more useful.
select {
case <-ctx.Done():
return nil, ctx.Err()
2018-06-14 05:37:48 +02:00
default:
}
return nil, err
2018-06-14 05:37:48 +02:00
}
defer resp.Body.Close() // nolint: errcheck
err = check(resp)
2018-06-14 05:37:48 +02:00
if err != nil {
// even though there was an error, we still return the response
// in case the caller wants to inspect it further
return resp, err
2018-06-14 05:37:48 +02:00
}
return resp, err
2018-06-14 05:37:48 +02:00
}
// targetData is used as a template struct for
// Artifactory.Target
type targetData struct {
Version string
Tag string
ProjectName string
// Only supported in mode binary
Os string
Arch string
Arm string
}
// resolveTargetTemplate returns the resolved target template with replaced variables
// Those variables can be replaced by the given context, goos, goarch, goarm and more
func resolveTargetTemplate(ctx *context.Context, put *config.Put, artifact artifact.Artifact) (string, error) {
2018-06-14 05:37:48 +02:00
data := targetData{
Version: ctx.Version,
Tag: ctx.Git.CurrentTag,
ProjectName: ctx.Config.ProjectName,
}
if put.Mode == ModeBinary {
2018-06-14 05:37:48 +02:00
data.Os = replace(ctx.Config.Archive.Replacements, artifact.Goos)
data.Arch = replace(ctx.Config.Archive.Replacements, artifact.Goarch)
data.Arm = replace(ctx.Config.Archive.Replacements, artifact.Goarm)
}
var out bytes.Buffer
t, err := template.New(ctx.Config.ProjectName).Parse(put.Target)
2018-06-14 05:37:48 +02:00
if err != nil {
return "", err
}
err = t.Execute(&out, data)
return out.String(), err
}
func replace(replacements map[string]string, original string) string {
result := replacements[original]
if result == "" {
return original
}
return result
}