jpictl: introduce initial jpictl list
to see the different networks and addresses currently limited to rings zero and one. Signed-off-by: Alejandro Mery <amery@jpi.io>
This commit is contained in:
@@ -0,0 +1,200 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"io"
|
||||
"net/netip"
|
||||
"os"
|
||||
|
||||
"darvaza.org/core"
|
||||
"github.com/spf13/cobra"
|
||||
|
||||
"git.jpi.io/amery/jpictl/pkg/cluster"
|
||||
"git.jpi.io/amery/jpictl/pkg/rings"
|
||||
"git.jpi.io/amery/jpictl/pkg/tools"
|
||||
)
|
||||
|
||||
type inventory struct {
|
||||
r []*cluster.Region
|
||||
z [][]*cluster.Zone
|
||||
}
|
||||
|
||||
func (g *inventory) renderRingZero(out *tools.LazyBuffer) error {
|
||||
ring0 := netip.PrefixFrom(rings.UnsafeRingZeroAddress(0, 0, 0), rings.RingZeroBits)
|
||||
from, to, _ := rings.PrefixToRange(ring0)
|
||||
|
||||
_ = out.Printf("; wg%v\n", 0)
|
||||
_ = out.Printf("%s\t%s-%s\n", ring0, from, to)
|
||||
|
||||
if err := g.renderRingZeroRegions(out); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return g.renderRingZeroZones(out)
|
||||
}
|
||||
|
||||
func (g *inventory) renderRingZeroRegions(out *tools.LazyBuffer) error {
|
||||
for _, r := range g.r {
|
||||
if err := g.renderRingZeroRegion(out, r); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (*inventory) renderRingZeroRegion(out *tools.LazyBuffer, r *cluster.Region) error {
|
||||
addr := rings.UnsafeRingZeroAddress(r.ID, 0, 0)
|
||||
ring0r := netip.PrefixFrom(addr, rings.RingZeroBits+4)
|
||||
from, to, _ := rings.PrefixToRange(ring0r)
|
||||
|
||||
_ = out.Printf("%s\t%s-%s\t# %s\n", ring0r, from, to, r.Name)
|
||||
return nil
|
||||
}
|
||||
|
||||
func (g *inventory) renderRingZeroZones(out *tools.LazyBuffer) error {
|
||||
for i, r := range g.r {
|
||||
for _, z := range g.z[i] {
|
||||
if err := g.renderRingZeroZone(out, r, z); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
func (*inventory) renderRingZeroZone(out *tools.LazyBuffer, r *cluster.Region, z *cluster.Zone) error {
|
||||
addr := rings.UnsafeRingZeroAddress(r.ID, z.ID, 0)
|
||||
ring0rz := netip.PrefixFrom(addr, rings.RingZeroBits+4+4)
|
||||
from, to, _ := rings.PrefixToRange(ring0rz)
|
||||
|
||||
_ = out.Printf("; wg%v: %s (%s)\n", 0, z.Name, r.Name)
|
||||
_ = out.Printf("%s\t%s-%s\t%s\n", ring0rz, from, to, z.Name)
|
||||
|
||||
z.ForEachMachine(func(m *cluster.Machine) bool {
|
||||
if m.IsGateway() {
|
||||
addr, _ := m.RingZeroAddress()
|
||||
cidr := netip.PrefixFrom(addr, 32)
|
||||
_ = out.Printf("%s\t\t%s-%v\n", cidr, m.Name, 0)
|
||||
}
|
||||
|
||||
return false
|
||||
})
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (g *inventory) renderRingOne(out *tools.LazyBuffer) error {
|
||||
for i, r := range g.r {
|
||||
for _, z := range g.z[i] {
|
||||
if err := g.renderRingOneZone(out, r, z); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (*inventory) renderRingOneZone(out *tools.LazyBuffer, r *cluster.Region, z *cluster.Zone) error {
|
||||
ring1, err := rings.RingOnePrefix(r.ID, z.ID)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
from, to, _ := rings.PrefixToRange(ring1)
|
||||
|
||||
_ = out.Printf("; wg%v: %s (%s)\n", 1, z.Name, r.Name)
|
||||
_ = out.Printf("%s\t%s-%s\t%s\n", ring1, from, to, z.Name)
|
||||
|
||||
z.ForEachMachine(func(m *cluster.Machine) bool {
|
||||
addr := m.RingOneAddress()
|
||||
cidr := netip.PrefixFrom(addr, 32)
|
||||
_ = out.Printf("%s\t\t%s-%v\n", cidr, m.Name, 1)
|
||||
return false
|
||||
})
|
||||
return nil
|
||||
}
|
||||
|
||||
func (g *inventory) Marshal() ([]byte, error) {
|
||||
var buf tools.LazyBuffer
|
||||
|
||||
if err := g.renderRingZero(&buf); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
if err := g.renderRingOne(&buf); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return buf.Bytes(), nil
|
||||
}
|
||||
|
||||
func (g *inventory) WriteTo(out io.Writer) (int64, error) {
|
||||
b, err := g.Marshal()
|
||||
if err != nil {
|
||||
return 0, err
|
||||
}
|
||||
|
||||
buf := bytes.NewBuffer(b)
|
||||
return buf.WriteTo(out)
|
||||
}
|
||||
|
||||
func genInventory(m *cluster.Cluster) (*inventory, error) {
|
||||
g := new(inventory)
|
||||
|
||||
g.populateRegions(m)
|
||||
g.populateZones()
|
||||
|
||||
return g, nil
|
||||
}
|
||||
|
||||
func (g *inventory) populateRegions(m *cluster.Cluster) {
|
||||
m.ForEachRegion(func(r *cluster.Region) bool {
|
||||
if r.IsPrimary() {
|
||||
g.r = append(g.r, r)
|
||||
}
|
||||
return false
|
||||
})
|
||||
|
||||
core.SliceSortFn(g.r, func(a, b *cluster.Region) bool {
|
||||
return a.ID < b.ID
|
||||
})
|
||||
}
|
||||
|
||||
func (g *inventory) populateZones() {
|
||||
g.z = make([][]*cluster.Zone, len(g.r))
|
||||
for i, r := range g.r {
|
||||
r.ForEachZone(func(z *cluster.Zone) bool {
|
||||
g.z[i] = append(g.z[i], z)
|
||||
return false
|
||||
})
|
||||
|
||||
core.SliceSortFn(g.z[i], func(a, b *cluster.Zone) bool {
|
||||
return a.ID < b.ID
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
// Command
|
||||
var listCmd = &cobra.Command{
|
||||
Use: "list",
|
||||
Short: "list shows the IP/CIDR inventory",
|
||||
PreRun: setVerbosity,
|
||||
RunE: func(_ *cobra.Command, _ []string) error {
|
||||
m, err := cfg.LoadZones(false)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
out, err := genInventory(m)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
_, err = out.WriteTo(os.Stdout)
|
||||
return err
|
||||
},
|
||||
}
|
||||
|
||||
func init() {
|
||||
rootCmd.AddCommand(listCmd)
|
||||
}
|
||||
Reference in New Issue
Block a user