mirror of
https://github.com/goreleaser/goreleaser.git
synced 2025-01-10 03:47:03 +02:00
f2281e8ff2
This PR adds support for generating the structure used to pack and push Chocolatey Packages. And will solve the #3154 Is not ready for merge yet, but has the main structure, and ready for comments. Accordingly to Chocolatey, in order to build a package, it's necessary a `.nuspec` and `chocolateyinstall.ps1` files at least, having these ones, we could pack and distribute without adding the binary inside the final package and that was implemented here. To complete, will be necessary to define the package build and distribute, however will be required to have Chocolatey installed (Windows Only). One of alternatives that I thought was, publish the files like Scoop and Brew in a separate repository, and there we could use `chocolatey` through [crazy-max/ghaction-chocolatey](https://github.com/crazy-max/ghaction-chocolatey). Chocolatey has a lot of good examples of repositories: https://github.com/chocolatey-community/chocolatey-packages/tree/master/automatic/curl A final compilation of the missing parts: - [x] How to pack and push (chocolatey) - [x] Documentation Sorry for the long description😄 All feedback very welcome! Co-authored-by: Carlos Alexandro Becker <caarlos0@users.noreply.github.com>
331 lines
7.2 KiB
Go
331 lines
7.2 KiB
Go
package chocolatey
|
|
|
|
import (
|
|
"errors"
|
|
"os"
|
|
"path/filepath"
|
|
"testing"
|
|
|
|
"github.com/goreleaser/goreleaser/internal/artifact"
|
|
"github.com/goreleaser/goreleaser/internal/client"
|
|
"github.com/goreleaser/goreleaser/internal/golden"
|
|
"github.com/goreleaser/goreleaser/internal/testlib"
|
|
"github.com/goreleaser/goreleaser/pkg/config"
|
|
"github.com/goreleaser/goreleaser/pkg/context"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestDescription(t *testing.T) {
|
|
require.NotEmpty(t, Pipe{}.String())
|
|
}
|
|
|
|
func TestSkip(t *testing.T) {
|
|
ctx := context.New(config.Project{})
|
|
require.True(t, Pipe{}.Skip(ctx))
|
|
}
|
|
|
|
func TestDefault(t *testing.T) {
|
|
testlib.Mktmp(t)
|
|
|
|
ctx := &context.Context{
|
|
TokenType: context.TokenTypeGitHub,
|
|
Config: config.Project{
|
|
ProjectName: "myproject",
|
|
Chocolateys: []config.Chocolatey{
|
|
{},
|
|
},
|
|
},
|
|
}
|
|
|
|
require.NoError(t, Pipe{}.Default(ctx))
|
|
require.Equal(t, ctx.Config.ProjectName, ctx.Config.Chocolateys[0].Name)
|
|
require.Equal(t, ctx.Config.ProjectName, ctx.Config.Chocolateys[0].Title)
|
|
require.Equal(t, "v1", ctx.Config.Chocolateys[0].Goamd64)
|
|
}
|
|
|
|
func Test_doRun(t *testing.T) {
|
|
folder := t.TempDir()
|
|
file := filepath.Join(folder, "archive")
|
|
require.NoError(t, os.WriteFile(file, []byte("lorem ipsum"), 0o644))
|
|
|
|
tests := []struct {
|
|
name string
|
|
choco config.Chocolatey
|
|
exec func() ([]byte, error)
|
|
published int
|
|
err string
|
|
}{
|
|
{
|
|
name: "no artifacts",
|
|
choco: config.Chocolatey{
|
|
Name: "app",
|
|
IDs: []string{"no-app"},
|
|
Goamd64: "v1",
|
|
},
|
|
err: "chocolatey requires a windows build and archive",
|
|
},
|
|
{
|
|
name: "choco command not found",
|
|
choco: config.Chocolatey{
|
|
Name: "app",
|
|
Goamd64: "v1",
|
|
},
|
|
exec: func() ([]byte, error) {
|
|
return nil, errors.New(`exec: "choco.exe": executable file not found in $PATH`)
|
|
},
|
|
err: `failed to generate chocolatey package: exec: "choco.exe": executable file not found in $PATH: `,
|
|
},
|
|
{
|
|
name: "skip publish",
|
|
choco: config.Chocolatey{
|
|
Name: "app",
|
|
Goamd64: "v1",
|
|
SkipPublish: true,
|
|
},
|
|
exec: func() ([]byte, error) {
|
|
return []byte("success"), nil
|
|
},
|
|
},
|
|
{
|
|
name: "success",
|
|
choco: config.Chocolatey{
|
|
Name: "app",
|
|
Goamd64: "v1",
|
|
},
|
|
exec: func() ([]byte, error) {
|
|
return []byte("success"), nil
|
|
},
|
|
published: 1,
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
cmd = fakeCmd{execFn: tt.exec}
|
|
t.Cleanup(func() {
|
|
cmd = stdCmd{}
|
|
})
|
|
|
|
ctx := &context.Context{
|
|
Git: context.GitInfo{
|
|
CurrentTag: "v1.0.1",
|
|
},
|
|
Version: "1.0.1",
|
|
Artifacts: artifact.New(),
|
|
Config: config.Project{
|
|
Dist: folder,
|
|
ProjectName: "run-all",
|
|
},
|
|
}
|
|
|
|
ctx.Artifacts.Add(&artifact.Artifact{
|
|
Name: "app_1.0.1_windows_amd64.zip",
|
|
Path: file,
|
|
Goos: "windows",
|
|
Goarch: "amd64",
|
|
Goamd64: "v1",
|
|
Type: artifact.UploadableArchive,
|
|
Extra: map[string]interface{}{
|
|
artifact.ExtraID: "app",
|
|
artifact.ExtraFormat: "zip",
|
|
},
|
|
})
|
|
|
|
client := client.NewMock()
|
|
got := doRun(ctx, client, tt.choco)
|
|
|
|
var err string
|
|
if got != nil {
|
|
err = got.Error()
|
|
}
|
|
if tt.err != err {
|
|
t.Errorf("Unexpected error: %s (expected %s)", err, tt.err)
|
|
}
|
|
|
|
list := ctx.Artifacts.Filter(artifact.ByType(artifact.PublishableChocolatey)).List()
|
|
require.Len(t, list, tt.published)
|
|
})
|
|
}
|
|
}
|
|
|
|
func Test_buildNuspec(t *testing.T) {
|
|
ctx := &context.Context{
|
|
Version: "1.12.3",
|
|
}
|
|
choco := config.Chocolatey{
|
|
Name: "goreleaser",
|
|
IDs: []string{},
|
|
Title: "GoReleaser",
|
|
Authors: "caarlos0",
|
|
ProjectURL: "https://goreleaser.com/",
|
|
Tags: "go docker homebrew golang package",
|
|
Summary: "Deliver Go binaries as fast and easily as possible",
|
|
Description: "GoReleaser builds Go binaries for several platforms, creates a GitHub release and then pushes a Homebrew formula to a tap repository. All that wrapped in your favorite CI.",
|
|
Dependencies: []config.ChocolateyDependency{
|
|
{ID: "nfpm"},
|
|
},
|
|
}
|
|
|
|
out, err := buildNuspec(ctx, choco)
|
|
require.NoError(t, err)
|
|
|
|
golden.RequireEqualExt(t, out, ".nuspec")
|
|
}
|
|
|
|
func Test_buildTemplate(t *testing.T) {
|
|
folder := t.TempDir()
|
|
file := filepath.Join(folder, "archive")
|
|
require.NoError(t, os.WriteFile(file, []byte("lorem ipsum"), 0o644))
|
|
|
|
ctx := &context.Context{
|
|
Version: "1.0.0",
|
|
Git: context.GitInfo{
|
|
CurrentTag: "v1.0.0",
|
|
},
|
|
}
|
|
|
|
artifacts := []*artifact.Artifact{
|
|
{
|
|
Name: "app_1.0.0_windows_386.zip",
|
|
Goos: "windows",
|
|
Goarch: "386",
|
|
Goamd64: "v1",
|
|
Path: file,
|
|
},
|
|
{
|
|
Name: "app_1.0.0_windows_amd64.zip",
|
|
Goos: "windows",
|
|
Goarch: "amd64",
|
|
Goamd64: "v1",
|
|
Path: file,
|
|
},
|
|
}
|
|
|
|
choco := config.Chocolatey{
|
|
Name: "app",
|
|
}
|
|
|
|
client := client.NewMock()
|
|
|
|
data, err := dataFor(ctx, client, choco, artifacts)
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
out, err := buildTemplate(choco.Name, scriptTemplate, data)
|
|
require.NoError(t, err)
|
|
|
|
golden.RequireEqualExt(t, out, ".script.ps1")
|
|
}
|
|
|
|
func TestPublish(t *testing.T) {
|
|
folder := t.TempDir()
|
|
file := filepath.Join(folder, "archive")
|
|
require.NoError(t, os.WriteFile(file, []byte("lorem ipsum"), 0o644))
|
|
|
|
tests := []struct {
|
|
name string
|
|
artifacts []artifact.Artifact
|
|
exec func() ([]byte, error)
|
|
skip bool
|
|
err string
|
|
}{
|
|
{
|
|
name: "skip publish",
|
|
skip: true,
|
|
err: "publishing is disabled",
|
|
},
|
|
{
|
|
name: "no artifacts",
|
|
},
|
|
{
|
|
name: "no api key",
|
|
artifacts: []artifact.Artifact{
|
|
{
|
|
Type: artifact.PublishableChocolatey,
|
|
Name: "app.1.0.1.nupkg",
|
|
Extra: map[string]interface{}{
|
|
artifact.ExtraFormat: nupkgFormat,
|
|
chocoConfigExtra: config.Chocolatey{},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "push error",
|
|
artifacts: []artifact.Artifact{
|
|
{
|
|
Type: artifact.PublishableChocolatey,
|
|
Name: "app.1.0.1.nupkg",
|
|
Extra: map[string]interface{}{
|
|
artifact.ExtraFormat: nupkgFormat,
|
|
chocoConfigExtra: config.Chocolatey{
|
|
APIKey: "abcd",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
exec: func() ([]byte, error) {
|
|
return nil, errors.New(`unable to push`)
|
|
},
|
|
err: "failed to push chocolatey package: unable to push: ",
|
|
},
|
|
{
|
|
name: "success",
|
|
artifacts: []artifact.Artifact{
|
|
{
|
|
Type: artifact.PublishableChocolatey,
|
|
Name: "app.1.0.1.nupkg",
|
|
Extra: map[string]interface{}{
|
|
artifact.ExtraFormat: nupkgFormat,
|
|
chocoConfigExtra: config.Chocolatey{
|
|
APIKey: "abcd",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
exec: func() ([]byte, error) {
|
|
return []byte("success"), nil
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
cmd = fakeCmd{execFn: tt.exec}
|
|
t.Cleanup(func() {
|
|
cmd = stdCmd{}
|
|
})
|
|
|
|
ctx := &context.Context{
|
|
SkipPublish: tt.skip,
|
|
Artifacts: artifact.New(),
|
|
}
|
|
|
|
for _, artifact := range tt.artifacts {
|
|
ctx.Artifacts.Add(&artifact)
|
|
}
|
|
|
|
got := Pipe{}.Publish(ctx)
|
|
|
|
var err string
|
|
if got != nil {
|
|
err = got.Error()
|
|
}
|
|
if tt.err != err {
|
|
t.Errorf("Unexpected error: %s (expected %s)", err, tt.err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
type fakeCmd struct {
|
|
execFn func() ([]byte, error)
|
|
}
|
|
|
|
var _ cmder = fakeCmd{}
|
|
|
|
func (f fakeCmd) Exec(ctx *context.Context, name string, args ...string) ([]byte, error) {
|
|
return f.execFn()
|
|
}
|