2019-08-26 09:31:38 +02:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2020-05-26 05:48:10 +02:00
|
|
|
// NewGitea returns a gitea client implementation.
|
2020-07-06 22:12:41 +02:00
|
|
|
func NewGitea(ctx *context.Context, token string) (Client, error) {
|
2019-08-26 09:31:38 +02:00
|
|
|
instanceURL, err := getInstanceURL(ctx.Config.GiteaURLs.API)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-07-06 22:12:41 +02:00
|
|
|
client := gitea.NewClient(instanceURL, token)
|
2019-08-26 09:31:38 +02:00
|
|
|
transport := &http.Transport{
|
|
|
|
TLSClientConfig: &tls.Config{
|
|
|
|
// nolint: gosec
|
|
|
|
InsecureSkipVerify: ctx.Config.GiteaURLs.SkipTLSVerify,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
httpClient := &http.Client{Transport: transport}
|
|
|
|
client.SetHTTPClient(httpClient)
|
|
|
|
return &giteaClient{client: client}, nil
|
|
|
|
}
|
|
|
|
|
2020-07-09 22:40:37 +02:00
|
|
|
// CloseMilestone closes a given milestone.
|
|
|
|
func (c *giteaClient) 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}
|
|
|
|
}
|
|
|
|
|
|
|
|
closedState := string(gitea.StateClosed)
|
|
|
|
|
|
|
|
opts := gitea.EditMilestoneOption{
|
|
|
|
Deadline: milestone.Deadline,
|
|
|
|
Description: &milestone.Description,
|
|
|
|
State: &closedState,
|
|
|
|
Title: milestone.Title,
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = c.client.EditMilestone(repo.Owner, repo.Name, milestone.ID, opts)
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-08-26 09:31:38 +02:00
|
|
|
// CreateFile creates a file in the repository at a given path
|
2020-05-26 05:48:10 +02:00
|
|
|
// or updates the file if it exists.
|
2019-08-26 09:31:38 +02:00
|
|
|
func (c *giteaClient) CreateFile(
|
|
|
|
ctx *context.Context,
|
|
|
|
commitAuthor config.CommitAuthor,
|
2020-07-06 22:12:41 +02:00
|
|
|
repo Repo,
|
2019-08-26 09:31:38 +02:00
|
|
|
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
|
|
|
|
|
2019-10-29 03:33:01 +02:00
|
|
|
opts := gitea.CreateReleaseOption{
|
2019-08-26 09:31:38 +02:00
|
|
|
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) {
|
2020-05-22 15:21:43 +02:00
|
|
|
releases, err := c.client.ListReleases(owner, repoName, gitea.ListReleasesOptions{})
|
2019-08-26 09:31:38 +02: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 03:33:01 +02:00
|
|
|
opts := gitea.EditReleaseOption{
|
2019-08-26 09:31:38 +02:00
|
|
|
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
|
2020-05-26 05:48:10 +02:00
|
|
|
// the release notes if it exists.
|
2019-08-26 09:31:38 +02: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 {
|
|
|
|
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 15:48:17 +02:00
|
|
|
func (c *giteaClient) ReleaseURLTemplate(ctx *context.Context) (string, error) {
|
|
|
|
return "", NotImplementedError{TokenType: context.TokenTypeGitea}
|
|
|
|
}
|
|
|
|
|
2020-05-26 05:48:10 +02:00
|
|
|
// Upload uploads a file into a release repository.
|
2019-08-26 09:31:38 +02: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
|
|
|
|
|
|
|
|
_, err = c.client.CreateReleaseAttachment(owner, repoName, giteaReleaseID, file, artifact.Name)
|
2020-03-22 22:03:31 +02:00
|
|
|
if err != nil {
|
|
|
|
return RetriableError{err}
|
|
|
|
}
|
|
|
|
return nil
|
2019-08-26 09:31:38 +02:00
|
|
|
}
|
2020-07-09 22:40:37 +02:00
|
|
|
|
|
|
|
// getMilestoneByTitle returns a milestone by title.
|
|
|
|
func (c *giteaClient) getMilestoneByTitle(repo Repo, title string) (*gitea.Milestone, error) {
|
|
|
|
// The Gitea API/SDK does not provide lookup by title functionality currently.
|
|
|
|
milestones, err := c.client.ListRepoMilestones(repo.Owner, repo.Name, gitea.ListMilestoneOption{})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, milestone := range milestones {
|
|
|
|
if milestone.Title == title {
|
|
|
|
return milestone, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
}
|