|
|
@ -2,17 +2,25 @@ package cluster |
|
|
|
|
|
|
|
|
|
|
|
import ( |
|
|
|
import ( |
|
|
|
"io/fs" |
|
|
|
"io/fs" |
|
|
|
|
|
|
|
"path" |
|
|
|
"sort" |
|
|
|
"sort" |
|
|
|
|
|
|
|
|
|
|
|
"darvaza.org/core" |
|
|
|
"darvaza.org/core" |
|
|
|
) |
|
|
|
) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const ( |
|
|
|
|
|
|
|
// ZoneRegionsFileName indicates the file containing
|
|
|
|
|
|
|
|
// region names as references
|
|
|
|
|
|
|
|
ZoneRegionsFileName = "regions" |
|
|
|
|
|
|
|
) |
|
|
|
|
|
|
|
|
|
|
|
func (m *Cluster) scan(opts *ScanOptions) error { |
|
|
|
func (m *Cluster) scan(opts *ScanOptions) error { |
|
|
|
for _, fn := range []func(*ScanOptions) error{ |
|
|
|
for _, fn := range []func(*ScanOptions) error{ |
|
|
|
m.scanDirectory, |
|
|
|
m.scanDirectory, |
|
|
|
m.scanMachines, |
|
|
|
m.scanMachines, |
|
|
|
m.scanZoneIDs, |
|
|
|
m.scanZoneIDs, |
|
|
|
m.scanSort, |
|
|
|
m.scanSort, |
|
|
|
|
|
|
|
m.initRegions, |
|
|
|
m.scanGateways, |
|
|
|
m.scanGateways, |
|
|
|
m.scanCephMonitors, |
|
|
|
m.scanCephMonitors, |
|
|
|
} { |
|
|
|
} { |
|
|
@ -24,7 +32,7 @@ func (m *Cluster) scan(opts *ScanOptions) error { |
|
|
|
return nil |
|
|
|
return nil |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
func (m *Cluster) scanDirectory(_ *ScanOptions) error { |
|
|
|
func (m *Cluster) scanDirectory(opts *ScanOptions) error { |
|
|
|
// each directory is a zone
|
|
|
|
// each directory is a zone
|
|
|
|
entries, err := fs.ReadDir(m.dir, ".") |
|
|
|
entries, err := fs.ReadDir(m.dir, ".") |
|
|
|
if err != nil { |
|
|
|
if err != nil { |
|
|
@ -33,16 +41,14 @@ func (m *Cluster) scanDirectory(_ *ScanOptions) error { |
|
|
|
|
|
|
|
|
|
|
|
for _, e := range entries { |
|
|
|
for _, e := range entries { |
|
|
|
if e.IsDir() { |
|
|
|
if e.IsDir() { |
|
|
|
z, err := m.newZone(e.Name()) |
|
|
|
ok, err := m.scanSubdirectory(opts, e.Name()) |
|
|
|
switch { |
|
|
|
switch { |
|
|
|
case err != nil: |
|
|
|
case err != nil: |
|
|
|
return core.Wrap(err, e.Name()) |
|
|
|
return core.Wrap(err, e.Name()) |
|
|
|
case z.Machines.Len() == 0: |
|
|
|
case !ok: |
|
|
|
z.warn(nil). |
|
|
|
m.warn(nil). |
|
|
|
WithField("zone", z.Name). |
|
|
|
WithField("zone", e.Name()). |
|
|
|
Print("empty") |
|
|
|
Print("empty") |
|
|
|
default: |
|
|
|
|
|
|
|
m.Zones = append(m.Zones, z) |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
@ -50,6 +56,27 @@ func (m *Cluster) scanDirectory(_ *ScanOptions) error { |
|
|
|
return nil |
|
|
|
return nil |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func (m *Cluster) scanSubdirectory(_ *ScanOptions, name string) (bool, error) { |
|
|
|
|
|
|
|
z, err := m.newZone(name) |
|
|
|
|
|
|
|
switch { |
|
|
|
|
|
|
|
case err != nil: |
|
|
|
|
|
|
|
// somewhere went wrong scanning the subdirectory
|
|
|
|
|
|
|
|
return false, err |
|
|
|
|
|
|
|
case z.Machines.Len() > 0: |
|
|
|
|
|
|
|
// zones have machines and the regions they belong
|
|
|
|
|
|
|
|
m.Zones = append(m.Zones, z) |
|
|
|
|
|
|
|
return true, nil |
|
|
|
|
|
|
|
case len(z.Regions) > 0: |
|
|
|
|
|
|
|
// regions have no machines but can include
|
|
|
|
|
|
|
|
// other regions
|
|
|
|
|
|
|
|
m.appendRegionRegions(name, z.Regions...) |
|
|
|
|
|
|
|
return true, nil |
|
|
|
|
|
|
|
default: |
|
|
|
|
|
|
|
// empty
|
|
|
|
|
|
|
|
return false, nil |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
func (m *Cluster) newZone(name string) (*Zone, error) { |
|
|
|
func (m *Cluster) newZone(name string) (*Zone, error) { |
|
|
|
z := &Zone{ |
|
|
|
z := &Zone{ |
|
|
|
zones: m, |
|
|
|
zones: m, |
|
|
@ -154,31 +181,65 @@ func (z *Zone) scan() error { |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
for _, e := range entries { |
|
|
|
for _, e := range entries { |
|
|
|
if e.IsDir() { |
|
|
|
name := e.Name() |
|
|
|
m := &Machine{ |
|
|
|
|
|
|
|
zone: z, |
|
|
|
|
|
|
|
logger: z, |
|
|
|
|
|
|
|
Name: e.Name(), |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
m.debug(). |
|
|
|
switch { |
|
|
|
WithField("node", m.Name). |
|
|
|
case name == ZoneRegionsFileName: |
|
|
|
|
|
|
|
err = z.loadRegions() |
|
|
|
|
|
|
|
case e.IsDir(): |
|
|
|
|
|
|
|
err = z.scanSubdirectory(name) |
|
|
|
|
|
|
|
default: |
|
|
|
|
|
|
|
z.warn(nil). |
|
|
|
WithField("zone", z.Name). |
|
|
|
WithField("zone", z.Name). |
|
|
|
Print("found") |
|
|
|
WithField("filename", name). |
|
|
|
|
|
|
|
Print("unknown") |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
if err := m.init(); err != nil { |
|
|
|
if err != nil { |
|
|
|
m.error(err). |
|
|
|
return err |
|
|
|
WithField("node", m.Name). |
|
|
|
} |
|
|
|
WithField("zone", z.Name). |
|
|
|
} |
|
|
|
Print() |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return err |
|
|
|
return nil |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
z.Machines = append(z.Machines, m) |
|
|
|
func (z *Zone) loadRegions() error { |
|
|
|
|
|
|
|
filename := path.Join(z.Name, ZoneRegionsFileName) |
|
|
|
|
|
|
|
regions, err := z.zones.ReadLines(filename) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if err == nil { |
|
|
|
|
|
|
|
// parsed
|
|
|
|
|
|
|
|
err = z.appendRegions(regions...) |
|
|
|
|
|
|
|
if err != nil { |
|
|
|
|
|
|
|
err = core.Wrap(err, filename) |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return err |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func (z *Zone) scanSubdirectory(name string) error { |
|
|
|
|
|
|
|
m := &Machine{ |
|
|
|
|
|
|
|
zone: z, |
|
|
|
|
|
|
|
logger: z, |
|
|
|
|
|
|
|
Name: name, |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
m.debug(). |
|
|
|
|
|
|
|
WithField("node", m.Name). |
|
|
|
|
|
|
|
WithField("zone", z.Name). |
|
|
|
|
|
|
|
Print("found") |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if err := m.init(); err != nil { |
|
|
|
|
|
|
|
m.error(err). |
|
|
|
|
|
|
|
WithField("node", m.Name). |
|
|
|
|
|
|
|
WithField("zone", z.Name). |
|
|
|
|
|
|
|
Print() |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return err |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
z.Machines = append(z.Machines, m) |
|
|
|
return nil |
|
|
|
return nil |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|