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"
|
2022-06-21 21:11:15 -03:00
|
|
|
"github.com/caarlos0/log"
|
2019-08-26 10:31:38 +03:00
|
|
|
"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
|
|
|
|
}
|
|
|
|
|
2023-04-06 22:58:06 -03:00
|
|
|
var _ Client = &giteaClient{}
|
|
|
|
|
2021-09-09 03:42:13 +02:00
|
|
|
func getInstanceURL(ctx *context.Context) (string, error) {
|
|
|
|
apiURL, err := tmpl.New(ctx).Apply(ctx.Config.GiteaURLs.API)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("templating Gitea API URL: %w", err)
|
|
|
|
}
|
|
|
|
|
2019-08-26 10:31:38 +03:00
|
|
|
u, err := url.Parse(apiURL)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
u.Path = ""
|
|
|
|
rawurl := u.String()
|
|
|
|
if rawurl == "" {
|
2023-07-03 19:01:11 +00:00
|
|
|
return "", fmt.Errorf("invalid URL: %q", ctx.Config.GiteaURLs.API)
|
2019-08-26 10:31:38 +03:00
|
|
|
}
|
|
|
|
return rawurl, nil
|
|
|
|
}
|
|
|
|
|
2023-04-06 22:58:06 -03:00
|
|
|
// newGitea returns a gitea client implementation.
|
|
|
|
func newGitea(ctx *context.Context, token string) (*giteaClient, error) {
|
2021-09-09 03:42:13 +02:00
|
|
|
instanceURL, err := getInstanceURL(ctx)
|
2019-08-26 10:31:38 +03:00
|
|
|
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}
|
2023-07-14 00:33:34 +00:00
|
|
|
options := []gitea.ClientOption{
|
2020-10-02 04:06:21 +02:00
|
|
|
gitea.SetHTTPClient(httpClient),
|
2023-07-14 00:33:34 +00:00
|
|
|
}
|
|
|
|
if token != "giteatoken" { // token used in tests
|
|
|
|
options = append(options, gitea.SetToken(token))
|
|
|
|
}
|
|
|
|
client, err := gitea.NewClient(instanceURL, options...)
|
2020-10-02 04:06:21 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if ctx != nil {
|
2022-01-10 11:18:26 -03:00
|
|
|
if err := gitea.SetContext(ctx)(client); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-10-02 04:06:21 +02:00
|
|
|
}
|
2019-08-26 10:31:38 +03:00
|
|
|
return &giteaClient{client: client}, nil
|
|
|
|
}
|
|
|
|
|
2023-03-19 22:17:18 -03:00
|
|
|
func (c *giteaClient) Changelog(_ *context.Context, _ Repo, _, _ string) (string, error) {
|
2021-10-04 09:32:30 -03:00
|
|
|
return "", ErrNotImplemented
|
|
|
|
}
|
|
|
|
|
2020-07-09 16:40:37 -04:00
|
|
|
// CloseMilestone closes a given milestone.
|
2023-03-19 22:17:18 -03:00
|
|
|
func (c *giteaClient) CloseMilestone(_ *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
|
|
|
|
}
|
|
|
|
|
2023-04-06 22:58:06 -03:00
|
|
|
func (c *giteaClient) getDefaultBranch(_ *context.Context, repo Repo) (string, error) {
|
2021-10-03 10:22:26 -04:00
|
|
|
projectID := repo.String()
|
|
|
|
p, res, err := c.client.GetRepo(repo.Owner, repo.Name)
|
|
|
|
if err != nil {
|
2023-05-02 09:06:35 -03:00
|
|
|
log.WithField("projectID", projectID).
|
|
|
|
WithField("statusCode", res.StatusCode).
|
|
|
|
WithError(err).
|
|
|
|
Warn("error checking for default branch")
|
2021-10-03 10:22:26 -04:00
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return p.DefaultBranch, nil
|
|
|
|
}
|
|
|
|
|
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
|
2021-10-03 10:22:26 -04:00
|
|
|
var branch string
|
|
|
|
var err error
|
|
|
|
if repo.Branch != "" {
|
|
|
|
branch = repo.Branch
|
|
|
|
} else {
|
2023-04-06 22:58:06 -03:00
|
|
|
branch, err = c.getDefaultBranch(ctx, repo)
|
2021-10-03 10:22:26 -04:00
|
|
|
if err != nil {
|
|
|
|
// Fall back to 'master' 😭
|
2023-05-02 09:06:35 -03:00
|
|
|
log.WithField("fileName", path).
|
|
|
|
WithField("projectID", repo.String()).
|
|
|
|
WithField("requestedBranch", branch).
|
|
|
|
WithError(err).
|
|
|
|
Warn("error checking for default branch, using master")
|
2021-10-03 10:22:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2020-11-21 14:36:29 +01:00
|
|
|
|
|
|
|
fileOptions := gitea.FileOptions{
|
|
|
|
Message: message,
|
2021-10-03 10:22:26 -04:00
|
|
|
BranchName: branch,
|
2020-11-21 14:36:29 +01:00
|
|
|
Author: gitea.Identity{
|
|
|
|
Name: commitAuthor.Name,
|
|
|
|
Email: commitAuthor.Email,
|
|
|
|
},
|
|
|
|
Committer: gitea.Identity{
|
|
|
|
Name: commitAuthor.Name,
|
|
|
|
Email: commitAuthor.Email,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2023-05-19 14:10:06 +00:00
|
|
|
log.
|
|
|
|
WithField("repository", repo.String()).
|
|
|
|
WithField("name", repo.Name).
|
|
|
|
WithField("name", repo.Name).
|
|
|
|
Info("pushing")
|
|
|
|
|
2021-10-03 10:22:26 -04:00
|
|
|
currentFile, resp, err := c.client.GetContents(repo.Owner, repo.Name, branch, path)
|
2020-11-21 14:36:29 +01:00
|
|
|
// 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 {
|
2023-05-02 09:06:35 -03:00
|
|
|
log.WithError(err).Debug("error creating Gitea release")
|
2019-08-26 10:31:38 +03:00
|
|
|
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 {
|
2023-05-02 09:06:35 -03:00
|
|
|
log.WithError(err).Debug("error updating Gitea release")
|
2019-08-26 10:31:38 +03:00
|
|
|
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-11-26 09:59:15 -03:00
|
|
|
body = getReleaseNotes(release.Note, body, ctx.Config.Release.ReleaseNotesMode)
|
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) {
|
2021-09-09 03:42:13 +02:00
|
|
|
downloadURL, err := tmpl.New(ctx).Apply(ctx.Config.GiteaURLs.Download)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("templating Gitea download URL: %w", err)
|
|
|
|
}
|
|
|
|
|
2020-11-21 14:36:29 +01:00
|
|
|
return fmt.Sprintf(
|
|
|
|
"%s/%s/%s/releases/download/{{ .Tag }}/{{ .ArtifactName }}",
|
2021-09-09 03:42:13 +02:00
|
|
|
downloadURL,
|
2020-11-21 14:36:29 +01:00
|
|
|
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
|
|
|
}
|