2018-06-14 05:37:48 +02:00
|
|
|
// Package http implements functionality common to HTTP uploading pipelines.
|
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2018-08-28 23:40:46 +02:00
|
|
|
"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"
|
2018-07-10 06:38:00 +02:00
|
|
|
"github.com/goreleaser/goreleaser/internal/semerrgroup"
|
2018-08-15 04:50:20 +02:00
|
|
|
"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
|
|
|
|
}
|
|
|
|
|
2019-11-18 15:34:17 +02:00
|
|
|
// Defaults sets default configuration options on upload structs
|
|
|
|
func Defaults(uploads []config.Upload) error {
|
|
|
|
for i := range uploads {
|
|
|
|
defaults(&uploads[i])
|
2018-06-14 05:37:48 +02:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-11-18 15:34:17 +02:00
|
|
|
func defaults(upload *config.Upload) {
|
|
|
|
if upload.Mode == "" {
|
|
|
|
upload.Mode = ModeArchive
|
|
|
|
}
|
|
|
|
if upload.Method == "" {
|
|
|
|
upload.Method = h.MethodPut
|
2018-06-14 05:37:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-18 15:34:17 +02:00
|
|
|
// CheckConfig validates an upload configuration returning a descriptive error when appropriate
|
|
|
|
func CheckConfig(ctx *context.Context, upload *config.Upload, kind string) error {
|
|
|
|
if upload.Target == "" {
|
|
|
|
return misconfigured(kind, upload, "missing target")
|
2018-06-25 06:38:11 +02:00
|
|
|
}
|
2018-06-14 05:37:48 +02:00
|
|
|
|
2019-11-18 15:34:17 +02:00
|
|
|
if upload.Name == "" {
|
|
|
|
return misconfigured(kind, upload, "missing name")
|
2018-06-14 05:37:48 +02:00
|
|
|
}
|
|
|
|
|
2019-11-18 15:34:17 +02:00
|
|
|
if upload.Mode != ModeArchive && upload.Mode != ModeBinary {
|
|
|
|
return misconfigured(kind, upload, "mode must be 'binary' or 'archive'")
|
2018-06-14 05:37:48 +02:00
|
|
|
}
|
|
|
|
|
2019-11-18 15:34:17 +02:00
|
|
|
if _, err := getUsername(ctx, upload, kind); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := getPassword(ctx, upload, kind); err != nil {
|
|
|
|
return err
|
2018-06-14 05:37:48 +02:00
|
|
|
}
|
|
|
|
|
2019-11-18 15:34:17 +02:00
|
|
|
if upload.TrustedCerts != "" && !x509.NewCertPool().AppendCertsFromPEM([]byte(upload.TrustedCerts)) {
|
|
|
|
return misconfigured(kind, upload, "no certificate could be added from the specified trusted_certificates configuration")
|
2018-08-28 23:40:46 +02:00
|
|
|
}
|
|
|
|
|
2018-06-14 05:37:48 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-11-18 15:34:17 +02:00
|
|
|
func getUsername(ctx *context.Context, upload *config.Upload, kind string) (string, error) {
|
|
|
|
if upload.Username != "" {
|
|
|
|
return upload.Username, nil
|
|
|
|
}
|
|
|
|
var key = fmt.Sprintf("%s_%s_USERNAME", strings.ToUpper(kind), strings.ToUpper(upload.Name))
|
|
|
|
user, ok := ctx.Env[key]
|
|
|
|
if !ok {
|
|
|
|
return "", misconfigured(kind, upload, fmt.Sprintf("missing username or %s environment variable", key))
|
|
|
|
}
|
|
|
|
return user, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getPassword(ctx *context.Context, upload *config.Upload, kind string) (string, error) {
|
|
|
|
var key = fmt.Sprintf("%s_%s_SECRET", strings.ToUpper(kind), strings.ToUpper(upload.Name))
|
|
|
|
pwd, ok := ctx.Env[key]
|
|
|
|
if !ok {
|
|
|
|
return "", misconfigured(kind, upload, fmt.Sprintf("missing %s environment variable", key))
|
|
|
|
}
|
|
|
|
return pwd, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func misconfigured(kind string, upload *config.Upload, 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.
|
2018-09-10 14:08:54 +02:00
|
|
|
// 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
|
2019-11-18 15:34:17 +02:00
|
|
|
func Upload(ctx *context.Context, uploads []config.Upload, 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
|
|
|
}
|
|
|
|
|
2019-11-18 15:34:17 +02:00
|
|
|
// Handle every configured upload
|
|
|
|
for _, upload := range uploads {
|
|
|
|
upload := upload
|
2018-06-25 06:36:23 +02:00
|
|
|
filters := []artifact.Filter{}
|
2019-11-18 15:34:17 +02:00
|
|
|
if upload.Checksum {
|
2018-06-25 06:36:23 +02:00
|
|
|
filters = append(filters, artifact.ByType(artifact.Checksum))
|
|
|
|
}
|
2019-11-18 15:34:17 +02:00
|
|
|
if upload.Signature {
|
2018-06-25 06:36:23 +02:00
|
|
|
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
|
2019-11-18 15:34:17 +02:00
|
|
|
switch v := strings.ToLower(upload.Mode); v {
|
2018-06-14 05:37:48 +02:00
|
|
|
case ModeArchive:
|
2018-06-25 06:36:23 +02:00
|
|
|
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{
|
2019-11-18 15:34:17 +02:00
|
|
|
kind: upload.Name,
|
2018-06-25 06:36:23 +02:00
|
|
|
"mode": v,
|
2018-06-14 05:37:48 +02:00
|
|
|
}).Error(err.Error())
|
|
|
|
return err
|
|
|
|
}
|
2019-05-17 15:25:01 +02:00
|
|
|
|
|
|
|
var filter = artifact.Or(filters...)
|
2019-11-18 15:34:17 +02:00
|
|
|
if len(upload.IDs) > 0 {
|
|
|
|
filter = artifact.And(filter, artifact.ByIDs(upload.IDs...))
|
2019-05-17 15:25:01 +02:00
|
|
|
}
|
2019-11-18 15:34:17 +02:00
|
|
|
if err := uploadWithFilter(ctx, &upload, filter, kind, check); err != nil {
|
2018-06-14 05:37:48 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-11-18 15:34:17 +02:00
|
|
|
func uploadWithFilter(ctx *context.Context, upload *config.Upload, filter artifact.Filter, kind string, check ResponseChecker) error {
|
2018-09-18 15:14:30 +02:00
|
|
|
var artifacts = ctx.Artifacts.Filter(filter).List()
|
2018-10-01 21:52:16 +02:00
|
|
|
log.Debugf("will upload %d artifacts", len(artifacts))
|
2018-07-10 06:38:00 +02:00
|
|
|
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 {
|
2019-11-18 15:34:17 +02:00
|
|
|
return uploadAsset(ctx, upload, artifact, kind, check)
|
2018-06-14 05:37:48 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
return g.Wait()
|
|
|
|
}
|
|
|
|
|
|
|
|
// uploadAsset uploads file to target and logs all actions
|
2019-11-18 15:34:17 +02:00
|
|
|
func uploadAsset(ctx *context.Context, upload *config.Upload, artifact *artifact.Artifact, kind string, check ResponseChecker) error {
|
|
|
|
username, err := getUsername(ctx, upload, kind)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
secret, err := getPassword(ctx, upload, kind)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2018-08-25 22:26:06 +02:00
|
|
|
}
|
2018-06-14 05:37:48 +02:00
|
|
|
|
|
|
|
// Generate the target url
|
2019-11-18 15:34:17 +02:00
|
|
|
targetURL, err := resolveTargetTemplate(ctx, upload, artifact)
|
2018-06-14 05:37:48 +02:00
|
|
|
if err != nil {
|
|
|
|
msg := fmt.Sprintf("%s: error while building the target url", kind)
|
2019-11-18 15:34:17 +02:00
|
|
|
log.WithField("instance", upload.Name).WithError(err).Error(msg)
|
2018-06-14 05:37:48 +02:00
|
|
|
return errors.Wrap(err, msg)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle the artifact
|
2019-08-12 22:44:48 +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
|
|
|
|
|
2018-10-01 21:52:16 +02:00
|
|
|
var headers = map[string]string{}
|
2019-11-18 15:34:17 +02:00
|
|
|
if upload.ChecksumHeader != "" {
|
2019-02-04 21:27:51 +02:00
|
|
|
sum, err := artifact.Checksum("sha256")
|
2018-10-01 21:52:16 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-11-18 15:34:17 +02:00
|
|
|
headers[upload.ChecksumHeader] = sum
|
2018-10-01 21:52:16 +02:00
|
|
|
}
|
|
|
|
|
2019-11-18 15:34:17 +02:00
|
|
|
res, err := uploadAssetToServer(ctx, upload, 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{
|
2019-11-18 15:34:17 +02:00
|
|
|
"instance": upload.Name,
|
2018-08-25 22:26:06 +02:00
|
|
|
"username": username,
|
2018-06-14 05:37:48 +02:00
|
|
|
}).Error(msg)
|
|
|
|
return errors.Wrap(err, msg)
|
|
|
|
}
|
2019-06-09 21:51:24 +02:00
|
|
|
if err := res.Body.Close(); err != nil {
|
|
|
|
log.WithError(err).Warn("failed to close response body")
|
|
|
|
}
|
2018-06-14 05:37:48 +02:00
|
|
|
|
|
|
|
log.WithFields(log.Fields{
|
2019-11-18 15:34:17 +02:00
|
|
|
"instance": upload.Name,
|
|
|
|
"mode": upload.Mode,
|
2018-06-14 05:37:48 +02:00
|
|
|
}).Info("uploaded successful")
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// uploadAssetToServer uploads the asset file to target
|
2019-11-18 15:34:17 +02:00
|
|
|
func uploadAssetToServer(ctx *context.Context, upload *config.Upload, target, username, secret string, headers map[string]string, a *asset, check ResponseChecker) (*h.Response, error) {
|
|
|
|
req, err := newUploadRequest(upload.Method, target, username, secret, headers, a)
|
2018-06-14 05:37:48 +02:00
|
|
|
if err != nil {
|
2018-09-10 14:08:54 +02:00
|
|
|
return nil, err
|
2018-06-14 05:37:48 +02:00
|
|
|
}
|
|
|
|
|
2019-11-18 15:34:17 +02:00
|
|
|
return executeHTTPRequest(ctx, upload, req, check)
|
2018-06-14 05:37:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// newUploadRequest creates a new h.Request for uploading
|
2019-11-18 15:34:17 +02:00
|
|
|
func newUploadRequest(method, target, username, secret string, headers map[string]string, a *asset) (*h.Request, error) {
|
|
|
|
req, err := h.NewRequest(method, 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)
|
|
|
|
|
2018-10-01 21:52:16 +02:00
|
|
|
for k, v := range headers {
|
|
|
|
req.Header.Add(k, v)
|
|
|
|
}
|
|
|
|
|
2018-06-14 05:37:48 +02:00
|
|
|
return req, err
|
|
|
|
}
|
|
|
|
|
2019-11-18 15:34:17 +02:00
|
|
|
func getHTTPClient(upload *config.Upload) (*h.Client, error) {
|
|
|
|
if upload.TrustedCerts == "" {
|
2018-09-08 21:33:46 +02:00
|
|
|
return h.DefaultClient, nil
|
|
|
|
}
|
2018-09-14 01:07:13 +02:00
|
|
|
pool, err := x509.SystemCertPool()
|
2018-09-08 21:33:46 +02:00
|
|
|
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
|
|
|
|
}
|
2018-09-08 21:33:46 +02:00
|
|
|
}
|
2019-11-18 15:34:17 +02:00
|
|
|
pool.AppendCertsFromPEM([]byte(upload.TrustedCerts)) // already validated certs checked by CheckConfig
|
2018-09-08 21:33:46 +02:00
|
|
|
return &h.Client{
|
|
|
|
Transport: &h.Transport{
|
|
|
|
TLSClientConfig: &tls.Config{
|
|
|
|
RootCAs: pool,
|
2018-08-28 23:40:46 +02:00
|
|
|
},
|
2018-09-08 21:33:46 +02:00
|
|
|
},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2018-06-14 05:37:48 +02:00
|
|
|
// executeHTTPRequest processes the http call with respect of context ctx
|
2019-11-18 15:34:17 +02:00
|
|
|
func executeHTTPRequest(ctx *context.Context, upload *config.Upload, req *h.Request, check ResponseChecker) (*h.Response, error) {
|
|
|
|
client, err := getHTTPClient(upload)
|
2018-09-08 21:33:46 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2018-08-28 23:40:46 +02:00
|
|
|
}
|
2018-09-14 02:53:05 +02:00
|
|
|
log.Debugf("executing request: %s %s (headers: %v)", req.Method, req.URL, req.Header)
|
2018-08-28 23:40:46 +02:00
|
|
|
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():
|
2018-09-10 14:08:54 +02:00
|
|
|
return nil, ctx.Err()
|
2018-06-14 05:37:48 +02:00
|
|
|
default:
|
|
|
|
}
|
2018-09-10 14:08:54 +02:00
|
|
|
return nil, err
|
2018-06-14 05:37:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
defer resp.Body.Close() // nolint: errcheck
|
|
|
|
|
2018-09-10 14:08:54 +02:00
|
|
|
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
|
2018-09-10 14:08:54 +02:00
|
|
|
return resp, err
|
2018-06-14 05:37:48 +02:00
|
|
|
}
|
|
|
|
|
2018-09-10 14:08:54 +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
|
2019-05-17 15:25:01 +02:00
|
|
|
// TODO: replace this with our internal template pkg
|
2019-11-18 15:34:17 +02:00
|
|
|
func resolveTargetTemplate(ctx *context.Context, upload *config.Upload, 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,
|
|
|
|
}
|
|
|
|
|
2019-11-18 15:34:17 +02:00
|
|
|
if upload.Mode == ModeBinary {
|
2019-05-29 14:34:44 +02:00
|
|
|
// TODO: multiple archives here
|
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
|
2019-11-18 15:34:17 +02:00
|
|
|
t, err := template.New(ctx.Config.ProjectName).Parse(upload.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
|
|
|
|
}
|