1
1
Fork 0
mirror of https://github.com/goreleaser/nfpm synced 2024-05-19 13:26:22 +02:00
nfpm/arch/arch_test.go
Carlos Alexandro Becker 9c4fc0e886
feat: reproducible packages (#748)
* feat: allow to set a build date

defaults to $SOURCE_DATE_EPOCH

closes #744
closes #734

Signed-off-by: Carlos Alexandro Becker <caarlos0@users.noreply.github.com>

* fix: rename to mtime

* docs: fix systemd note

closes #739

* fix: improve arch packager

* fix: arch test

Signed-off-by: Carlos Alexandro Becker <caarlos0@users.noreply.github.com>

* fix: improve apk packager

* fix: improve deb special files

* fix: reuse keys func

* fix: deps

Signed-off-by: Carlos Alexandro Becker <caarlos0@users.noreply.github.com>

---------

Signed-off-by: Carlos Alexandro Becker <caarlos0@users.noreply.github.com>
2023-12-07 13:58:36 -03:00

391 lines
10 KiB
Go

package arch
import (
"archive/tar"
"bytes"
"fmt"
"io"
"os"
"strings"
"testing"
"time"
"github.com/goreleaser/nfpm/v2"
"github.com/goreleaser/nfpm/v2/files"
"github.com/klauspost/compress/zstd"
"github.com/klauspost/pgzip"
"github.com/stretchr/testify/require"
)
var mtime = time.Date(2023, 11, 5, 23, 15, 17, 0, time.UTC)
func exampleInfo() *nfpm.Info {
return nfpm.WithDefaults(&nfpm.Info{
Name: "foo-test",
Arch: "amd64",
Description: "Foo does things",
Priority: "extra",
Maintainer: "Carlos A Becker <pkg@carlosbecker.com>",
Version: "1.0.0",
Prerelease: "beta-1",
Section: "default",
Homepage: "http://carlosbecker.com",
Vendor: "nope",
License: "MIT",
Overridables: nfpm.Overridables{
Depends: []string{
"bash",
},
Replaces: []string{
"svn",
},
Provides: []string{
"bzr",
},
Conflicts: []string{
"zsh",
},
Contents: []*files.Content{
{
Source: "../testdata/fake",
Destination: "/usr/bin/fake",
},
{
Source: "../testdata/whatever.conf",
Destination: "/etc/fake/fake.conf",
Type: files.TypeConfig,
},
{
Destination: "/var/log/whatever",
Type: files.TypeDir,
},
{
Destination: "/usr/share/whatever",
Type: files.TypeDir,
},
{
Source: "/etc/fake/fake.conf",
Destination: "/etc/fake/fake-link.conf",
Type: files.TypeSymlink,
},
{
Source: "../testdata/something",
Destination: "/etc/something",
},
},
Scripts: nfpm.Scripts{
PreInstall: "../testdata/scripts/preinstall.sh",
PostInstall: "../testdata/scripts/postinstall.sh",
PreRemove: "../testdata/scripts/preremove.sh",
PostRemove: "../testdata/scripts/postremove.sh",
},
ArchLinux: nfpm.ArchLinux{
Scripts: nfpm.ArchLinuxScripts{
PreUpgrade: "../testdata/scripts/preupgrade.sh",
PostUpgrade: "../testdata/scripts/postupgrade.sh",
},
},
},
})
}
func TestConventionalExtension(t *testing.T) {
require.Equal(t, ".pkg.tar.zst", Default.ConventionalExtension())
}
func TestArch(t *testing.T) {
for _, arch := range []string{"386", "amd64", "arm64"} {
arch := arch
t.Run(arch, func(t *testing.T) {
info := exampleInfo()
info.Arch = arch
err := Default.Package(info, io.Discard)
require.NoError(t, err)
})
}
}
func TestArchPlatform(t *testing.T) {
f, err := os.CreateTemp(t.TempDir(), "test*.pkg.tar.zstd")
require.NoError(t, err)
t.Cleanup(func() { require.NoError(t, f.Close()) })
info := exampleInfo()
info.Platform = "darwin"
err = Default.Package(info, f)
require.Error(t, err)
}
func TestArchNoFiles(t *testing.T) {
info := exampleInfo()
info.Contents = nil
info.Scripts = nfpm.Scripts{}
info.ArchLinux = nfpm.ArchLinux{}
err := Default.Package(info, io.Discard)
require.NoError(t, err)
}
func TestArchNoInfo(t *testing.T) {
err := Default.Package(nfpm.WithDefaults(&nfpm.Info{}), io.Discard)
require.Error(t, err)
}
func TestArchConventionalFileName(t *testing.T) {
for _, arch := range []string{"386", "amd64", "arm64"} {
arch := arch
t.Run(arch, func(t *testing.T) {
info := exampleInfo()
info.Arch = arch
name := Default.ConventionalFileName(info)
require.Equal(t,
"foo-test-1.0.0beta_1-1-"+archToArchLinux[arch]+".pkg.tar.zst",
name,
)
})
}
}
func TestArchPkginfo(t *testing.T) {
info := exampleInfo()
pkginfoData, err := makeTestPkginfo(t, info)
require.NoError(t, err)
fields := extractPkginfoFields(pkginfoData)
require.Equal(t, "foo-test", fields["pkgname"])
require.Equal(t, "foo-test", fields["pkgbase"])
require.Equal(t, "1.0.0-1", fields["pkgver"])
require.Equal(t, "Foo does things", fields["pkgdesc"])
require.Equal(t, "http://carlosbecker.com", fields["url"])
require.Equal(t, "Unknown Packager", fields["packager"])
require.Equal(t, "x86_64", fields["arch"])
require.Equal(t, "MIT", fields["license"])
require.Equal(t, "1234", fields["size"])
require.Equal(t, "svn", fields["replaces"])
require.Equal(t, "zsh", fields["conflict"])
require.Equal(t, "bzr", fields["provides"])
require.Equal(t, "bash", fields["depend"])
require.Equal(t, "etc/fake/fake.conf", fields["backup"])
}
func TestArchPkgbase(t *testing.T) {
info := exampleInfo()
info.ArchLinux.Pkgbase = "foo"
pkginfoData, err := makeTestPkginfo(t, info)
require.NoError(t, err)
fields := extractPkginfoFields(pkginfoData)
require.Equal(t, "foo", fields["pkgbase"])
}
func TestArchInvalidName(t *testing.T) {
info := exampleInfo()
info.Name = "#"
_, err := makeTestPkginfo(t, info)
require.ErrorIs(t, err, ErrInvalidPkgName)
}
func TestArchVersionWithRelease(t *testing.T) {
info := exampleInfo()
info.Version = "0.0.1"
info.Release = "4"
pkginfoData, err := makeTestPkginfo(t, info)
require.NoError(t, err)
fields := extractPkginfoFields(pkginfoData)
require.Equal(t, "0.0.1-4", fields["pkgver"])
}
func TestArchVersionWithEpoch(t *testing.T) {
info := exampleInfo()
info.Version = "0.0.1"
info.Epoch = "2"
pkginfoData, err := makeTestPkginfo(t, info)
require.NoError(t, err)
fields := extractPkginfoFields(pkginfoData)
require.Equal(t, "2:0.0.1beta_1-1", fields["pkgver"])
}
func TestArchOverrideArchitecture(t *testing.T) {
info := exampleInfo()
info.ArchLinux.Arch = "randomarch"
pkginfoData, err := makeTestPkginfo(t, info)
require.NoError(t, err)
fields := extractPkginfoFields(pkginfoData)
require.Equal(t, "randomarch", fields["arch"])
}
func makeTestPkginfo(t *testing.T, info *nfpm.Info) ([]byte, error) {
t.Helper()
require.NoError(t, nfpm.PrepareForPackager(info, packagerName))
buf := &bytes.Buffer{}
tw := tar.NewWriter(buf)
entry, err := createPkginfo(info, tw, 1234)
if err != nil {
return nil, err
}
tw.Close()
tr := tar.NewReader(buf)
_, err = tr.Next()
require.NoError(t, err)
pkginfoData := make([]byte, entry.Size)
_, err = io.ReadFull(tr, pkginfoData)
if err != nil {
return nil, err
}
return pkginfoData, nil
}
func extractPkginfoFields(data []byte) map[string]string {
strData := string(data)
strData = strings.TrimPrefix(strData, "# Generated by nfpm\n")
strData = strings.TrimSpace(strData)
splitData := strings.Split(strData, "\n")
out := map[string]string{}
for _, kvPair := range splitData {
splitPair := strings.Split(kvPair, " = ")
out[splitPair[0]] = splitPair[1]
}
return out
}
const correctMtree = `#mtree
./foo/bar time=1234.0 mode=755 type=dir
./foo/bar/file time=1234.0 mode=600 size=143 type=file md5digest=abcd sha256digest=ef12
./3 time=12345.0 mode=644 size=100 type=file md5digest=abcd sha256digest=ef12
./sh time=123456.0 mode=777 type=link link=/bin/bash
`
func TestArchMtree(t *testing.T) {
info := exampleInfo()
require.NoError(t, nfpm.PrepareForPackager(info, packagerName))
buf := &bytes.Buffer{}
tw := tar.NewWriter(buf)
err := createMtree(tw, []MtreeEntry{
{
Destination: "foo/bar",
Time: 1234,
Type: files.TypeDir,
Mode: 0o755,
},
{
Destination: "foo/bar/file",
Time: 1234,
Type: files.TypeFile,
Mode: 0o600,
Size: 143,
MD5: []byte{0xAB, 0xCD},
SHA256: []byte{0xEF, 0x12},
},
{
Destination: "3",
Time: 12345,
Mode: 0o644,
Size: 100,
Type: files.TypeFile,
MD5: []byte{0xAB, 0xCD},
SHA256: []byte{0xEF, 0x12},
},
{
LinkSource: "/bin/bash",
Destination: "sh",
Time: 123456,
Mode: 0o777,
Type: files.TypeSymlink,
},
}, mtime)
require.NoError(t, err)
tw.Close()
tr := tar.NewReader(buf)
_, err = tr.Next()
require.NoError(t, err)
gr, err := pgzip.NewReader(tr)
require.NoError(t, err)
defer gr.Close()
mtree, err := io.ReadAll(gr)
require.NoError(t, err)
require.Equal(t, correctMtree, string(mtree))
}
func TestGlob(t *testing.T) {
var pkg bytes.Buffer
require.NoError(t, Default.Package(nfpm.WithDefaults(&nfpm.Info{
Name: "nfpm-repro",
Version: "1.0.0",
Maintainer: "asdfasdf",
MTime: mtime,
Overridables: nfpm.Overridables{
Contents: files.Contents{
{
Destination: "/usr/share/nfpm-repro",
Source: "../files/testdata/globtest/different-sizes/*/*.txt",
FileInfo: &files.ContentFileInfo{
Mode: 0o644,
MTime: mtime,
},
},
},
},
}), &pkg))
pkgZstd, err := zstd.NewReader(&pkg)
require.NoError(t, err)
t.Cleanup(func() { pkgZstd.Close() })
pkgTar := tar.NewReader(pkgZstd)
for {
f, err := pkgTar.Next()
if err == io.EOF || f == nil {
break
}
if f.Name == ".MTREE" {
break
}
}
mtreeTarBts, err := io.ReadAll(pkgTar)
require.NoError(t, err)
mtreeGzip, err := pgzip.NewReader(bytes.NewReader(mtreeTarBts))
require.NoError(t, err)
t.Cleanup(func() { require.NoError(t, mtreeGzip.Close()) })
mtreeContentBts, err := io.ReadAll(mtreeGzip)
require.NoError(t, err)
expectedTime := fmt.Sprintf("time=%d.0", mtime.Unix())
expected := map[string][]string{
"./.PKGINFO": {expectedTime, "mode=644", "size=185", "type=file", "md5digest=408daafbd01f6622f0bfd6ccdf96735f", "sha256digest=98468a4b87a677958f872662f476b14ff28cc1f8c6bd0029869e21946b4cd8d2"},
"./usr/": {expectedTime, "mode=755", "type=dir"},
"./usr/share/": {expectedTime, "mode=755", "type=dir"},
"./usr/share/nfpm-repro/": {expectedTime, "mode=755", "type=dir"},
"./usr/share/nfpm-repro/a/": {expectedTime, "mode=755", "type=dir"},
"./usr/share/nfpm-repro/a/a.txt": {expectedTime, "mode=644", "size=4", "type=file", "md5digest=d3b07384d113edec49eaa6238ad5ff00", "sha256digest=b5bb9d8014a0f9b1d61e21e796d78dccdf1352f23cd32812f4850b878ae4944c"},
"./usr/share/nfpm-repro/b/": {expectedTime, "mode=755", "type=dir"},
"./usr/share/nfpm-repro/b/b.txt": {expectedTime, "mode=644", "size=7", "type=file", "md5digest=551a67cc6e06de1910061fe318d28f72", "sha256digest=73a2c64f9545172c1195efb6616ca5f7afd1df6f245407cafb90de3998a1c97f"},
}
for _, line := range strings.Split(string(mtreeContentBts), "\n") {
if line == "#mtree" || line == "" {
continue
}
parts := strings.Fields(line)
filename := parts[0]
expect := expected[filename]
require.Equal(t, expect, strings.Split(line, " ")[1:], filename)
}
}