376 lines
9.3 KiB
Go
376 lines
9.3 KiB
Go
|
// Package unixpkgs contains dist Targets for building unix Tailscale packages.
|
||
|
package unixpkgs
|
||
|
|
||
|
import (
|
||
|
"archive/tar"
|
||
|
"compress/gzip"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"log"
|
||
|
"os"
|
||
|
"path/filepath"
|
||
|
"strings"
|
||
|
"time"
|
||
|
|
||
|
"github.com/goreleaser/nfpm"
|
||
|
"tailscale.com/release/dist"
|
||
|
)
|
||
|
|
||
|
type tgzTarget struct {
|
||
|
filenameArch string // arch to use in filename instead of deriving from goenv["GOARCH"]
|
||
|
goenv map[string]string
|
||
|
}
|
||
|
|
||
|
func (t *tgzTarget) arch() string {
|
||
|
if t.filenameArch != "" {
|
||
|
return t.filenameArch
|
||
|
}
|
||
|
return t.goenv["GOARCH"]
|
||
|
}
|
||
|
|
||
|
func (t *tgzTarget) os() string {
|
||
|
return t.goenv["GOOS"]
|
||
|
}
|
||
|
|
||
|
func (t *tgzTarget) String() string {
|
||
|
return fmt.Sprintf("%s/%s/tgz", t.os(), t.arch())
|
||
|
}
|
||
|
|
||
|
func (t *tgzTarget) Build(b *dist.Build) ([]string, error) {
|
||
|
var filename string
|
||
|
if t.goenv["GOOS"] == "linux" {
|
||
|
// Linux used to be the only tgz architecture, so we didn't put the OS
|
||
|
// name in the filename.
|
||
|
filename = fmt.Sprintf("tailscale_%s_%s.tgz", b.Version.Short, t.arch())
|
||
|
} else {
|
||
|
filename = fmt.Sprintf("tailscale_%s_%s_%s.tgz", b.Version.Short, t.os(), t.arch())
|
||
|
}
|
||
|
ts, err := b.BuildGoBinary("tailscale.com/cmd/tailscale", t.goenv)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
tsd, err := b.BuildGoBinary("tailscale.com/cmd/tailscaled", t.goenv)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
log.Printf("Building %s", filename)
|
||
|
|
||
|
out := filepath.Join(b.Out, filename)
|
||
|
f, err := os.Create(out)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
defer f.Close()
|
||
|
gw := gzip.NewWriter(f)
|
||
|
defer gw.Close()
|
||
|
tw := tar.NewWriter(gw)
|
||
|
defer tw.Close()
|
||
|
|
||
|
buildTime := time.Now()
|
||
|
addFile := func(src, dst string, mode int64) error {
|
||
|
f, err := os.Open(src)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
defer f.Close()
|
||
|
fi, err := f.Stat()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
hdr := &tar.Header{
|
||
|
Name: dst,
|
||
|
Size: fi.Size(),
|
||
|
Mode: mode,
|
||
|
ModTime: buildTime,
|
||
|
Uid: 0,
|
||
|
Gid: 0,
|
||
|
Uname: "root",
|
||
|
Gname: "root",
|
||
|
}
|
||
|
if err := tw.WriteHeader(hdr); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if _, err = io.Copy(tw, f); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
addDir := func(name string) error {
|
||
|
hdr := &tar.Header{
|
||
|
Name: name + "/",
|
||
|
Mode: 0755,
|
||
|
ModTime: buildTime,
|
||
|
Uid: 0,
|
||
|
Gid: 0,
|
||
|
Uname: "root",
|
||
|
Gname: "root",
|
||
|
}
|
||
|
return tw.WriteHeader(hdr)
|
||
|
}
|
||
|
dir := strings.TrimSuffix(filename, ".tgz")
|
||
|
if err := addDir(dir); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if err := addFile(tsd, filepath.Join(dir, "tailscaled"), 0755); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if err := addFile(ts, filepath.Join(dir, "tailscale"), 0755); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if t.os() == "linux" {
|
||
|
dir = filepath.Join(dir, "systemd")
|
||
|
if err := addDir(dir); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
tailscaledDir, err := b.GoPkg("tailscale.com/cmd/tailscaled")
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if err := addFile(filepath.Join(tailscaledDir, "tailscaled.service"), filepath.Join(dir, "tailscaled.service"), 0644); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if err := addFile(filepath.Join(tailscaledDir, "tailscaled.defaults"), filepath.Join(dir, "tailscaled.defaults"), 0644); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
}
|
||
|
if err := tw.Close(); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if err := gw.Close(); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if err := f.Close(); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return []string{filename}, nil
|
||
|
}
|
||
|
|
||
|
type debTarget struct {
|
||
|
goenv map[string]string
|
||
|
}
|
||
|
|
||
|
func (t *debTarget) os() string {
|
||
|
return t.goenv["GOOS"]
|
||
|
}
|
||
|
|
||
|
func (t *debTarget) arch() string {
|
||
|
return t.goenv["GOARCH"]
|
||
|
}
|
||
|
|
||
|
func (t *debTarget) String() string {
|
||
|
return fmt.Sprintf("linux/%s/deb", t.goenv["GOARCH"])
|
||
|
}
|
||
|
|
||
|
func (t *debTarget) Build(b *dist.Build) ([]string, error) {
|
||
|
if t.os() != "linux" {
|
||
|
return nil, errors.New("deb only supported on linux")
|
||
|
}
|
||
|
|
||
|
ts, err := b.BuildGoBinary("tailscale.com/cmd/tailscale", t.goenv)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
tsd, err := b.BuildGoBinary("tailscale.com/cmd/tailscaled", t.goenv)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
tailscaledDir, err := b.GoPkg("tailscale.com/cmd/tailscaled")
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
repoDir, err := b.GoPkg("tailscale.com")
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
arch := debArch(t.arch())
|
||
|
info := nfpm.WithDefaults(&nfpm.Info{
|
||
|
Name: "tailscale",
|
||
|
Arch: arch,
|
||
|
Platform: "linux",
|
||
|
Version: b.Version.Short,
|
||
|
Maintainer: "Tailscale Inc <info@tailscale.com>",
|
||
|
Description: "The easiest, most secure, cross platform way to use WireGuard + oauth2 + 2FA/SSO",
|
||
|
Homepage: "https://www.tailscale.com",
|
||
|
License: "MIT",
|
||
|
Section: "net",
|
||
|
Priority: "extra",
|
||
|
Overridables: nfpm.Overridables{
|
||
|
Files: map[string]string{
|
||
|
ts: "/usr/bin/tailscale",
|
||
|
tsd: "/usr/sbin/tailscaled",
|
||
|
filepath.Join(tailscaledDir, "tailscaled.service"): "/lib/systemd/system/tailscaled.service",
|
||
|
},
|
||
|
ConfigFiles: map[string]string{
|
||
|
filepath.Join(tailscaledDir, "tailscaled.defaults"): "/etc/default/tailscaled",
|
||
|
},
|
||
|
Scripts: nfpm.Scripts{
|
||
|
PostInstall: filepath.Join(repoDir, "release/deb/debian.postinst.sh"),
|
||
|
PreRemove: filepath.Join(repoDir, "release/deb/debian.prerm.sh"),
|
||
|
PostRemove: filepath.Join(repoDir, "release/deb/debian.postrm.sh"),
|
||
|
},
|
||
|
Depends: []string{"iptables", "iproute2"},
|
||
|
Recommends: []string{"tailscale-archive-keyring (>= 1.35.181)"},
|
||
|
Replaces: []string{"tailscale-relay"},
|
||
|
Conflicts: []string{"tailscale-relay"},
|
||
|
},
|
||
|
})
|
||
|
pkg, err := nfpm.Get("deb")
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
filename := fmt.Sprintf("tailscale_%s_%s.deb", b.Version.Short, arch)
|
||
|
log.Printf("Building %s", filename)
|
||
|
f, err := os.Create(filepath.Join(b.Out, filename))
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
defer f.Close()
|
||
|
if err := pkg.Package(info, f); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if err := f.Close(); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return []string{filename}, nil
|
||
|
}
|
||
|
|
||
|
type rpmTarget struct {
|
||
|
goenv map[string]string
|
||
|
}
|
||
|
|
||
|
func (t *rpmTarget) os() string {
|
||
|
return t.goenv["GOOS"]
|
||
|
}
|
||
|
|
||
|
func (t *rpmTarget) arch() string {
|
||
|
return t.goenv["GOARCH"]
|
||
|
}
|
||
|
|
||
|
func (t *rpmTarget) String() string {
|
||
|
return fmt.Sprintf("linux/%s/rpm", t.arch())
|
||
|
}
|
||
|
|
||
|
func (t *rpmTarget) Build(b *dist.Build) ([]string, error) {
|
||
|
if t.os() != "linux" {
|
||
|
return nil, errors.New("rpm only supported on linux")
|
||
|
}
|
||
|
|
||
|
ts, err := b.BuildGoBinary("tailscale.com/cmd/tailscale", t.goenv)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
tsd, err := b.BuildGoBinary("tailscale.com/cmd/tailscaled", t.goenv)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
tailscaledDir, err := b.GoPkg("tailscale.com/cmd/tailscaled")
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
repoDir, err := b.GoPkg("tailscale.com")
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
arch := rpmArch(t.arch())
|
||
|
info := nfpm.WithDefaults(&nfpm.Info{
|
||
|
Name: "tailscale",
|
||
|
Arch: arch,
|
||
|
Platform: "linux",
|
||
|
Version: b.Version.Short,
|
||
|
Maintainer: "Tailscale Inc <info@tailscale.com>",
|
||
|
Description: "The easiest, most secure, cross platform way to use WireGuard + oauth2 + 2FA/SSO",
|
||
|
Homepage: "https://www.tailscale.com",
|
||
|
License: "MIT",
|
||
|
Overridables: nfpm.Overridables{
|
||
|
Files: map[string]string{
|
||
|
ts: "/usr/bin/tailscale",
|
||
|
tsd: "/usr/sbin/tailscaled",
|
||
|
filepath.Join(tailscaledDir, "tailscaled.service"): "/lib/systemd/system/tailscaled.service",
|
||
|
},
|
||
|
ConfigFiles: map[string]string{
|
||
|
filepath.Join(tailscaledDir, "tailscaled.defaults"): "/etc/default/tailscaled",
|
||
|
},
|
||
|
// SELinux policy on e.g. CentOS 8 forbids writing to /var/cache.
|
||
|
// Creating an empty directory at install time resolves this issue.
|
||
|
EmptyFolders: []string{"/var/cache/tailscale"},
|
||
|
Scripts: nfpm.Scripts{
|
||
|
PostInstall: filepath.Join(repoDir, "release/rpm/rpm.postinst.sh"),
|
||
|
PreRemove: filepath.Join(repoDir, "release/rpm/rpm.prerm.sh"),
|
||
|
PostRemove: filepath.Join(repoDir, "release/rpm/rpm.postrm.sh"),
|
||
|
},
|
||
|
Depends: []string{"iptables", "iproute"},
|
||
|
Replaces: []string{"tailscale-relay"},
|
||
|
Conflicts: []string{"tailscale-relay"},
|
||
|
RPM: nfpm.RPM{
|
||
|
Group: "Network",
|
||
|
},
|
||
|
},
|
||
|
})
|
||
|
pkg, err := nfpm.Get("rpm")
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
filename := fmt.Sprintf("tailscale_%s_%s.rpm", b.Version.Short, arch)
|
||
|
log.Printf("Building %s", filename)
|
||
|
|
||
|
f, err := os.Create(filepath.Join(b.Out, filename))
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
defer f.Close()
|
||
|
if err := pkg.Package(info, f); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if err := f.Close(); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return []string{filename}, nil
|
||
|
}
|
||
|
|
||
|
// debArch returns the debian arch name for the given Go arch name.
|
||
|
// nfpm also does this translation internally, but we need to do it outside nfpm
|
||
|
// because we also need the filename to be correct.
|
||
|
func debArch(arch string) string {
|
||
|
switch arch {
|
||
|
case "386":
|
||
|
return "i386"
|
||
|
case "arm":
|
||
|
// TODO: this is supposed to be "armel" for GOARM=5, and "armhf" for
|
||
|
// GOARM=6 and 7. But we have some tech debt to pay off here before we
|
||
|
// can ship more than 1 ARM deb, so for now match redo's behavior of
|
||
|
// shipping armv5 binaries in an armv7 trenchcoat.
|
||
|
return "armhf"
|
||
|
default:
|
||
|
return arch
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// rpmArch returns the RPM arch name for the given Go arch name.
|
||
|
// nfpm also does this translation internally, but we need to do it outside nfpm
|
||
|
// because we also need the filename to be correct.
|
||
|
func rpmArch(arch string) string {
|
||
|
switch arch {
|
||
|
case "amd64":
|
||
|
return "x86_64"
|
||
|
case "386":
|
||
|
return "i386"
|
||
|
case "arm":
|
||
|
return "armv7hl"
|
||
|
case "arm64":
|
||
|
return "aarch64"
|
||
|
default:
|
||
|
return arch
|
||
|
}
|
||
|
}
|