1
0
mirror of https://github.com/goreleaser/goreleaser.git synced 2025-01-10 03:47:03 +02:00
goreleaser/internal/pipe/chocolatey/chocolatey_test.go
Fabio Ribeiro f2281e8ff2
feat: chocolatey support (#3509)
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>
2022-11-11 23:52:32 -03:00

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()
}