2017-12-02 19:41:05 +01:00
|
|
|
// Package artifactory provides a Pipe that push to artifactory
|
|
|
|
package artifactory
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
2018-06-14 00:37:48 -03:00
|
|
|
h "net/http"
|
2017-12-02 19:41:05 +01:00
|
|
|
|
2018-06-14 00:37:48 -03:00
|
|
|
"github.com/goreleaser/goreleaser/internal/http"
|
2018-09-12 14:18:01 -03:00
|
|
|
"github.com/goreleaser/goreleaser/internal/pipe"
|
2018-08-14 23:50:20 -03:00
|
|
|
"github.com/goreleaser/goreleaser/pkg/context"
|
2017-12-02 19:41:05 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// artifactoryResponse reflects the response after an upload request
|
|
|
|
// to Artifactory.
|
|
|
|
type artifactoryResponse struct {
|
|
|
|
Repo string `json:"repo,omitempty"`
|
|
|
|
Path string `json:"path,omitempty"`
|
|
|
|
Created string `json:"created,omitempty"`
|
|
|
|
CreatedBy string `json:"createdBy,omitempty"`
|
|
|
|
DownloadURI string `json:"downloadUri,omitempty"`
|
|
|
|
MimeType string `json:"mimeType,omitempty"`
|
|
|
|
Size string `json:"size,omitempty"`
|
|
|
|
Checksums artifactoryChecksums `json:"checksums,omitempty"`
|
|
|
|
OriginalChecksums artifactoryChecksums `json:"originalChecksums,omitempty"`
|
|
|
|
URI string `json:"uri,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// artifactoryChecksums reflects the checksums generated by
|
|
|
|
// Artifactory
|
|
|
|
type artifactoryChecksums struct {
|
|
|
|
SHA1 string `json:"sha1,omitempty"`
|
|
|
|
MD5 string `json:"md5,omitempty"`
|
|
|
|
SHA256 string `json:"sha256,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pipe for Artifactory
|
|
|
|
type Pipe struct{}
|
|
|
|
|
2017-12-09 20:54:04 +01:00
|
|
|
// String returns the description of the pipe
|
2017-12-08 21:50:02 +01:00
|
|
|
func (Pipe) String() string {
|
2020-03-06 01:25:09 -03:00
|
|
|
return "artifactory"
|
2017-12-02 19:41:05 +01:00
|
|
|
}
|
|
|
|
|
2017-12-09 20:54:04 +01:00
|
|
|
// Default sets the pipe defaults
|
|
|
|
func (Pipe) Default(ctx *context.Context) error {
|
2018-10-01 16:52:16 -03:00
|
|
|
for i := range ctx.Config.Artifactories {
|
|
|
|
ctx.Config.Artifactories[i].ChecksumHeader = "X-Checksum-SHA256"
|
2019-11-18 10:34:17 -03:00
|
|
|
ctx.Config.Artifactories[i].Method = h.MethodPut
|
2018-10-01 16:52:16 -03:00
|
|
|
}
|
2018-06-14 00:37:48 -03:00
|
|
|
return http.Defaults(ctx.Config.Artifactories)
|
2017-12-09 20:54:04 +01:00
|
|
|
}
|
|
|
|
|
2018-10-16 22:29:02 -03:00
|
|
|
// Publish artifacts to artifactory
|
2017-12-02 19:41:05 +01:00
|
|
|
//
|
|
|
|
// Docs: https://www.jfrog.com/confluence/display/RTF/Artifactory+REST+API#ArtifactoryRESTAPI-Example-DeployinganArtifact
|
2018-10-16 22:29:02 -03:00
|
|
|
func (Pipe) Publish(ctx *context.Context) error {
|
2017-12-10 14:23:32 +01:00
|
|
|
if len(ctx.Config.Artifactories) == 0 {
|
2018-09-12 14:18:01 -03:00
|
|
|
return pipe.Skip("artifactory section is not configured")
|
2017-12-02 19:41:05 +01:00
|
|
|
}
|
|
|
|
|
2017-12-09 18:17:37 +01:00
|
|
|
// Check requirements for every instance we have configured.
|
|
|
|
// If not fulfilled, we can skip this pipeline
|
|
|
|
for _, instance := range ctx.Config.Artifactories {
|
2018-11-07 22:04:49 -02:00
|
|
|
instance := instance
|
2018-06-14 00:37:48 -03:00
|
|
|
if skip := http.CheckConfig(ctx, &instance, "artifactory"); skip != nil {
|
2018-09-12 14:18:01 -03:00
|
|
|
return pipe.Skip(skip.Error())
|
2017-12-09 18:17:37 +01:00
|
|
|
}
|
2017-12-09 20:54:04 +01:00
|
|
|
}
|
2017-12-02 19:41:05 +01:00
|
|
|
|
2018-09-10 09:08:54 -03:00
|
|
|
return http.Upload(ctx, ctx.Config.Artifactories, "artifactory", func(res *h.Response) error {
|
2018-06-14 00:37:48 -03:00
|
|
|
if err := checkResponse(res); err != nil {
|
2018-09-10 09:08:54 -03:00
|
|
|
return err
|
2017-12-02 19:41:05 +01:00
|
|
|
}
|
2018-06-14 00:37:48 -03:00
|
|
|
var r artifactoryResponse
|
|
|
|
err := json.NewDecoder(res.Body).Decode(&r)
|
2018-09-10 09:08:54 -03:00
|
|
|
return err
|
2018-06-14 00:37:48 -03:00
|
|
|
})
|
2017-12-02 19:41:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// An ErrorResponse reports one or more errors caused by an API request.
|
|
|
|
type errorResponse struct {
|
2018-06-14 00:37:48 -03:00
|
|
|
Response *h.Response // HTTP response that caused this error
|
|
|
|
Errors []Error `json:"errors"` // more detail on individual errors
|
2017-12-02 19:41:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r *errorResponse) Error() string {
|
|
|
|
return fmt.Sprintf("%v %v: %d %+v",
|
|
|
|
r.Response.Request.Method, r.Response.Request.URL,
|
|
|
|
r.Response.StatusCode, r.Errors)
|
|
|
|
}
|
|
|
|
|
|
|
|
// An Error reports more details on an individual error in an ErrorResponse.
|
|
|
|
type Error struct {
|
|
|
|
Status int `json:"status"` // Error code
|
|
|
|
Message string `json:"message"` // Message describing the error.
|
|
|
|
}
|
|
|
|
|
|
|
|
// checkResponse checks the API response for errors, and returns them if
|
|
|
|
// present. A response is considered an error if it has a status code outside
|
|
|
|
// the 200 range.
|
|
|
|
// API error responses are expected to have either no response
|
|
|
|
// body, or a JSON response body that maps to ErrorResponse. Any other
|
|
|
|
// response body will be silently ignored.
|
2018-06-14 00:37:48 -03:00
|
|
|
func checkResponse(r *h.Response) error {
|
2017-12-02 19:41:05 +01:00
|
|
|
if c := r.StatusCode; 200 <= c && c <= 299 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
errorResponse := &errorResponse{Response: r}
|
|
|
|
data, err := ioutil.ReadAll(r.Body)
|
|
|
|
if err == nil && data != nil {
|
2017-12-02 20:46:12 +01:00
|
|
|
err := json.Unmarshal(data, errorResponse)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-12-02 19:41:05 +01:00
|
|
|
}
|
|
|
|
return errorResponse
|
|
|
|
}
|