mirror of
https://github.com/goreleaser/goreleaser.git
synced 2025-01-24 04:16:27 +02:00
c8488dc825
- Add `strings` package import to `gitea.go` - Implement `Changelog` function in `gitea.go` - Update `useGitea` constant in `changelog.go` - Add test for `useGitea` in `changelog_test.go` - Update `changelog.md` with information about `gitea` customization ref: * Server API: https://github.com/go-gitea/gitea/pull/30349 * SDK: https://gitea.com/gitea/go-sdk/pulls/659 --------- Signed-off-by: appleboy <appleboy.tw@gmail.com> Co-authored-by: Carlos Alexandro Becker <caarlos0@users.noreply.github.com>
336 lines
8.5 KiB
Go
336 lines
8.5 KiB
Go
package client
|
|
|
|
import (
|
|
"crypto/tls"
|
|
"encoding/base64"
|
|
"fmt"
|
|
"net/http"
|
|
"net/url"
|
|
"os"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"code.gitea.io/sdk/gitea"
|
|
"github.com/caarlos0/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
|
|
}
|
|
|
|
var _ Client = &giteaClient{}
|
|
|
|
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)
|
|
}
|
|
|
|
u, err := url.Parse(apiURL)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
u.Path = ""
|
|
rawurl := u.String()
|
|
if rawurl == "" {
|
|
return "", fmt.Errorf("invalid URL: %q", ctx.Config.GiteaURLs.API)
|
|
}
|
|
return rawurl, nil
|
|
}
|
|
|
|
// newGitea returns a gitea client implementation.
|
|
func newGitea(ctx *context.Context, token string) (*giteaClient, error) {
|
|
instanceURL, err := getInstanceURL(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
transport := &http.Transport{
|
|
Proxy: http.ProxyFromEnvironment,
|
|
TLSClientConfig: &tls.Config{
|
|
// nolint: gosec
|
|
InsecureSkipVerify: ctx.Config.GiteaURLs.SkipTLSVerify,
|
|
},
|
|
}
|
|
httpClient := &http.Client{Transport: transport}
|
|
options := []gitea.ClientOption{
|
|
gitea.SetHTTPClient(httpClient),
|
|
}
|
|
if token != "giteatoken" { // token used in tests
|
|
options = append(options, gitea.SetToken(token))
|
|
}
|
|
client, err := gitea.NewClient(instanceURL, options...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if ctx != nil {
|
|
if err := gitea.SetContext(ctx)(client); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
return &giteaClient{client: client}, nil
|
|
}
|
|
|
|
// Changelog fetches the changelog between two revisions.
|
|
func (c *giteaClient) Changelog(_ *context.Context, repo Repo, prev, current string) (string, error) {
|
|
result, _, err := c.client.CompareCommits(repo.Owner, repo.Name, prev, current)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
var log []string
|
|
|
|
for _, commit := range result.Commits {
|
|
log = append(log, fmt.Sprintf(
|
|
"%s: %s (@%s <%s>)",
|
|
commit.SHA[:7],
|
|
strings.Split(commit.RepoCommit.Message, "\n")[0],
|
|
commit.Author.UserName,
|
|
commit.RepoCommit.Author.Email,
|
|
))
|
|
}
|
|
return strings.Join(log, "\n"), nil
|
|
}
|
|
|
|
// CloseMilestone closes a given milestone.
|
|
func (c *giteaClient) CloseMilestone(_ *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
|
|
}
|
|
|
|
func (c *giteaClient) getDefaultBranch(_ *context.Context, repo Repo) (string, error) {
|
|
projectID := repo.String()
|
|
p, res, err := c.client.GetRepo(repo.Owner, repo.Name)
|
|
if err != nil {
|
|
log := log.WithField("projectID", projectID)
|
|
if res != nil {
|
|
log = log.WithField("statusCode", res.StatusCode)
|
|
}
|
|
log.WithError(err).
|
|
Warn("error checking for default branch")
|
|
return "", err
|
|
}
|
|
return p.DefaultBranch, nil
|
|
}
|
|
|
|
// 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 {
|
|
// use default branch
|
|
var branch string
|
|
var err error
|
|
if repo.Branch != "" {
|
|
branch = repo.Branch
|
|
} else {
|
|
branch, err = c.getDefaultBranch(ctx, repo)
|
|
if err != nil {
|
|
// Fall back to 'master' 😭
|
|
log.WithField("fileName", path).
|
|
WithField("projectID", repo.String()).
|
|
WithField("requestedBranch", branch).
|
|
WithError(err).
|
|
Warn("error checking for default branch, using master")
|
|
}
|
|
|
|
}
|
|
|
|
fileOptions := gitea.FileOptions{
|
|
Message: message,
|
|
BranchName: branch,
|
|
Author: gitea.Identity{
|
|
Name: commitAuthor.Name,
|
|
Email: commitAuthor.Email,
|
|
},
|
|
Committer: gitea.Identity{
|
|
Name: commitAuthor.Name,
|
|
Email: commitAuthor.Email,
|
|
},
|
|
}
|
|
|
|
log.
|
|
WithField("repository", repo.String()).
|
|
WithField("name", repo.Name).
|
|
WithField("name", repo.Name).
|
|
Info("pushing")
|
|
|
|
currentFile, resp, err := c.client.GetContents(repo.Owner, repo.Name, branch, 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
|
|
}
|
|
|
|
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.WithError(err).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.WithError(err).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 {
|
|
body = getReleaseNotes(release.Note, body, ctx.Config.Release.ReleaseNotesMode)
|
|
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) PublishRelease(_ *context.Context, _ string /* releaseID */) (err error) {
|
|
// TODO: Create release as draft while uploading artifacts and only publish it here.
|
|
return nil
|
|
}
|
|
|
|
func (c *giteaClient) ReleaseURLTemplate(ctx *context.Context) (string, error) {
|
|
downloadURL, err := tmpl.New(ctx).Apply(ctx.Config.GiteaURLs.Download)
|
|
if err != nil {
|
|
return "", fmt.Errorf("templating Gitea download URL: %w", err)
|
|
}
|
|
|
|
return fmt.Sprintf(
|
|
"%s/%s/%s/releases/download/{{ .Tag }}/{{ .ArtifactName }}",
|
|
downloadURL,
|
|
ctx.Config.Release.Gitea.Owner,
|
|
ctx.Config.Release.Gitea.Name,
|
|
), nil
|
|
}
|
|
|
|
// 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
|
|
}
|