2019-08-26 10:31:38 +03:00
|
|
|
package client
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
2020-11-21 14:36:29 +01:00
|
|
|
"encoding/base64"
|
2019-08-26 10:31:38 +03:00
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"os"
|
|
|
|
"strconv"
|
|
|
|
|
|
|
|
"code.gitea.io/sdk/gitea"
|
|
|
|
"github.com/apex/log"
|
|
|
|
"github.com/goreleaser/goreleaser/internal/artifact"
|
|
|
|
"github.com/goreleaser/goreleaser/internal/tmpl"
|
|
|
|
"github.com/goreleaser/goreleaser/pkg/config"
|
|
|
|
"github.com/goreleaser/goreleaser/pkg/context"
|
|
|
|
)
|
|
|
|
|
|
|
|
type giteaClient struct {
|
|
|
|
client *gitea.Client
|
|
|
|
}
|
|
|
|
|
|
|
|
func getInstanceURL(apiURL string) (string, error) {
|
|
|
|
u, err := url.Parse(apiURL)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
u.Path = ""
|
|
|
|
rawurl := u.String()
|
|
|
|
if rawurl == "" {
|
|
|
|
return "", fmt.Errorf("invalid URL: %v", apiURL)
|
|
|
|
}
|
|
|
|
return rawurl, nil
|
|
|
|
}
|
|
|
|
|
2020-05-26 00:48:10 -03:00
|
|
|
// NewGitea returns a gitea client implementation.
|
2020-07-06 21:12:41 +01:00
|
|
|
func NewGitea(ctx *context.Context, token string) (Client, error) {
|
2019-08-26 10:31:38 +03:00
|
|
|
instanceURL, err := getInstanceURL(ctx.Config.GiteaURLs.API)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
transport := &http.Transport{
|
2020-11-05 05:16:25 -03:00
|
|
|
Proxy: http.ProxyFromEnvironment,
|
2019-08-26 10:31:38 +03:00
|
|
|
TLSClientConfig: &tls.Config{
|
|
|
|
// nolint: gosec
|
|
|
|
InsecureSkipVerify: ctx.Config.GiteaURLs.SkipTLSVerify,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
httpClient := &http.Client{Transport: transport}
|
2020-10-02 04:06:21 +02:00
|
|
|
client, err := gitea.NewClient(instanceURL,
|
|
|
|
gitea.SetToken(token),
|
|
|
|
gitea.SetHTTPClient(httpClient),
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if ctx != nil {
|
|
|
|
gitea.SetContext(ctx)(client)
|
|
|
|
}
|
2019-08-26 10:31:38 +03:00
|
|
|
return &giteaClient{client: client}, nil
|
|
|
|
}
|
|
|
|
|
2020-07-09 16:40:37 -04:00
|
|
|
// CloseMilestone closes a given milestone.
|
|
|
|
func (c *giteaClient) CloseMilestone(ctx *context.Context, repo Repo, title string) error {
|
2020-10-02 04:06:21 +02:00
|
|
|
closedState := gitea.StateClosed
|
|
|
|
opts := gitea.EditMilestoneOption{
|
|
|
|
State: &closedState,
|
|
|
|
Title: title,
|
2020-07-09 16:40:37 -04:00
|
|
|
}
|
|
|
|
|
2020-10-02 04:06:21 +02:00
|
|
|
_, resp, err := c.client.EditMilestoneByName(repo.Owner, repo.Name, title, opts)
|
|
|
|
if resp != nil && resp.StatusCode == http.StatusNotFound {
|
2020-07-09 16:40:37 -04:00
|
|
|
return ErrNoMilestoneFound{Title: title}
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-08-26 10:31:38 +03:00
|
|
|
// CreateFile creates a file in the repository at a given path
|
2020-05-26 00:48:10 -03:00
|
|
|
// or updates the file if it exists.
|
2019-08-26 10:31:38 +03:00
|
|
|
func (c *giteaClient) CreateFile(
|
|
|
|
ctx *context.Context,
|
|
|
|
commitAuthor config.CommitAuthor,
|
2020-07-06 21:12:41 +01:00
|
|
|
repo Repo,
|
2019-08-26 10:31:38 +03:00
|
|
|
content []byte,
|
|
|
|
path,
|
|
|
|
message string,
|
|
|
|
) error {
|
2020-11-21 14:36:29 +01:00
|
|
|
// use default branch
|
|
|
|
branchName := ""
|
|
|
|
|
|
|
|
fileOptions := gitea.FileOptions{
|
|
|
|
Message: message,
|
|
|
|
BranchName: branchName,
|
|
|
|
Author: gitea.Identity{
|
|
|
|
Name: commitAuthor.Name,
|
|
|
|
Email: commitAuthor.Email,
|
|
|
|
},
|
|
|
|
Committer: gitea.Identity{
|
|
|
|
Name: commitAuthor.Name,
|
|
|
|
Email: commitAuthor.Email,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
currentFile, resp, err := c.client.GetContents(repo.Owner, repo.Name, branchName, path)
|
|
|
|
// file not exist, create it
|
|
|
|
if err != nil {
|
|
|
|
if resp == nil || resp.StatusCode != http.StatusNotFound {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
_, _, err = c.client.CreateFile(repo.Owner, repo.Name, path, gitea.CreateFileOptions{
|
|
|
|
FileOptions: fileOptions,
|
|
|
|
Content: base64.StdEncoding.EncodeToString(content),
|
|
|
|
})
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// update file
|
|
|
|
_, _, err = c.client.UpdateFile(repo.Owner, repo.Name, path, gitea.UpdateFileOptions{
|
|
|
|
FileOptions: fileOptions,
|
|
|
|
SHA: currentFile.SHA,
|
|
|
|
Content: base64.StdEncoding.EncodeToString(content),
|
|
|
|
})
|
|
|
|
return err
|
2019-08-26 10:31:38 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *giteaClient) createRelease(ctx *context.Context, title, body string) (*gitea.Release, error) {
|
|
|
|
releaseConfig := ctx.Config.Release
|
|
|
|
owner := releaseConfig.Gitea.Owner
|
|
|
|
repoName := releaseConfig.Gitea.Name
|
|
|
|
tag := ctx.Git.CurrentTag
|
|
|
|
|
2019-10-29 02:33:01 +01:00
|
|
|
opts := gitea.CreateReleaseOption{
|
2019-08-26 10:31:38 +03:00
|
|
|
TagName: tag,
|
|
|
|
Target: ctx.Git.Commit,
|
|
|
|
Title: title,
|
|
|
|
Note: body,
|
|
|
|
IsDraft: releaseConfig.Draft,
|
|
|
|
IsPrerelease: ctx.PreRelease,
|
|
|
|
}
|
2020-10-02 04:06:21 +02:00
|
|
|
release, _, err := c.client.CreateRelease(owner, repoName, opts)
|
2019-08-26 10:31:38 +03:00
|
|
|
if err != nil {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"err": err.Error(),
|
|
|
|
}).Debug("error creating Gitea release")
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
log.WithField("id", release.ID).Info("Gitea release created")
|
|
|
|
return release, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *giteaClient) getExistingRelease(owner, repoName, tagName string) (*gitea.Release, error) {
|
2020-10-02 04:06:21 +02:00
|
|
|
releases, _, err := c.client.ListReleases(owner, repoName, gitea.ListReleasesOptions{})
|
2019-08-26 10:31:38 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, release := range releases {
|
|
|
|
if release.TagName == tagName {
|
|
|
|
return release, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *giteaClient) updateRelease(ctx *context.Context, title, body string, id int64) (*gitea.Release, error) {
|
|
|
|
releaseConfig := ctx.Config.Release
|
|
|
|
owner := releaseConfig.Gitea.Owner
|
|
|
|
repoName := releaseConfig.Gitea.Name
|
|
|
|
tag := ctx.Git.CurrentTag
|
|
|
|
|
2019-10-29 02:33:01 +01:00
|
|
|
opts := gitea.EditReleaseOption{
|
2019-08-26 10:31:38 +03:00
|
|
|
TagName: tag,
|
|
|
|
Target: ctx.Git.Commit,
|
|
|
|
Title: title,
|
|
|
|
Note: body,
|
|
|
|
IsDraft: &releaseConfig.Draft,
|
|
|
|
IsPrerelease: &ctx.PreRelease,
|
|
|
|
}
|
|
|
|
|
2020-10-02 04:06:21 +02:00
|
|
|
release, _, err := c.client.EditRelease(owner, repoName, id, opts)
|
2019-08-26 10:31:38 +03:00
|
|
|
if err != nil {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"err": err.Error(),
|
|
|
|
}).Debug("error updating Gitea release")
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
log.WithField("id", release.ID).Info("Gitea release updated")
|
|
|
|
return release, 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-08-26 10:31:38 +03:00
|
|
|
func (c *giteaClient) CreateRelease(ctx *context.Context, body string) (string, error) {
|
|
|
|
var release *gitea.Release
|
|
|
|
var err error
|
|
|
|
|
|
|
|
releaseConfig := ctx.Config.Release
|
|
|
|
|
|
|
|
title, err := tmpl.New(ctx).Apply(releaseConfig.NameTemplate)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
release, err = c.getExistingRelease(
|
|
|
|
releaseConfig.Gitea.Owner,
|
|
|
|
releaseConfig.Gitea.Name,
|
|
|
|
ctx.Git.CurrentTag,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
if release != nil {
|
2021-05-31 01:25:29 +00:00
|
|
|
if release.Note != "" {
|
|
|
|
body = release.Note
|
|
|
|
}
|
2019-08-26 10:31:38 +03:00
|
|
|
release, err = c.updateRelease(ctx, title, body, release.ID)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
release, err = c.createRelease(ctx, title, body)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return strconv.FormatInt(release.ID, 10), nil
|
|
|
|
}
|
|
|
|
|
2020-07-06 14:48:17 +01:00
|
|
|
func (c *giteaClient) ReleaseURLTemplate(ctx *context.Context) (string, error) {
|
2020-11-21 14:36:29 +01:00
|
|
|
return fmt.Sprintf(
|
|
|
|
"%s/%s/%s/releases/download/{{ .Tag }}/{{ .ArtifactName }}",
|
|
|
|
ctx.Config.GiteaURLs.Download,
|
|
|
|
ctx.Config.Release.Gitea.Owner,
|
|
|
|
ctx.Config.Release.Gitea.Name,
|
|
|
|
), 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-08-26 10:31:38 +03:00
|
|
|
func (c *giteaClient) Upload(
|
|
|
|
ctx *context.Context,
|
|
|
|
releaseID string,
|
|
|
|
artifact *artifact.Artifact,
|
|
|
|
file *os.File,
|
|
|
|
) error {
|
|
|
|
giteaReleaseID, err := strconv.ParseInt(releaseID, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
releaseConfig := ctx.Config.Release
|
|
|
|
owner := releaseConfig.Gitea.Owner
|
|
|
|
repoName := releaseConfig.Gitea.Name
|
|
|
|
|
2020-10-02 04:06:21 +02:00
|
|
|
_, _, err = c.client.CreateReleaseAttachment(owner, repoName, giteaReleaseID, file, artifact.Name)
|
2020-03-22 17:03:31 -03:00
|
|
|
if err != nil {
|
|
|
|
return RetriableError{err}
|
|
|
|
}
|
|
|
|
return nil
|
2019-08-26 10:31:38 +03:00
|
|
|
}
|