1
0
mirror of https://github.com/goreleaser/goreleaser.git synced 2025-01-08 03:31:59 +02:00
goreleaser/internal/client/gitea.go
6543 1bb832a764
chore(deps): bump code.gitea.io/sdk/gitea from v0.12.1 to v0.13.1 (#1828)
* chore(deps): bump code.gitea.io/sdk/gitea from v0.12.1 to v0.13.1

* only set Context if not nil
2020-10-02 02:06:21 +00:00

219 lines
5.3 KiB
Go

package client
import (
"crypto/tls"
"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
}
// NewGitea returns a gitea client implementation.
func NewGitea(ctx *context.Context, token string) (Client, error) {
instanceURL, err := getInstanceURL(ctx.Config.GiteaURLs.API)
if err != nil {
return nil, err
}
transport := &http.Transport{
TLSClientConfig: &tls.Config{
// nolint: gosec
InsecureSkipVerify: ctx.Config.GiteaURLs.SkipTLSVerify,
},
}
httpClient := &http.Client{Transport: transport}
client, err := gitea.NewClient(instanceURL,
gitea.SetToken(token),
gitea.SetHTTPClient(httpClient),
)
if err != nil {
return nil, err
}
if ctx != nil {
gitea.SetContext(ctx)(client)
}
return &giteaClient{client: client}, nil
}
// CloseMilestone closes a given milestone.
func (c *giteaClient) CloseMilestone(ctx *context.Context, repo Repo, title string) error {
closedState := gitea.StateClosed
opts := gitea.EditMilestoneOption{
State: &closedState,
Title: title,
}
_, resp, err := c.client.EditMilestoneByName(repo.Owner, repo.Name, title, opts)
if resp != nil && resp.StatusCode == http.StatusNotFound {
return ErrNoMilestoneFound{Title: title}
}
return err
}
// CreateFile creates a file in the repository at a given path
// or updates the file if it exists.
func (c *giteaClient) CreateFile(
ctx *context.Context,
commitAuthor config.CommitAuthor,
repo Repo,
content []byte,
path,
message string,
) error {
//TODO: implement for brew and scoop support for Gitea-hosted repos
return nil
}
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
opts := gitea.CreateReleaseOption{
TagName: tag,
Target: ctx.Git.Commit,
Title: title,
Note: body,
IsDraft: releaseConfig.Draft,
IsPrerelease: ctx.PreRelease,
}
release, _, err := c.client.CreateRelease(owner, repoName, opts)
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) {
releases, _, err := c.client.ListReleases(owner, repoName, gitea.ListReleasesOptions{})
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
opts := gitea.EditReleaseOption{
TagName: tag,
Target: ctx.Git.Commit,
Title: title,
Note: body,
IsDraft: &releaseConfig.Draft,
IsPrerelease: &ctx.PreRelease,
}
release, _, err := c.client.EditRelease(owner, repoName, id, opts)
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
// the release notes if it exists.
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 {
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
}
func (c *giteaClient) ReleaseURLTemplate(ctx *context.Context) (string, error) {
return "", NotImplementedError{TokenType: context.TokenTypeGitea}
}
// Upload uploads a file into a release repository.
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
_, _, err = c.client.CreateReleaseAttachment(owner, repoName, giteaReleaseID, file, artifact.Name)
if err != nil {
return RetriableError{err}
}
return nil
}