control-panel-dns/hostnameStatus.go

362 lines
8.4 KiB
Go

/*
figures out if your hostname is valid
then checks if your DNS is setup correctly
*/
package main
import (
"os"
"fmt"
"time"
"reflect"
"strings"
"errors"
"go.wit.com/log"
"go.wit.com/gui/gui"
"go.wit.com/gui/gadgets"
"go.wit.com/gui/cloudflare"
)
type hostnameStatus struct {
ready bool
hidden bool
// hostname string // my hostname. Example: "test.wit.com"
lastname string // used to watch for changes in the hostname
window *gadgets.BasicWindow
// Primary Directives
status *gadgets.OneLiner
summary *gadgets.OneLiner
statusIPv4 *gadgets.OneLiner
statusIPv6 *gadgets.OneLiner
// Details Group
hostShort *gadgets.OneLiner
domainname *gadgets.OneLiner
// what the current IP address your network has given you
currentIPv4 *gadgets.OneLiner
currentIPv6 *gadgets.OneLiner
// what the DNS servers have
dnsA *gadgets.OneLiner
dnsAAAA *gadgets.OneLiner
dnsAPI *gadgets.OneLiner
speed *gadgets.OneLiner
speedActual *gadgets.OneLiner
// Actions
dnsValue *gui.Node
dnsAction *gui.Node
}
func NewHostnameStatusWindow(p *gui.Node) *hostnameStatus {
var hs *hostnameStatus
hs = new(hostnameStatus)
hs.ready = false
hs.hidden = true
// hs.hostname = me.hostname
hs.window = gadgets.NewBasicWindow(p, "fix hostname here" + " Status")
hs.window.Draw()
hs.window.Hide()
group := hs.window.Box().NewGroup("Summary")
grid := group.NewGrid("LookupStatus", 2, 2)
hs.status = gadgets.NewOneLiner(grid, "status").Set("unknown")
hs.statusIPv4 = gadgets.NewOneLiner(grid, "IPv4").Set("unknown")
hs.statusIPv6 = gadgets.NewOneLiner(grid, "IPv6").Set("unknown")
group.Pad()
grid.Pad()
group = hs.window.Box().NewGroup("Details")
grid = group.NewGrid("LookupDetails", 2, 2)
hs.hostShort = gadgets.NewOneLiner(grid, "hostname -s")
hs.domainname = gadgets.NewOneLiner(grid, "domain name")
hs.currentIPv4 = gadgets.NewOneLiner(grid, "Current IPv4")
hs.currentIPv6 = gadgets.NewOneLiner(grid, "Current IPv6")
hs.dnsAPI = gadgets.NewOneLiner(grid, "dns API provider").Set("unknown")
hs.dnsA = gadgets.NewOneLiner(grid, "dns IPv4 resource records").Set("unknown")
hs.dnsAAAA = gadgets.NewOneLiner(grid, "dns IPv6 resource records").Set("unknown")
hs.speed = gadgets.NewOneLiner(grid, "speed").Set("unknown")
hs.speedActual = gadgets.NewOneLiner(grid, "actual").Set("unknown")
group.Pad()
grid.Pad()
group = hs.window.Box().NewGroup("Actions")
grid = group.NewGrid("LookupDetails", 2, 2)
hs.dnsValue = grid.NewLabel("3.4.5.6")
hs.dnsAction = grid.NewButton("CHECK", func () {
log.Warn("should", hs.dnsAction.S, "here for", hs.dnsValue.S)
if (hs.dnsAction.S == "DELETE") {
hs.deleteDNSrecord(hs.dnsValue.S)
}
if (hs.dnsAction.S == "CREATE") {
hs.createDNSrecord(hs.dnsValue.S)
}
})
group.Pad()
grid.Pad()
hs.hidden = false
hs.ready = true
return hs
}
func (hs *hostnameStatus) ValidHostname() bool {
return goodHostname()
}
func (hs *hostnameStatus) GetHostname() string {
return hs.lastname
}
func (hs *hostnameStatus) SetHostname(hostname string) {
if hostname == hs.lastname {return}
log.Log(CHANGE, "the hostname is changing from", hs.lastname, "to", hostname)
hs.lastname = hostname
me.changed = true
if (me.fqdn != nil) {
me.fqdn.SetText(hostname)
}
}
func (hs *hostnameStatus) Domain() string {
if ! hs.Ready() {return ""}
return hs.domainname.Get()
}
func (hs *hostnameStatus) API() string {
if ! hs.Ready() {return ""}
return hs.dnsAPI.Get()
}
func (hs *hostnameStatus) deleteDNSrecord(value string) bool {
log.Info("deleteDNSrecord() START for", value)
log.Info("deleteDNSrecord() hostname =", me.status.GetHostname())
log.Info("deleteDNSrecord() domain =", hs.Domain())
log.Info("deleteDNSrecord() DNS API Provider =", hs.API())
if (hs.API() == "cloudflare") {
log.Info("deleteDNSrecord() Try to delete via cloudflare")
return cloudflare.Delete(hs.Domain(), me.status.GetHostname(), value)
}
return false
}
func (hs *hostnameStatus) createDNSrecord(value string) bool {
log.Info("createDNSrecord() START for", value)
log.Info("createDNSrecord() hostname =", me.status.GetHostname())
log.Info("createDNSrecord() domain =", hs.Domain())
log.Info("createDNSrecord() DNS API Provider =", hs.API())
if (hs.API() == "cloudflare") {
log.Warn("createDNSrecord() Try to create via cloudflare:", me.status.GetHostname(), value)
return cloudflare.Create(hs.Domain(), me.status.GetHostname(), value)
}
return false
}
func (hs *hostnameStatus) Update() {
log.Info("hostnameStatus() Update() START")
if hs == nil {
log.Error(errors.New("hostnameStatus() Update() hs == nil"))
return
}
duration := timeFunction(func () {
hs.updateStatus()
})
s := fmt.Sprint(duration)
hs.set(hs.speedActual, s)
if (duration > 500 * time.Millisecond ) {
hs.set(hs.speed, "SLOW")
} else if (duration > 100 * time.Millisecond ) {
hs.set(hs.speed, "OK")
} else {
hs.set(hs.speed, "FAST")
}
log.Info("hostnameStatus() Update() END")
}
// Returns true if the status is valid
func (hs *hostnameStatus) Ready() bool {
if hs == nil {return false}
return hs.ready
}
// Returns true if IPv4 is working
func (hs *hostnameStatus) IPv4() bool {
if ! hs.Ready() {return false}
if (hs.statusIPv4.Get() == "OK") {
return true
}
if (hs.statusIPv4.Get() == "GOOD") {
return true
}
return false
}
// Returns true if IPv6 is working
func (hs *hostnameStatus) IPv6() bool {
if ! hs.Ready() {return false}
if (hs.statusIPv6.Get() == "GOOD") {
return true
}
return false
}
func (hs *hostnameStatus) setIPv4(s string) {
if ! hs.Ready() {return}
hs.statusIPv4.Set(s)
}
func (hs *hostnameStatus) setIPv6(s string) {
if ! hs.Ready() {return}
hs.statusIPv6.Set(s)
}
func (hs *hostnameStatus) set(a any, s string) {
if ! hs.Ready() {return}
if hs.hidden {
return
}
if a == nil {
return
}
var n *gui.Node
if reflect.TypeOf(a) == reflect.TypeOf(n) {
n = a.(*gui.Node)
n.SetText(s)
return
}
var ol *gadgets.OneLiner
if reflect.TypeOf(a) == reflect.TypeOf(ol) {
ol = a.(*gadgets.OneLiner)
if ol == nil {
// log.Println("ol = nil", reflect.TypeOf(a), "a =", a)
return
}
// log.Println("SETTING ol:", ol)
ol.Set(s)
return
}
log.Warn("unknown type TypeOf(a) =", reflect.TypeOf(a), "a =", a)
os.Exit(0)
}
// returns true if AAAA record already exists in DNS
func (hs *hostnameStatus) existsAAAA(s string) bool {
log.Log(NOW, "existsAAAA() try to see if AAAA is already set", s)
return false
}
// figure out if I'm missing any IPv6 address in DNS
func (hs *hostnameStatus) missingAAAA() bool {
var aaaa []string
aaaa = dhcpAAAA()
for _, s := range aaaa {
log.Log(NET, "my actual AAAA = ",s)
if hs.existsAAAA(s) {
log.Log(NOW, "my actual AAAA already exists in DNS =",s)
} else {
log.Log(NOW, "my actual AAAA is missing from DNS",s)
hs.dnsValue.SetText(s)
hs.dnsAction.SetText("CREATE")
return true
}
}
return false
}
func (hs *hostnameStatus) updateStatus() {
if ! hs.Ready() { return }
var s string
var vals []string
log.Log(STATUS, "updateStatus() START")
hs.hostShort.Set(me.hostshort.S)
hs.domainname.Set(me.domainname.S)
if hs.ValidHostname() {
vals = lookupDoH(hs.GetHostname(), "AAAA")
log.Log(STATUS, "DNS IPv6 Addresses for ", hs.GetHostname(), "=", vals)
if len(vals) == 0 {
s = "(none)"
} else {
hs.setIPv6("Check for real IPv6 addresses here")
if hs.missingAAAA() {
hs.setIPv6("Add the missing IPv6 address")
}
for _, addr := range vals {
log.Log(STATUS, addr)
s += addr + " (DELETE)" + "\n"
hs.setIPv6("NEEDS DELETE")
hs.dnsValue.SetText(addr)
hs.dnsAction.SetText("DELETE")
}
}
hs.set(hs.dnsAAAA, s)
vals = lookupDoH(hs.GetHostname(), "A")
log.Log(STATUS, "IPv4 Addresses for ", hs.GetHostname(), "=", vals)
s = strings.Join(vals, "\n")
if (s == "") {
s = "(none)"
hs.setIPv4("NEEDS CNAME")
}
hs.set(hs.dnsA, s)
vals = lookupDoH(hs.GetHostname(), "CNAME")
s = strings.Join(vals, "\n")
if (s != "") {
hs.set(hs.dnsA, "CNAME " + s)
hs.setIPv4("GOOD")
}
}
hs.currentIPv4.Set(me.IPv4.S)
hs.currentIPv6.Set(me.IPv6.S)
if hs.IPv4() && hs.IPv6() {
hs.status.Set("GOOD")
} else {
hs.status.Set("BROKEN")
}
hs.dnsAPI.Set(me.DnsAPI.S)
}
func (hs *hostnameStatus) Show() {
log.Log(STATUS, "hostnameStatus.Show() window")
if hs.hidden {
hs.window.Show()
}
hs.hidden = false
}
func (hs *hostnameStatus) Hide() {
log.Log(STATUS, "hostnameStatus.Hide() window")
if ! hs.hidden {
hs.window.Hide()
}
hs.hidden = true
}