You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
309 lines
7.3 KiB
309 lines
7.3 KiB
package main
|
|
|
|
import (
|
|
"bufio"
|
|
"fmt"
|
|
"io"
|
|
"io/fs"
|
|
"math"
|
|
"os"
|
|
"time"
|
|
|
|
"errors"
|
|
"os/exec"
|
|
"strings"
|
|
|
|
"github.com/goccy/go-yaml"
|
|
"github.com/jwalton/gchalk"
|
|
"github.com/rs/zerolog"
|
|
"github.com/rs/zerolog/log"
|
|
"github.com/spf13/pflag"
|
|
"github.com/thoas/go-funk"
|
|
)
|
|
|
|
type PackageManager struct {
|
|
Name string
|
|
}
|
|
|
|
func (pm *PackageManager) List() (packages []string, err error) {
|
|
switch pm.Name {
|
|
case "pacman":
|
|
out, err := exec.Command("pacman", "-Sql").Output()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return strings.Split(string(out), "\n"), nil
|
|
|
|
case "apt-get":
|
|
out, err := exec.Command("apt-cache", "pkgnames", "-g").Output()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return strings.Split(string(out), "\n"), nil
|
|
|
|
case "tdnf", "dnf", "yum":
|
|
out, err := exec.Command(pm.Name, "list", "available").Output()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return funk.Map(strings.Split(string(out), "\n"), func(p string) string {
|
|
return strings.SplitN(p, " ", 1)[0]
|
|
}).([]string), nil
|
|
|
|
case "brew": // please no
|
|
out, err := exec.Command("brew", "formulae").Output()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return strings.Split(string(out), "\n"), nil
|
|
|
|
default:
|
|
return nil, errors.New("Unknown package manager " + pm.Name)
|
|
}
|
|
}
|
|
|
|
func (pm *PackageManager) ListInstalled() (packages []string, err error) {
|
|
switch pm.Name {
|
|
case "pacman":
|
|
out, err := exec.Command("pacman", "-Qq").Output()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return strings.Split(string(out), "\n"), nil
|
|
|
|
case "apt-get":
|
|
out, err := exec.Command("dpkg-query", "-f", "${Package}\n", "-W").Output()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return strings.Split(string(out), "\n"), nil
|
|
|
|
case "tdnf", "dnf", "yum":
|
|
out, err := exec.Command(pm.Name, "list", "installed").Output()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return funk.Map(strings.Split(string(out), "\n"), func(p string) string {
|
|
return strings.SplitN(p, " ", 1)[0]
|
|
}).([]string), nil
|
|
|
|
case "brew": // please no
|
|
out, err := exec.Command("brew", "list", "--formula").Output()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return strings.Split(string(out), "\n"), nil
|
|
|
|
default:
|
|
return nil, errors.New("Unknown package manager " + pm.Name)
|
|
}
|
|
}
|
|
|
|
func (pm *PackageManager) Install(pkg string) error {
|
|
switch pm.Name {
|
|
case "pacman":
|
|
return executeCommandWithFormatting("pacman", "-S", "--noconfirm", pkg)
|
|
|
|
case "apt-get":
|
|
return executeCommandWithFormatting("apt-get", "install", "-y", pkg)
|
|
|
|
case "tdnf", "dnf", "yum":
|
|
return executeCommandWithFormatting(pm.Name, "install", "-y", pkg)
|
|
|
|
case "brew": // please no
|
|
return executeCommandWithFormatting("brew", "install", pkg)
|
|
|
|
default:
|
|
return errors.New("Unknown package manager " + pm.Name)
|
|
}
|
|
}
|
|
|
|
func (pm *PackageManager) Remove(pkg string) error {
|
|
switch pm.Name {
|
|
case "pacman":
|
|
exec.Command("pacman", "-R", pkg).Run()
|
|
|
|
case "apt-get":
|
|
err := exec.Command("apt-get", "remove", pkg).Run()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
case "tdnf", "dnf", "yum":
|
|
err := exec.Command(pm.Name, "remove", pkg).Run()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
case "brew": // please no
|
|
err := exec.Command("brew", "remove", pkg).Run()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
default:
|
|
return errors.New("Unknown package manager " + pm.Name)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func executeCommandWithFormatting(command string, args ...string) error {
|
|
cmd := exec.Command(command, args...)
|
|
cmd.Env = append(cmd.Env, "LC_ALL=C.UTF-8")
|
|
stdout, err := cmd.StdoutPipe()
|
|
checkError(err)
|
|
stderr, err := cmd.StderrPipe()
|
|
checkError(err)
|
|
stdoutReader := bufio.NewReader(stdout)
|
|
stderrReader := bufio.NewReader(stderr)
|
|
ch := make(chan uint8)
|
|
go func() {
|
|
for {
|
|
line, err := stdoutReader.ReadString('\n')
|
|
if err == io.EOF {
|
|
break
|
|
} else {
|
|
checkError(err)
|
|
}
|
|
delim := gchalk.Gray("│") + gchalk.Blue(" |")
|
|
fmt.Println(delim, strings.TrimSpace(line))
|
|
}
|
|
ch <- 0
|
|
}()
|
|
go func() {
|
|
for {
|
|
line, err := stderrReader.ReadString('\n')
|
|
if err == io.EOF {
|
|
break
|
|
} else {
|
|
checkError(err)
|
|
}
|
|
delim := gchalk.Gray("│") + gchalk.Red(" |")
|
|
fmt.Println(delim, strings.TrimSpace(line))
|
|
}
|
|
ch <- 0
|
|
}()
|
|
err = cmd.Start()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
<-ch
|
|
<-ch
|
|
return cmd.Wait()
|
|
}
|
|
|
|
func commandExists(cmd string) bool {
|
|
command := exec.Command(cmd)
|
|
err := command.Start()
|
|
return err == nil
|
|
}
|
|
|
|
func detectPackageManager() (manager string) {
|
|
managers := []string{"pacman", "apt-get", "tdnf", "dnf", "yum", "brew"}
|
|
for _, manager := range managers {
|
|
if commandExists(manager) {
|
|
return manager
|
|
}
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func terminateWithError(err error) {
|
|
log.Fatal().Err(err).Msg("")
|
|
}
|
|
|
|
func checkUserError(err error) {
|
|
if err != nil {
|
|
terminateWithError(err)
|
|
}
|
|
}
|
|
|
|
func checkError(err error) {
|
|
if err != nil {
|
|
log.Panic().Err(err).Msg("")
|
|
}
|
|
}
|
|
|
|
func executeRunlevel(runlevel string, tasks []string, shell string) {
|
|
fmt.Printf("# Running runlevel '%s'\n", runlevel)
|
|
for _, task := range tasks {
|
|
fmt.Println(gchalk.Gray("│ # Running task: " + task))
|
|
checkUserError(executeCommandWithFormatting(shell, "-c", task))
|
|
}
|
|
}
|
|
|
|
func main() {
|
|
start := time.Now()
|
|
log.Logger = log.Output(zerolog.ConsoleWriter{Out: os.Stderr})
|
|
var configData []byte
|
|
if cfg := *pflag.StringP("config", "c", "", "Provide a path to an alternative config file"); cfg == "" {
|
|
rawConfig, err := os.ReadFile("polymer.yml")
|
|
if err == fs.ErrNotExist {
|
|
rawConfig, err = os.ReadFile("polymer.yaml")
|
|
if err == fs.ErrNotExist {
|
|
log.Fatal().Msg("Neither polymer.yml or polymer.yaml were found. Aborting.")
|
|
} else if err != nil {
|
|
log.Panic().Err(err).Msg("")
|
|
}
|
|
} else if err != nil {
|
|
log.Panic().Err(err).Msg("")
|
|
}
|
|
configData = rawConfig
|
|
} else {
|
|
rawConfig, err := os.ReadFile(cfg)
|
|
if err == fs.ErrNotExist {
|
|
log.Fatal().Msg("Config file " + cfg + " was not found. Aborting.")
|
|
} else if err != nil {
|
|
log.Panic().Err(err).Msg("")
|
|
}
|
|
configData = rawConfig
|
|
}
|
|
var tasks Config
|
|
yaml.Unmarshal(configData, &tasks)
|
|
if tasks.Shell == "" {
|
|
tasks.Shell = "/bin/sh"
|
|
}
|
|
pmName := ""
|
|
if len(tasks.Packages) > 0 {
|
|
pmName = detectPackageManager()
|
|
if pmName == "" {
|
|
log.Fatal().Msg("No supported package manager (pacman, apt-get, yum, dnf, tdnf, brew) was found!")
|
|
}
|
|
}
|
|
executeRunlevel("init", tasks.Execute.Init, tasks.Shell)
|
|
pm := PackageManager{Name: pmName}
|
|
fmt.Printf("\n# Installing packages: '%s'\n", strings.Join(tasks.Packages, ", "))
|
|
fmt.Println(gchalk.Gray("│ # Invoking " + pmName))
|
|
checkUserError(pm.Install(strings.Join(tasks.Packages, "")))
|
|
fmt.Printf(
|
|
gchalk.Green("# Finished with 0 errors in %g seconds!\n"),
|
|
math.Round(time.Since(start).Seconds()*1000)/1000,
|
|
)
|
|
}
|
|
|
|
type Services struct {
|
|
Stop []string `yaml:"stop"`
|
|
Start []string `yaml:"start"`
|
|
Enable []string `yaml:"enable"`
|
|
Disable []string `yaml:"disable"`
|
|
EnableStart []string `yaml:"startup"`
|
|
DisableStop []string `yaml:"disableAndStop"`
|
|
}
|
|
|
|
type Runlevels struct {
|
|
Init []string `yaml:"init"`
|
|
WithPackages []string `yaml:"withPackages"`
|
|
WithServices []string `yaml:"withServices"`
|
|
Final []string `yaml:"final"`
|
|
}
|
|
|
|
type Config struct {
|
|
Shell string `yaml:"shell"`
|
|
Packages []string `yaml:"packages"`
|
|
Systemd Services `yaml:"systemd"`
|
|
OpenRC Services `yaml:"openrc"`
|
|
Firewall []string `yaml:"firewall"`
|
|
IPtables []string `yaml:"iptables"`
|
|
Execute Runlevels `yaml:"runlevels"`
|
|
}
|