2019-06-29 16:02:40 +02:00
|
|
|
package client
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
2020-07-06 14:48:17 +01:00
|
|
|
"fmt"
|
2019-06-29 16:02:40 +02:00
|
|
|
"net/http"
|
|
|
|
"os"
|
2021-10-10 09:52:46 -04:00
|
|
|
"strings"
|
2019-06-29 16:02:40 +02:00
|
|
|
|
|
|
|
"github.com/apex/log"
|
2019-08-13 20:28:03 +02:00
|
|
|
"github.com/goreleaser/goreleaser/internal/artifact"
|
2019-06-29 16:02:40 +02:00
|
|
|
"github.com/goreleaser/goreleaser/internal/tmpl"
|
|
|
|
"github.com/goreleaser/goreleaser/pkg/config"
|
|
|
|
"github.com/goreleaser/goreleaser/pkg/context"
|
|
|
|
"github.com/xanzy/go-gitlab"
|
|
|
|
)
|
|
|
|
|
2020-07-06 14:48:17 +01:00
|
|
|
const DefaultGitLabDownloadURL = "https://gitlab.com"
|
|
|
|
|
2019-06-29 16:02:40 +02:00
|
|
|
type gitlabClient struct {
|
|
|
|
client *gitlab.Client
|
|
|
|
}
|
|
|
|
|
2020-05-26 00:48:10 -03:00
|
|
|
// NewGitLab returns a gitlab client implementation.
|
2020-07-06 21:12:41 +01:00
|
|
|
func NewGitLab(ctx *context.Context, token string) (Client, error) {
|
2019-06-29 16:02:40 +02:00
|
|
|
transport := &http.Transport{
|
2020-11-05 05:16:25 -03:00
|
|
|
Proxy: http.ProxyFromEnvironment,
|
2019-06-29 16:02:40 +02:00
|
|
|
TLSClientConfig: &tls.Config{
|
|
|
|
// nolint: gosec
|
|
|
|
InsecureSkipVerify: ctx.Config.GitLabURLs.SkipTLSVerify,
|
|
|
|
},
|
|
|
|
}
|
2021-04-25 14:20:49 -03:00
|
|
|
options := []gitlab.ClientOptionFunc{
|
2020-04-21 16:24:28 -03:00
|
|
|
gitlab.WithHTTPClient(&http.Client{
|
|
|
|
Transport: transport,
|
|
|
|
}),
|
|
|
|
}
|
2019-06-29 16:02:40 +02:00
|
|
|
if ctx.Config.GitLabURLs.API != "" {
|
2021-09-09 03:42:13 +02:00
|
|
|
apiURL, err := tmpl.New(ctx).Apply(ctx.Config.GitLabURLs.API)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("templating GitLab API URL: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
options = append(options, gitlab.WithBaseURL(apiURL))
|
2020-04-21 16:24:28 -03:00
|
|
|
}
|
|
|
|
client, err := gitlab.NewClient(token, options...)
|
|
|
|
if err != nil {
|
|
|
|
return &gitlabClient{}, err
|
2019-06-29 16:02:40 +02:00
|
|
|
}
|
|
|
|
return &gitlabClient{client: client}, nil
|
|
|
|
}
|
|
|
|
|
2021-10-04 09:32:30 -03:00
|
|
|
func (c *gitlabClient) Changelog(ctx *context.Context, repo Repo, prev, current string) (string, error) {
|
2021-10-10 09:52:46 -04:00
|
|
|
cmpOpts := &gitlab.CompareOptions{
|
|
|
|
From: &prev,
|
|
|
|
To: ¤t,
|
|
|
|
}
|
|
|
|
result, _, err := c.client.Repositories.Compare(repo.String(), cmpOpts)
|
|
|
|
var log []string
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, commit := range result.Commits {
|
|
|
|
log = append(log, fmt.Sprintf(
|
|
|
|
"%s: %s (%s <%s>)",
|
|
|
|
commit.ShortID,
|
|
|
|
strings.Split(commit.Message, "\n")[0],
|
|
|
|
commit.AuthorName,
|
|
|
|
commit.AuthorEmail,
|
|
|
|
))
|
|
|
|
}
|
|
|
|
return strings.Join(log, "\n"), nil
|
2021-10-04 09:32:30 -03:00
|
|
|
}
|
|
|
|
|
2021-10-03 10:22:26 -04:00
|
|
|
// GetDefaultBranch get the default branch
|
|
|
|
func (c *gitlabClient) GetDefaultBranch(ctx *context.Context, repo Repo) (string, error) {
|
|
|
|
projectID := repo.String()
|
|
|
|
p, res, err := c.client.Projects.GetProject(projectID, nil)
|
|
|
|
if err != nil {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"projectID": projectID,
|
|
|
|
"statusCode": res.StatusCode,
|
|
|
|
"err": err.Error(),
|
|
|
|
}).Warn("error checking for default branch")
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return p.DefaultBranch, nil
|
|
|
|
}
|
|
|
|
|
2020-07-09 16:40:37 -04:00
|
|
|
// CloseMilestone closes a given milestone.
|
|
|
|
func (c *gitlabClient) CloseMilestone(ctx *context.Context, repo Repo, title string) error {
|
|
|
|
milestone, err := c.getMilestoneByTitle(repo, title)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if milestone == nil {
|
|
|
|
return ErrNoMilestoneFound{Title: title}
|
|
|
|
}
|
|
|
|
|
|
|
|
closeStateEvent := "close"
|
|
|
|
|
|
|
|
opts := &gitlab.UpdateMilestoneOptions{
|
|
|
|
Description: &milestone.Description,
|
|
|
|
DueDate: milestone.DueDate,
|
|
|
|
StartDate: milestone.StartDate,
|
|
|
|
StateEvent: &closeStateEvent,
|
|
|
|
Title: &milestone.Title,
|
|
|
|
}
|
|
|
|
|
|
|
|
_, _, err = c.client.Milestones.UpdateMilestone(
|
|
|
|
repo.String(),
|
|
|
|
milestone.ID,
|
|
|
|
opts,
|
|
|
|
)
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-08-13 20:28:03 +02:00
|
|
|
// CreateFile gets a file in the repository at a given path
|
2020-05-26 00:48:10 -03:00
|
|
|
// and updates if it exists or creates it for later pipes in the pipeline.
|
2019-06-29 16:02:40 +02:00
|
|
|
func (c *gitlabClient) CreateFile(
|
|
|
|
ctx *context.Context,
|
|
|
|
commitAuthor config.CommitAuthor,
|
2020-07-06 21:12:41 +01:00
|
|
|
repo Repo,
|
2019-08-13 20:28:03 +02:00
|
|
|
content []byte, // the content of the formula.rb
|
|
|
|
path, // the path to the formula.rb
|
|
|
|
message string, // the commit msg
|
2019-06-29 16:02:40 +02:00
|
|
|
) error {
|
2019-08-13 20:28:03 +02:00
|
|
|
fileName := path
|
2021-10-03 10:22:26 -04:00
|
|
|
projectID := repo.String()
|
|
|
|
|
|
|
|
// Use the project default branch if we can get it...otherwise, just use
|
|
|
|
// 'master'
|
|
|
|
var branch, ref string
|
|
|
|
var err error
|
|
|
|
// Use the branch if given one
|
|
|
|
if repo.Branch != "" {
|
|
|
|
branch = repo.Branch
|
|
|
|
} else {
|
|
|
|
// Try to get the default branch from the Git provider
|
|
|
|
branch, err = c.GetDefaultBranch(ctx, repo)
|
|
|
|
if err != nil {
|
|
|
|
// Fall back to 'master' 😭
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"fileName": fileName,
|
|
|
|
"projectID": repo.String(),
|
|
|
|
"requestedBranch": branch,
|
|
|
|
"err": err.Error(),
|
|
|
|
}).Warn("error checking for default branch, using master")
|
|
|
|
ref = "master"
|
|
|
|
branch = "master"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ref = branch
|
2019-08-13 20:28:03 +02:00
|
|
|
opts := &gitlab.GetFileOptions{Ref: &ref}
|
|
|
|
castedContent := string(content)
|
|
|
|
|
|
|
|
log.WithFields(log.Fields{
|
2021-10-03 10:22:26 -04:00
|
|
|
"owner": repo.Owner,
|
|
|
|
"name": repo.Name,
|
|
|
|
"ref": ref,
|
|
|
|
"branch": branch,
|
2019-08-13 20:28:03 +02:00
|
|
|
}).Debug("projectID at brew")
|
|
|
|
|
2021-10-03 10:22:26 -04:00
|
|
|
_, res, err := c.client.RepositoryFiles.GetFile(repo.String(), fileName, opts)
|
2019-08-13 20:28:03 +02:00
|
|
|
if err != nil && (res == nil || res.StatusCode != 404) {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"fileName": fileName,
|
|
|
|
"ref": ref,
|
|
|
|
"projectID": projectID,
|
|
|
|
"statusCode": res.StatusCode,
|
|
|
|
"err": err.Error(),
|
|
|
|
}).Error("error getting file for brew formula")
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"fileName": fileName,
|
|
|
|
"branch": branch,
|
|
|
|
"projectID": projectID,
|
|
|
|
}).Debug("found already existing brew formula file")
|
|
|
|
|
|
|
|
if res.StatusCode == 404 {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"fileName": fileName,
|
|
|
|
"ref": ref,
|
|
|
|
"projectID": projectID,
|
|
|
|
}).Debug("creating brew formula")
|
|
|
|
createOpts := &gitlab.CreateFileOptions{
|
|
|
|
AuthorName: &commitAuthor.Name,
|
|
|
|
AuthorEmail: &commitAuthor.Email,
|
|
|
|
Content: &castedContent,
|
|
|
|
Branch: &branch,
|
|
|
|
CommitMessage: &message,
|
|
|
|
}
|
|
|
|
fileInfo, res, err := c.client.RepositoryFiles.CreateFile(projectID, fileName, createOpts)
|
|
|
|
if err != nil {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"fileName": fileName,
|
|
|
|
"branch": branch,
|
|
|
|
"projectID": projectID,
|
|
|
|
"statusCode": res.StatusCode,
|
|
|
|
"err": err.Error(),
|
|
|
|
}).Error("error creating brew formula file")
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"fileName": fileName,
|
|
|
|
"branch": branch,
|
|
|
|
"projectID": projectID,
|
|
|
|
"filePath": fileInfo.FilePath,
|
|
|
|
}).Debug("created brew formula file")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"fileName": fileName,
|
|
|
|
"ref": ref,
|
|
|
|
"projectID": projectID,
|
|
|
|
}).Debug("updating brew formula")
|
|
|
|
updateOpts := &gitlab.UpdateFileOptions{
|
|
|
|
AuthorName: &commitAuthor.Name,
|
|
|
|
AuthorEmail: &commitAuthor.Email,
|
|
|
|
Content: &castedContent,
|
|
|
|
Branch: &branch,
|
|
|
|
CommitMessage: &message,
|
|
|
|
}
|
|
|
|
|
|
|
|
updateFileInfo, res, err := c.client.RepositoryFiles.UpdateFile(projectID, fileName, updateOpts)
|
|
|
|
if err != nil {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"fileName": fileName,
|
|
|
|
"branch": branch,
|
|
|
|
"projectID": projectID,
|
|
|
|
"statusCode": res.StatusCode,
|
|
|
|
"err": err.Error(),
|
|
|
|
}).Error("error updating brew formula file")
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"fileName": fileName,
|
|
|
|
"branch": branch,
|
|
|
|
"projectID": projectID,
|
|
|
|
"filePath": updateFileInfo.FilePath,
|
|
|
|
"statusCode": res.StatusCode,
|
|
|
|
}).Debug("updated brew formula file")
|
2019-06-29 16:02:40 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreateRelease creates a new release or updates it by keeping
|
2020-05-26 00:48:10 -03:00
|
|
|
// the release notes if it exists.
|
2019-06-29 16:02:40 +02:00
|
|
|
func (c *gitlabClient) CreateRelease(ctx *context.Context, body string) (releaseID string, err error) {
|
|
|
|
title, err := tmpl.New(ctx).Apply(ctx.Config.Release.NameTemplate)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2021-06-26 16:44:24 -03:00
|
|
|
gitlabName, err := tmpl.New(ctx).Apply(ctx.Config.Release.GitLab.Name)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
projectID := gitlabName
|
|
|
|
if ctx.Config.Release.GitLab.Owner != "" {
|
|
|
|
projectID = ctx.Config.Release.GitLab.Owner + "/" + projectID
|
|
|
|
}
|
2019-06-29 16:02:40 +02:00
|
|
|
log.WithFields(log.Fields{
|
2021-06-26 16:44:24 -03:00
|
|
|
"owner": ctx.Config.Release.GitLab.Owner,
|
|
|
|
"name": gitlabName,
|
|
|
|
"projectID": projectID,
|
2019-06-29 16:02:40 +02:00
|
|
|
}).Debug("projectID")
|
|
|
|
|
|
|
|
name := title
|
|
|
|
tagName := ctx.Git.CurrentTag
|
|
|
|
release, resp, err := c.client.Releases.GetRelease(projectID, tagName)
|
2021-09-10 21:35:42 +02:00
|
|
|
if err != nil && (resp == nil || (resp.StatusCode != 403 && resp.StatusCode != 404)) {
|
2019-06-29 16:02:40 +02:00
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
2021-07-06 10:42:56 +02:00
|
|
|
if resp.StatusCode == 403 || resp.StatusCode == 404 {
|
2019-06-29 16:02:40 +02:00
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"err": err.Error(),
|
|
|
|
}).Debug("get release")
|
|
|
|
|
|
|
|
description := body
|
|
|
|
ref := ctx.Git.Commit
|
|
|
|
gitURL := ctx.Git.URL
|
|
|
|
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"name": name,
|
|
|
|
"description": description,
|
|
|
|
"ref": ref,
|
|
|
|
"url": gitURL,
|
|
|
|
}).Debug("creating release")
|
|
|
|
release, _, err = c.client.Releases.CreateRelease(projectID, &gitlab.CreateReleaseOptions{
|
|
|
|
Name: &name,
|
|
|
|
Description: &description,
|
|
|
|
Ref: &ref,
|
|
|
|
TagName: &tagName,
|
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"err": err.Error(),
|
|
|
|
}).Debug("error create release")
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
log.WithField("name", release.Name).Info("release created")
|
|
|
|
} else {
|
|
|
|
desc := body
|
|
|
|
if release != nil && release.DescriptionHTML != "" {
|
|
|
|
desc = release.DescriptionHTML
|
|
|
|
}
|
|
|
|
|
|
|
|
release, _, err = c.client.Releases.UpdateRelease(projectID, tagName, &gitlab.UpdateReleaseOptions{
|
|
|
|
Name: &name,
|
|
|
|
Description: &desc,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"err": err.Error(),
|
|
|
|
}).Debug("error update release")
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.WithField("name", release.Name).Info("release updated")
|
|
|
|
}
|
|
|
|
|
|
|
|
return tagName, err // gitlab references a tag in a repo by its name
|
|
|
|
}
|
|
|
|
|
2020-07-06 14:48:17 +01:00
|
|
|
func (c *gitlabClient) ReleaseURLTemplate(ctx *context.Context) (string, error) {
|
2021-06-26 16:44:24 -03:00
|
|
|
var urlTemplate string
|
|
|
|
gitlabName, err := tmpl.New(ctx).Apply(ctx.Config.Release.GitLab.Name)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2021-09-09 03:42:13 +02:00
|
|
|
downloadURL, err := tmpl.New(ctx).Apply(ctx.Config.GitLabURLs.Download)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
2021-06-26 16:44:24 -03:00
|
|
|
if ctx.Config.Release.GitLab.Owner != "" {
|
|
|
|
urlTemplate = fmt.Sprintf(
|
|
|
|
"%s/%s/%s/-/releases/{{ .Tag }}/downloads/{{ .ArtifactName }}",
|
2021-09-09 03:42:13 +02:00
|
|
|
downloadURL,
|
2021-06-26 16:44:24 -03:00
|
|
|
ctx.Config.Release.GitLab.Owner,
|
|
|
|
gitlabName,
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
urlTemplate = fmt.Sprintf(
|
|
|
|
"%s/%s/-/releases/{{ .Tag }}/downloads/{{ .ArtifactName }}",
|
2021-09-09 03:42:13 +02:00
|
|
|
downloadURL,
|
2021-06-26 16:44:24 -03:00
|
|
|
gitlabName,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
return urlTemplate, nil
|
2020-07-06 14:48:17 +01:00
|
|
|
}
|
|
|
|
|
2020-05-26 00:48:10 -03:00
|
|
|
// Upload uploads a file into a release repository.
|
2019-06-29 16:02:40 +02:00
|
|
|
func (c *gitlabClient) Upload(
|
|
|
|
ctx *context.Context,
|
|
|
|
releaseID string,
|
2019-08-13 20:28:03 +02:00
|
|
|
artifact *artifact.Artifact,
|
2019-06-29 16:02:40 +02:00
|
|
|
file *os.File,
|
|
|
|
) error {
|
2021-06-26 16:44:24 -03:00
|
|
|
// create new template and apply name field
|
|
|
|
gitlabName, err := tmpl.New(ctx).Apply(ctx.Config.Release.GitLab.Name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
projectID := gitlabName
|
|
|
|
// check if owner is empty
|
|
|
|
if ctx.Config.Release.GitLab.Owner != "" {
|
|
|
|
projectID = ctx.Config.Release.GitLab.Owner + "/" + projectID
|
|
|
|
}
|
2019-06-29 16:02:40 +02:00
|
|
|
|
|
|
|
log.WithField("file", file.Name()).Debug("uploading file")
|
|
|
|
projectFile, _, err := c.client.Projects.UploadFile(
|
|
|
|
projectID,
|
|
|
|
file.Name(),
|
|
|
|
nil,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"file": file.Name(),
|
|
|
|
"url": projectFile.URL,
|
|
|
|
}).Debug("uploaded file")
|
|
|
|
|
2021-06-26 16:44:24 -03:00
|
|
|
// search for project details based on projectID
|
|
|
|
projectDetails, _, err := c.client.Projects.GetProject(projectID, nil)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-09-09 03:42:13 +02:00
|
|
|
gitlabBaseURL, err := tmpl.New(ctx).Apply(ctx.Config.GitLabURLs.Download)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("templating GitLab Download URL: %w", err)
|
|
|
|
}
|
|
|
|
|
2021-06-26 16:44:24 -03:00
|
|
|
linkURL := gitlabBaseURL + "/" + projectDetails.PathWithNamespace + projectFile.URL
|
2019-08-13 20:28:03 +02:00
|
|
|
name := artifact.Name
|
2021-05-17 18:33:04 +01:00
|
|
|
filename := "/" + name
|
2019-06-29 16:02:40 +02:00
|
|
|
releaseLink, _, err := c.client.ReleaseLinks.CreateReleaseLink(
|
|
|
|
projectID,
|
|
|
|
releaseID,
|
|
|
|
&gitlab.CreateReleaseLinkOptions{
|
2021-05-17 18:33:04 +01:00
|
|
|
Name: &name,
|
|
|
|
URL: &linkURL,
|
|
|
|
FilePath: &filename,
|
2019-06-29 16:02:40 +02:00
|
|
|
})
|
|
|
|
if err != nil {
|
2020-03-22 17:03:31 -03:00
|
|
|
return RetriableError{err}
|
2019-06-29 16:02:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"id": releaseLink.ID,
|
2021-05-17 18:33:04 +01:00
|
|
|
"url": releaseLink.DirectAssetURL,
|
2019-06-29 16:02:40 +02:00
|
|
|
}).Debug("created release link")
|
|
|
|
|
2019-08-13 20:28:03 +02:00
|
|
|
// for checksums.txt the field is nil, so we initialize it
|
|
|
|
if artifact.Extra == nil {
|
|
|
|
artifact.Extra = make(map[string]interface{})
|
|
|
|
}
|
|
|
|
|
2020-03-22 17:03:31 -03:00
|
|
|
return nil
|
2019-06-29 16:02:40 +02:00
|
|
|
}
|
2019-08-13 20:28:03 +02:00
|
|
|
|
2020-07-09 16:40:37 -04:00
|
|
|
// getMilestoneByTitle returns a milestone by title.
|
|
|
|
func (c *gitlabClient) getMilestoneByTitle(repo Repo, title string) (*gitlab.Milestone, error) {
|
|
|
|
opts := &gitlab.ListMilestonesOptions{
|
|
|
|
Title: &title,
|
|
|
|
}
|
|
|
|
|
|
|
|
for {
|
|
|
|
milestones, resp, err := c.client.Milestones.ListMilestones(repo.String(), opts)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, milestone := range milestones {
|
|
|
|
if milestone != nil && milestone.Title == title {
|
|
|
|
return milestone, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if resp.NextPage == 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
opts.Page = resp.NextPage
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
}
|