util: distro: Refactor family and distro code

I hate writing abstraction code like this, but I'm hoping it will be
useful.
This commit is contained in:
James Shubin
2024-10-15 19:45:48 -04:00
parent d403f18b2a
commit d24149518c
8 changed files with 114 additions and 92 deletions

View File

@@ -2,8 +2,8 @@ import "fmt"
import "os" import "os"
print "debian" { print "debian" {
msg => fmt.printf("is_debian: %t", os.is_debian()), msg => fmt.printf("is_family_debian: %t", os.is_family_debian()),
} }
print "redhat" { print "redhat" {
msg => fmt.printf("is_redhat: %t", os.is_redhat()), msg => fmt.printf("is_family_redhat: %t", os.is_family_redhat()),
} }

View File

@@ -2,5 +2,5 @@ import "os"
pkg "cowsay" { pkg "cowsay" {
state => "installed", state => "installed",
allowunsupported => os.is_debian() ?: true, allowunsupported => os.is_family_debian() ?: true,
} }

View File

@@ -31,11 +31,11 @@ package coreos
import ( import (
"context" "context"
"os"
"github.com/purpleidea/mgmt/lang/funcs/simple" "github.com/purpleidea/mgmt/lang/funcs/simple"
"github.com/purpleidea/mgmt/lang/funcs/vars" "github.com/purpleidea/mgmt/lang/funcs/vars"
"github.com/purpleidea/mgmt/lang/types" "github.com/purpleidea/mgmt/lang/types"
distroUtil "github.com/purpleidea/mgmt/util/distro"
) )
func init() { func init() {
@@ -46,129 +46,70 @@ func init() {
vars.ModuleRegister(ModuleName, "family_redhat", func() vars.Value { vars.ModuleRegister(ModuleName, "family_redhat", func() vars.Value {
return &types.StrValue{ return &types.StrValue{
V: familyRedHat, V: distroUtil.FamilyRedHat,
} }
}) })
vars.ModuleRegister(ModuleName, "family_debian", func() vars.Value { vars.ModuleRegister(ModuleName, "family_debian", func() vars.Value {
return &types.StrValue{ return &types.StrValue{
V: familyDebian, V: distroUtil.FamilyDebian,
} }
}) })
vars.ModuleRegister(ModuleName, "family_archlinux", func() vars.Value { vars.ModuleRegister(ModuleName, "family_archlinux", func() vars.Value {
return &types.StrValue{ return &types.StrValue{
V: familyArchLinux, V: distroUtil.FamilyArchLinux,
} }
}) })
// TODO: Create a family method that will return a giant struct. // TODO: Create a family method that will return a giant struct.
simple.ModuleRegister(ModuleName, "is_redhat", &simple.Scaffold{ simple.ModuleRegister(ModuleName, "is_family_redhat", &simple.Scaffold{
T: types.NewType("func() bool"), T: types.NewType("func() bool"),
F: IsRedHat, F: IsFamilyRedHat,
}) })
simple.ModuleRegister(ModuleName, "is_debian", &simple.Scaffold{ simple.ModuleRegister(ModuleName, "is_family_debian", &simple.Scaffold{
T: types.NewType("func() bool"), T: types.NewType("func() bool"),
F: IsDebian, F: IsFamilyDebian,
}) })
simple.ModuleRegister(ModuleName, "is_archlinux", &simple.Scaffold{ simple.ModuleRegister(ModuleName, "is_family_archlinux", &simple.Scaffold{
T: types.NewType("func() bool"), T: types.NewType("func() bool"),
F: IsArchLinux, F: IsFamilyArchLinux,
}) })
} }
const (
familyRedHat = "redhat"
familyDebian = "debian"
familyArchLinux = "archlinux"
)
// Family returns the distro family. // Family returns the distro family.
// TODO: Detect OS changes. // TODO: Detect OS changes.
func Family(ctx context.Context, input []types.Value) (types.Value, error) { func Family(ctx context.Context, input []types.Value) (types.Value, error) {
if b, err := isRedHat(ctx); err != nil { s, err := distroUtil.Family(ctx)
if err != nil {
return nil, err return nil, err
} else if b {
return &types.StrValue{
V: familyRedHat,
}, nil
}
if b, err := isDebian(ctx); err != nil {
return nil, err
} else if b {
return &types.StrValue{
V: familyDebian,
}, nil
}
if b, err := isArchLinux(ctx); err != nil {
return nil, err
} else if b {
return &types.StrValue{
V: familyArchLinux,
}, nil
} }
return &types.StrValue{ return &types.StrValue{
V: "", // unknown V: s, // empty if unknown
}, nil }, nil
} }
// IsRedHat detects if the os family is redhat. // IsFamilyRedHat detects if the os family is redhat.
// TODO: Detect OS changes. // TODO: Detect OS changes.
func IsRedHat(ctx context.Context, input []types.Value) (types.Value, error) { func IsFamilyRedHat(ctx context.Context, input []types.Value) (types.Value, error) {
b, err := isRedHat(ctx) b, err := distroUtil.IsFamilyRedHat(ctx)
return &types.BoolValue{ return &types.BoolValue{
V: b, V: b,
}, err }, err
} }
// IsDebian detects if the os family is debian. // IsFamilyDebian detects if the os family is debian.
// TODO: Detect OS changes. // TODO: Detect OS changes.
func IsDebian(ctx context.Context, input []types.Value) (types.Value, error) { func IsFamilyDebian(ctx context.Context, input []types.Value) (types.Value, error) {
b, err := isDebian(ctx) b, err := distroUtil.IsFamilyDebian(ctx)
return &types.BoolValue{ return &types.BoolValue{
V: b, V: b,
}, err }, err
} }
// IsArchLinux detects if the os family is archlinux. // IsFamilyArchLinux detects if the os family is archlinux.
// TODO: Detect OS changes. // TODO: Detect OS changes.
func IsArchLinux(ctx context.Context, input []types.Value) (types.Value, error) { func IsFamilyArchLinux(ctx context.Context, input []types.Value) (types.Value, error) {
b, err := isArchLinux(ctx) b, err := distroUtil.IsFamilyArchLinux(ctx)
return &types.BoolValue{ return &types.BoolValue{
V: b, V: b,
}, err }, err
} }
func isRedHat(ctx context.Context) (bool, error) {
// TODO: use ctx around io operations
_, err := os.Stat("/etc/redhat-release")
if os.IsNotExist(err) {
return false, nil
}
if err != nil {
return false, err
}
return true, nil
}
func isDebian(ctx context.Context) (bool, error) {
// TODO: use ctx around io operations
_, err := os.Stat("/etc/debian_version")
if os.IsNotExist(err) {
return false, nil
}
if err != nil {
return false, err
}
return true, nil
}
func isArchLinux(ctx context.Context) (bool, error) {
// TODO: use ctx around io operations
_, err := os.Stat("/etc/arch-release")
if os.IsNotExist(err) {
return false, nil
}
if err != nil {
return false, err
}
return true, nil
}

View File

@@ -9,7 +9,7 @@ import "fmt"
class xclass { class xclass {
#import "os" # this should not be required, top-level should be enough #import "os" # this should not be required, top-level should be enough
$aaa = if os.is_debian() { "bbb" } else { "ccc" } $aaa = if os.is_family_debian() { "bbb" } else { "ccc" }
print "${aaa}" { print "${aaa}" {
msg => "hello", msg => "hello",

View File

@@ -9,11 +9,11 @@ import "fmt"
class xclass { class xclass {
# note that `os` is not imported here # note that `os` is not imported here
$aaa = if os.is_debian() { "bbb" } else { "ccc" } $aaa = if os.is_family_debian() { "bbb" } else { "ccc" }
print "${aaa}" { print "${aaa}" {
msg => "hello", msg => "hello",
} }
} }
-- OUTPUT -- -- OUTPUT --
# err: errSetScope: func `os.is_debian` does not exist in this scope # err: errSetScope: func `os.is_family_debian` does not exist in this scope

View File

@@ -8,7 +8,7 @@ import "fmt"
class xclass { class xclass {
import "os" # we can also use a scoped local import import "os" # we can also use a scoped local import
$aaa = if os.is_debian() { "bbb" } else { "ccc" } $aaa = if os.is_family_debian() { "bbb" } else { "ccc" }
print "${aaa}" { print "${aaa}" {
msg => "hello", msg => "hello",

View File

@@ -31,7 +31,7 @@ import "os"
# base contains the personal tweaks and utilities of james (purpleidea) # base contains the personal tweaks and utilities of james (purpleidea)
class base() { class base() {
if os.is_redhat() { if os.is_family_redhat() {
pkg [ pkg [
"ack", "ack",
"bash-completion", "bash-completion",
@@ -53,7 +53,7 @@ class base() {
state => "installed", state => "installed",
} }
} }
if os.is_debian() { if os.is_family_debian() {
pkg [ pkg [
"ack", "ack",
"bash-completion", "bash-completion",

View File

@@ -31,17 +31,39 @@
// these all in one place so that adding this data happens all in the same file. // these all in one place so that adding this data happens all in the same file.
package distro package distro
import (
"context"
"os"
)
const (
// FamilyRedHat represents distros like Fedora and RHEL.
FamilyRedHat = "redhat"
// FamilyDebian represents distros like Debian and Ubuntu.
FamilyDebian = "debian"
// FamilyArchLinux represents primarily ArchLinux.
FamilyArchLinux = "archlinux"
// DistroDebian is the Debian distro.
DistroDebian = "debian"
// DistroFedora is the Fedora distro.
DistroFedora = "fedora"
)
var ( var (
// MapDistroToBootstrapPackages is a map of distro to packages needed to // MapDistroToBootstrapPackages is a map of distro to packages needed to
// run our software. // run our software.
MapDistroToBootstrapPackages = map[string][]string{ MapDistroToBootstrapPackages = map[string][]string{
// TODO: add more values // TODO: add more values
"debian": { DistroDebian: {
"libaugeas-dev", "libaugeas-dev",
"libvirt-dev", "libvirt-dev",
"packagekit-tools", "packagekit-tools",
}, },
"fedora": { DistroFedora: {
"augeas-devel", "augeas-devel",
"libvirt-devel", "libvirt-devel",
"PackageKit", "PackageKit",
@@ -55,3 +77,62 @@ func DistroToBootstrapPackages(distro string) ([]string, bool) {
l, exists := MapDistroToBootstrapPackages[distro] l, exists := MapDistroToBootstrapPackages[distro]
return l, exists return l, exists
} }
// Family returns the distro family.
func Family(ctx context.Context) (string, error) {
if b, err := IsFamilyRedHat(ctx); err != nil {
return "", err
} else if b {
return FamilyRedHat, nil
}
if b, err := IsFamilyDebian(ctx); err != nil {
return "", err
} else if b {
return FamilyDebian, nil
}
if b, err := IsFamilyArchLinux(ctx); err != nil {
return "", err
} else if b {
return FamilyArchLinux, nil
}
return "", nil // unknown
}
// IsFamilyRedHat detects if the os family is redhat.
func IsFamilyRedHat(ctx context.Context) (bool, error) {
// TODO: use ctx around io operations
_, err := os.Stat("/etc/redhat-release")
if os.IsNotExist(err) {
return false, nil
}
if err != nil {
return false, err
}
return true, nil
}
// IsFamilyDebian detects if the os family is debian.
func IsFamilyDebian(ctx context.Context) (bool, error) {
// TODO: use ctx around io operations
_, err := os.Stat("/etc/debian_version")
if os.IsNotExist(err) {
return false, nil
}
if err != nil {
return false, err
}
return true, nil
}
// IsFamilyArchLinux detects if the os family is archlinux.
func IsFamilyArchLinux(ctx context.Context) (bool, error) {
// TODO: use ctx around io operations
_, err := os.Stat("/etc/arch-release")
if os.IsNotExist(err) {
return false, nil
}
if err != nil {
return false, err
}
return true, nil
}