1
0
mirror of https://github.com/goreleaser/goreleaser.git synced 2025-01-26 04:22:05 +02:00
2023-06-25 05:16:37 +00:00

412 lines
10 KiB
Go

// Package krew implements Piper and Publisher, providing krew plugin manifest
// creation and upload to a repository (aka krew plugin index).
//
// nolint:tagliatelle
package krew
import (
"errors"
"fmt"
"os"
"path"
"path/filepath"
"reflect"
"sort"
"strings"
"github.com/caarlos0/log"
"github.com/goreleaser/goreleaser/internal/artifact"
"github.com/goreleaser/goreleaser/internal/client"
"github.com/goreleaser/goreleaser/internal/commitauthor"
"github.com/goreleaser/goreleaser/internal/deprecate"
"github.com/goreleaser/goreleaser/internal/pipe"
"github.com/goreleaser/goreleaser/internal/tmpl"
"github.com/goreleaser/goreleaser/internal/yaml"
"github.com/goreleaser/goreleaser/pkg/config"
"github.com/goreleaser/goreleaser/pkg/context"
)
const (
krewConfigExtra = "KrewConfig"
manifestsFolder = "plugins"
kind = "Plugin"
apiVersion = "krew.googlecontainertools.github.com/v1alpha2"
)
var ErrNoArchivesFound = errors.New("no archives found")
// Pipe for krew manifest deployment.
type Pipe struct{}
func (Pipe) String() string { return "krew plugin manifest" }
func (Pipe) ContinueOnError() bool { return true }
func (Pipe) Skip(ctx *context.Context) bool { return len(ctx.Config.Krews) == 0 }
func (Pipe) Default(ctx *context.Context) error {
for i := range ctx.Config.Krews {
krew := &ctx.Config.Krews[i]
krew.CommitAuthor = commitauthor.Default(krew.CommitAuthor)
if krew.CommitMessageTemplate == "" {
krew.CommitMessageTemplate = "Krew manifest update for {{ .ProjectName }} version {{ .Tag }}"
}
if krew.Name == "" {
krew.Name = ctx.Config.ProjectName
}
if krew.Goamd64 == "" {
krew.Goamd64 = "v1"
}
if !reflect.DeepEqual(krew.Index, config.RepoRef{}) {
krew.Repository = krew.Index
deprecate.Notice(ctx, "krews.index")
}
}
return nil
}
func (Pipe) Run(ctx *context.Context) error {
cli, err := client.New(ctx)
if err != nil {
return err
}
return runAll(ctx, cli)
}
func runAll(ctx *context.Context, cli client.Client) error {
for _, krew := range ctx.Config.Krews {
err := doRun(ctx, krew, cli)
if err != nil {
return err
}
}
return nil
}
func doRun(ctx *context.Context, krew config.Krew, cl client.ReleaserURLTemplater) error {
if krew.Name == "" {
return pipe.Skip("krew: manifest name is not set")
}
if krew.Description == "" {
return fmt.Errorf("krew: manifest description is not set")
}
if krew.ShortDescription == "" {
return fmt.Errorf("krew: manifest short description is not set")
}
filters := []artifact.Filter{
artifact.Or(
artifact.ByGoos("darwin"),
artifact.ByGoos("linux"),
artifact.ByGoos("windows"),
),
artifact.Or(
artifact.And(
artifact.ByGoarch("amd64"),
artifact.ByGoamd64(krew.Goamd64),
),
artifact.ByGoarch("arm64"),
artifact.ByGoarch("all"),
artifact.And(
artifact.ByGoarch("arm"),
artifact.ByGoarm(krew.Goarm),
),
),
artifact.ByType(artifact.UploadableArchive),
artifact.OnlyReplacingUnibins,
}
if len(krew.IDs) > 0 {
filters = append(filters, artifact.ByIDs(krew.IDs...))
}
archives := ctx.Artifacts.Filter(artifact.And(filters...)).List()
if len(archives) == 0 {
return ErrNoArchivesFound
}
krew, err := templateFields(ctx, krew)
if err != nil {
return err
}
content, err := buildmanifest(ctx, krew, cl, archives)
if err != nil {
return err
}
filename := krew.Name + ".yaml"
yamlPath := filepath.Join(ctx.Config.Dist, "krew", filename)
if err := os.MkdirAll(filepath.Dir(yamlPath), 0o755); err != nil {
return err
}
log.WithField("manifest", yamlPath).Info("writing")
if err := os.WriteFile(yamlPath, []byte("# This file was generated by GoReleaser. DO NOT EDIT.\n"+content), 0o644); err != nil { //nolint: gosec
return fmt.Errorf("failed to write krew manifest: %w", err)
}
ctx.Artifacts.Add(&artifact.Artifact{
Name: filename,
Path: yamlPath,
Type: artifact.KrewPluginManifest,
Extra: map[string]interface{}{
krewConfigExtra: krew,
},
})
return nil
}
func templateFields(ctx *context.Context, krew config.Krew) (config.Krew, error) {
t := tmpl.New(ctx)
var err error
krew.Name, err = t.Apply(krew.Name)
if err != nil {
return config.Krew{}, err
}
krew.Homepage, err = t.Apply(krew.Homepage)
if err != nil {
return config.Krew{}, err
}
krew.Description, err = t.Apply(krew.Description)
if err != nil {
return config.Krew{}, err
}
krew.Caveats, err = t.Apply(krew.Caveats)
if err != nil {
return config.Krew{}, err
}
krew.ShortDescription, err = t.Apply(krew.ShortDescription)
if err != nil {
return config.Krew{}, err
}
return krew, nil
}
func buildmanifest(
ctx *context.Context,
krew config.Krew,
client client.ReleaserURLTemplater,
artifacts []*artifact.Artifact,
) (string, error) {
data, err := manifestFor(ctx, krew, client, artifacts)
if err != nil {
return "", err
}
return doBuildManifest(data)
}
func doBuildManifest(data Manifest) (string, error) {
out, err := yaml.Marshal(data)
if err != nil {
return "", fmt.Errorf("krew: failed to marshal yaml: %w", err)
}
return string(out), nil
}
func manifestFor(
ctx *context.Context,
cfg config.Krew,
cl client.ReleaserURLTemplater,
artifacts []*artifact.Artifact,
) (Manifest, error) {
result := Manifest{
APIVersion: apiVersion,
Kind: kind,
Metadata: Metadata{
Name: cfg.Name,
},
Spec: Spec{
Homepage: cfg.Homepage,
Version: "v" + ctx.Version,
ShortDescription: cfg.ShortDescription,
Description: cfg.Description,
Caveats: cfg.Caveats,
},
}
for _, art := range artifacts {
sum, err := art.Checksum("sha256")
if err != nil {
return result, err
}
if cfg.URLTemplate == "" {
url, err := cl.ReleaseURLTemplate(ctx)
if err != nil {
return result, err
}
cfg.URLTemplate = url
}
url, err := tmpl.New(ctx).WithArtifact(art).Apply(cfg.URLTemplate)
if err != nil {
return result, err
}
goarch := []string{art.Goarch}
if art.Goarch == "all" {
goarch = []string{"amd64", "arm64"}
}
for _, arch := range goarch {
bins := artifact.ExtraOr(*art, artifact.ExtraBinaries, []string{})
if len(bins) != 1 {
return result, fmt.Errorf("krew: only one binary per archive allowed, got %d on %q", len(bins), art.Name)
}
result.Spec.Platforms = append(result.Spec.Platforms, Platform{
Bin: bins[0],
URI: url,
Sha256: sum,
Selector: Selector{
MatchLabels: MatchLabels{
Os: art.Goos,
Arch: arch,
},
},
})
}
}
sort.Slice(result.Spec.Platforms, func(i, j int) bool {
return result.Spec.Platforms[i].URI > result.Spec.Platforms[j].URI
})
return result, nil
}
// Publish krew manifest.
func (Pipe) Publish(ctx *context.Context) error {
cli, err := client.New(ctx)
if err != nil {
return err
}
return publishAll(ctx, cli)
}
func publishAll(ctx *context.Context, cli client.Client) error {
skips := pipe.SkipMemento{}
for _, manifest := range ctx.Artifacts.Filter(artifact.ByType(artifact.KrewPluginManifest)).List() {
err := doPublish(ctx, manifest, cli)
if err != nil && pipe.IsSkip(err) {
skips.Remember(err)
continue
}
if err != nil {
return err
}
}
return skips.Evaluate()
}
func doPublish(ctx *context.Context, manifest *artifact.Artifact, cl client.Client) error {
cfg, err := artifact.Extra[config.Krew](*manifest, krewConfigExtra)
if err != nil {
return err
}
if strings.TrimSpace(cfg.SkipUpload) == "true" {
return pipe.Skip("krews.skip_upload is set")
}
if strings.TrimSpace(cfg.SkipUpload) == "auto" && ctx.Semver.Prerelease != "" {
return pipe.Skip("prerelease detected with 'auto' upload, skipping krew publish")
}
ref, err := client.TemplateRef(tmpl.New(ctx).Apply, cfg.Repository)
if err != nil {
return err
}
cfg.Repository = ref
repo := client.RepoFromRef(cfg.Repository)
gpath := buildManifestPath(manifestsFolder, manifest.Name)
msg, err := tmpl.New(ctx).Apply(cfg.CommitMessageTemplate)
if err != nil {
return err
}
author, err := commitauthor.Get(ctx, cfg.CommitAuthor)
if err != nil {
return err
}
content, err := os.ReadFile(manifest.Path)
if err != nil {
return err
}
if cfg.Repository.Git.URL != "" {
return client.NewGitUploadClient(repo.Branch).
CreateFile(ctx, author, repo, content, gpath, msg)
}
cl, err = client.NewIfToken(ctx, cl, cfg.Repository.Token)
if err != nil {
return err
}
if !cfg.Repository.PullRequest.Enabled {
return cl.CreateFile(ctx, author, repo, content, gpath, msg)
}
log.Info("brews.pull_request enabled, creating a PR")
pcl, ok := cl.(client.PullRequestOpener)
if !ok {
return fmt.Errorf("client does not support pull requests")
}
if err := cl.CreateFile(ctx, author, repo, content, gpath, msg); err != nil {
return err
}
return pcl.OpenPullRequest(ctx, client.Repo{
Name: cfg.Repository.PullRequest.Base.Name,
Owner: cfg.Repository.PullRequest.Base.Owner,
Branch: cfg.Repository.PullRequest.Base.Branch,
}, repo, msg, cfg.Repository.PullRequest.Draft)
}
func buildManifestPath(folder, filename string) string {
return path.Join(folder, filename)
}
type Manifest struct {
APIVersion string `yaml:"apiVersion,omitempty"`
Kind string `yaml:"kind,omitempty"`
Metadata Metadata `yaml:"metadata,omitempty"`
Spec Spec `yaml:"spec,omitempty"`
}
type Metadata struct {
Name string `yaml:"name,omitempty"`
}
type MatchLabels struct {
Os string `yaml:"os,omitempty"`
Arch string `yaml:"arch,omitempty"`
}
type Selector struct {
MatchLabels MatchLabels `yaml:"matchLabels,omitempty"`
}
type Platform struct {
Bin string `yaml:"bin,omitempty"`
URI string `yaml:"uri,omitempty"`
Sha256 string `yaml:"sha256,omitempty"`
Selector Selector `yaml:"selector,omitempty"`
}
type Spec struct {
Version string `yaml:"version,omitempty"`
Platforms []Platform `yaml:"platforms,omitempty"`
ShortDescription string `yaml:"shortDescription,omitempty"`
Homepage string `yaml:"homepage,omitempty"`
Caveats string `yaml:"caveats,omitempty"`
Description string `yaml:"description,omitempty"`
}