|
|
|
package cluster
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"darvaza.org/core"
|
|
|
|
|
|
|
|
"git.jpi.io/amery/jpictl/pkg/rings"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Env is a shell environment factory for this cluster
|
|
|
|
type Env struct {
|
|
|
|
ZoneIterator
|
|
|
|
RegionIterator
|
|
|
|
|
|
|
|
cephFSID string
|
|
|
|
export bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// Env returns a shell environment factory
|
|
|
|
func (m *Cluster) Env(export bool) (*Env, error) {
|
|
|
|
fsid, err := m.GetCephFSID()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
env := &Env{
|
|
|
|
ZoneIterator: m,
|
|
|
|
RegionIterator: m,
|
|
|
|
cephFSID: fsid.String(),
|
|
|
|
export: export,
|
|
|
|
}
|
|
|
|
|
|
|
|
return env, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Zones returns the list of Zone IDs of a region,
|
|
|
|
// or from all if none is specified.
|
|
|
|
func (m *Env) Zones(r *Region) []rings.ZoneID {
|
|
|
|
var zones []rings.ZoneID
|
|
|
|
|
|
|
|
iter := core.IIf[ZoneIterator](r != nil, r, m)
|
|
|
|
|
|
|
|
iter.ForEachZone(func(z *Zone) bool {
|
|
|
|
zones = append(zones, z.ID)
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
|
|
|
|
core.SliceSort(zones, cmpOrdered[rings.ZoneID])
|
|
|
|
return zones
|
|
|
|
}
|
|
|
|
|
|
|
|
// RegionsNames returns a sorted list of primary regions names
|
|
|
|
func (m *Env) RegionsNames() []string {
|
|
|
|
var regions []string
|
|
|
|
|
|
|
|
m.ForEachRegion(func(r *Region) bool {
|
|
|
|
if r.IsPrimary() {
|
|
|
|
regions = append(regions, r.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
|
|
|
|
sort.Strings(regions)
|
|
|
|
return regions
|
|
|
|
}
|
|
|
|
|
|
|
|
// Regions returns a sorted list of primary regions IDs
|
|
|
|
func (m *Env) Regions() (regions []rings.RegionID) {
|
|
|
|
m.ForEachRegion(func(r *Region) bool {
|
|
|
|
if r.IsPrimary() {
|
|
|
|
regions = append(regions, r.ID)
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
|
|
|
|
core.SliceSort(regions, cmpOrdered[rings.RegionID])
|
|
|
|
return regions
|
|
|
|
}
|
|
|
|
|
|
|
|
// WriteTo generates environment variables for shell scripts
|
|
|
|
func (m *Env) WriteTo(w io.Writer) (int64, error) {
|
|
|
|
var buf bytes.Buffer
|
|
|
|
|
|
|
|
if m.cephFSID != "" {
|
|
|
|
m.writeEnvVar(&buf, m.cephFSID, "FSID")
|
|
|
|
}
|
|
|
|
|
|
|
|
regions := m.getRegions()
|
|
|
|
ids := core.SliceMap(regions, func(_ []rings.RegionID, r *Region) (out []rings.RegionID) {
|
|
|
|
return append(out, r.ID)
|
|
|
|
})
|
|
|
|
names := core.SliceMap(regions, func(_ []string, r *Region) (out []string) {
|
|
|
|
return append(out, r.Name)
|
|
|
|
})
|
|
|
|
|
|
|
|
m.writeEnvVar(&buf, genEnvInts(ids), "REGIONS")
|
|
|
|
m.writeEnvVar(&buf, genEnvStrings(names), "REGIONS_NAMES")
|
|
|
|
|
|
|
|
for _, r := range regions {
|
|
|
|
m.writeEnvRegion(&buf, r)
|
|
|
|
}
|
|
|
|
|
|
|
|
return buf.WriteTo(w)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Env) getRegions() (out []*Region) {
|
|
|
|
m.ForEachRegion(func(r *Region) bool {
|
|
|
|
if r.IsPrimary() {
|
|
|
|
out = append(out, r)
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
|
|
|
|
core.SliceSort(out, func(a, b *Region) int {
|
|
|
|
return cmpOrdered(a.ID, b.ID)
|
|
|
|
})
|
|
|
|
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
func (m *Env) writeEnvRegion(w io.Writer, r *Region) {
|
|
|
|
regionID := r.ID
|
|
|
|
|
|
|
|
// REGION{regionID}_NAME
|
|
|
|
m.writeEnvVar(w, r.Name, "REGION%v_%s", regionID, "NAME")
|
|
|
|
|
|
|
|
// REGION{regionID}_ZONES
|
|
|
|
m.writeEnvVar(w, genEnvInts(m.Zones(r)), "REGION%v_%s", regionID, "ZONES")
|
|
|
|
|
|
|
|
r.ForEachZone(func(z *Zone) bool {
|
|
|
|
m.writeEnvZone(w, r, z)
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Env) writeEnvZone(w io.Writer, r *Region, z *Zone) {
|
|
|
|
zonePrefix := fmt.Sprintf("REGION%v_ZONE%v", r.ID, z.ID)
|
|
|
|
monPrefix := zonePrefix + "_MON"
|
|
|
|
|
|
|
|
// ZONE{zoneID}
|
|
|
|
m.writeEnvVar(w, genEnvZoneNodes(z), zonePrefix)
|
|
|
|
|
|
|
|
// ZONE{zoneID}_NAME
|
|
|
|
m.writeEnvVar(w, z.Name, zonePrefix+"_NAME")
|
|
|
|
|
|
|
|
// ZONE{zoneID}_GW
|
|
|
|
gateways, _ := z.GatewayIDs()
|
|
|
|
m.writeEnvVar(w, genEnvInts(gateways), zonePrefix+"_GW")
|
|
|
|
|
|
|
|
// Ceph
|
|
|
|
monitors := z.GetCephMonitors()
|
|
|
|
// MON{zoneID}_NAME
|
|
|
|
m.writeEnvVar(w, genEnvZoneCephMonNames(monitors), monPrefix)
|
|
|
|
// MON{zoneID}_IP
|
|
|
|
m.writeEnvVar(w, genEnvZoneCephMonIPs(monitors), monPrefix+"_IP")
|
|
|
|
// MON{zoneID}_ID
|
|
|
|
m.writeEnvVar(w, genEnvZoneCephMonIDs(monitors), monPrefix+"_ID")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Env) writeEnvVar(w io.Writer, value string, name string, args ...any) {
|
|
|
|
var prefix string
|
|
|
|
|
|
|
|
if m.export {
|
|
|
|
prefix = "export "
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(args) > 0 {
|
|
|
|
name = fmt.Sprintf(name, args...)
|
|
|
|
}
|
|
|
|
|
|
|
|
if name != "" {
|
|
|
|
value = strings.TrimSpace(value)
|
|
|
|
|
|
|
|
if value == "" {
|
|
|
|
_, _ = fmt.Fprintf(w, "%s%s=\n", prefix, name)
|
|
|
|
} else {
|
|
|
|
_, _ = fmt.Fprintf(w, "%s%s=%q\n", prefix, name, value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func genEnvInts[T core.Signed](values []T) string {
|
|
|
|
var buf bytes.Buffer
|
|
|
|
|
|
|
|
for _, v := range values {
|
|
|
|
if buf.Len() > 0 {
|
|
|
|
_, _ = buf.WriteRune(' ')
|
|
|
|
}
|
|
|
|
|
|
|
|
_, _ = buf.WriteString(fmt.Sprintf("%v", int64(v)))
|
|
|
|
}
|
|
|
|
|
|
|
|
return buf.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func genEnvStrings(values []string) string {
|
|
|
|
return strings.Join(values, " ")
|
|
|
|
}
|
|
|
|
|
|
|
|
func genEnvZoneNodes(z *Zone) string {
|
|
|
|
if n := z.Len(); n > 0 {
|
|
|
|
s := make([]string, 0, n)
|
|
|
|
|
|
|
|
z.ForEachMachine(func(p *Machine) bool {
|
|
|
|
s = append(s, p.Name)
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
|
|
|
|
return genEnvStrings(s)
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func genEnvZoneRegion(z *Zone) string {
|
|
|
|
if z != nil && z.region != nil {
|
|
|
|
return z.region.Name
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func genEnvZoneCephMonNames(m Machines) string {
|
|
|
|
var buf strings.Builder
|
|
|
|
m.ForEachMachine(func(p *Machine) bool {
|
|
|
|
if buf.Len() > 0 {
|
|
|
|
_, _ = buf.WriteRune(' ')
|
|
|
|
}
|
|
|
|
_, _ = buf.WriteString(p.Name)
|
|
|
|
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
return buf.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func genEnvZoneCephMonIPs(m Machines) string {
|
|
|
|
var buf strings.Builder
|
|
|
|
m.ForEachMachine(func(p *Machine) bool {
|
|
|
|
addr := p.RingOneAddress()
|
|
|
|
|
|
|
|
if buf.Len() > 0 {
|
|
|
|
_, _ = buf.WriteRune(' ')
|
|
|
|
}
|
|
|
|
_, _ = buf.WriteString(addr.String())
|
|
|
|
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
return buf.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func genEnvZoneCephMonIDs(m Machines) string {
|
|
|
|
var buf strings.Builder
|
|
|
|
m.ForEachMachine(func(p *Machine) bool {
|
|
|
|
if buf.Len() > 0 {
|
|
|
|
_, _ = buf.WriteRune(' ')
|
|
|
|
}
|
|
|
|
_, _ = fmt.Fprintf(&buf, "%v", p.ID)
|
|
|
|
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
return buf.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func cmpOrdered[T core.Ordered](a, b T) int {
|
|
|
|
switch {
|
|
|
|
case a == b:
|
|
|
|
return 0
|
|
|
|
case a < b:
|
|
|
|
return -1
|
|
|
|
default:
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
}
|