2022-06-24 02:41:58 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"infra-sshfp-cf/cloudflare"
|
|
|
|
"infra-sshfp-cf/config"
|
|
|
|
"infra-sshfp-cf/consul"
|
|
|
|
"infra-sshfp-cf/sshfp"
|
|
|
|
"infra-sshfp-cf/statestore"
|
2022-06-27 17:47:07 +02:00
|
|
|
"os"
|
2022-06-24 02:41:58 +02:00
|
|
|
|
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
//Debug loglevel
|
2022-06-27 17:47:07 +02:00
|
|
|
logrus.SetLevel(logrus.DebugLevel)
|
2022-06-24 02:41:58 +02:00
|
|
|
|
|
|
|
//Create configuration components
|
2022-06-27 17:47:07 +02:00
|
|
|
// Get config file name from args, if empty - try to configure from ENVs
|
|
|
|
var configFilename string = ""
|
|
|
|
if len(os.Args) > 1 {
|
|
|
|
configFilename = os.Args[1]
|
|
|
|
}
|
|
|
|
|
2022-06-24 02:41:58 +02:00
|
|
|
cfgService := config.NewService(config.NewFileRepository())
|
2022-06-27 17:47:07 +02:00
|
|
|
config, err := cfgService.LoadConfig(configFilename)
|
2022-06-24 02:41:58 +02:00
|
|
|
if err != nil {
|
|
|
|
logrus.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
//Create cloudflare components
|
|
|
|
cloudflare := cloudflare.NewService(cloudflare.NewRepository(config.CloudflareToken, config.DomainName))
|
|
|
|
|
|
|
|
//Create STDIN Listener and start listening on (blocking operation)
|
|
|
|
consul := consul.NewService(consul.NewStdinRepository())
|
|
|
|
err = consul.LoadData()
|
|
|
|
|
|
|
|
//Code below is executed upon data receipt
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
logrus.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
hosts := consul.GetHostnames()
|
|
|
|
|
|
|
|
//Open statestore
|
|
|
|
statestore := statestore.NewService(statestore.NewMapRepository("statestore.json"))
|
|
|
|
|
|
|
|
//Iterate over hosts and check modify indexes
|
|
|
|
for _, hostname := range hosts {
|
|
|
|
modifiedIndex := consul.GetModifiedIndex(hostname)
|
|
|
|
modified, _ := statestore.CheckIfModified(hostname, modifiedIndex)
|
|
|
|
|
|
|
|
if modified {
|
|
|
|
//Check if hosts has A record in CF - if not ignore
|
|
|
|
exists, err := cloudflare.FindHostByName(hostname)
|
|
|
|
if err != nil || !exists {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
//Generate DNS records based on metadata
|
|
|
|
sshfp := sshfp.NewService()
|
|
|
|
consulKeys := sshfp.ParseConsulSSHRecords(consul.GetMetaData(hostname))
|
|
|
|
|
|
|
|
//GetCurrent configuration
|
|
|
|
cloudflareKeys, err := cloudflare.GetSSHFPRecordsForHost(hostname)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
configPlan := sshfp.PrepareConfiguration(hostname, cloudflareKeys, consulKeys)
|
|
|
|
|
|
|
|
//ConfigPlan is empty, but host was flagged as modified. It's very likely that is a new host, or db is corrupted
|
|
|
|
if len(configPlan) == 0 {
|
|
|
|
statestore.SaveState(hostname, modifiedIndex)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(configPlan) > 0 {
|
|
|
|
sshfp.PrintConfigPlan(configPlan)
|
|
|
|
itemsApplied, err := cloudflare.ApplyConfigPlan(configPlan)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Error(err)
|
|
|
|
}
|
|
|
|
if itemsApplied == len(configPlan) {
|
|
|
|
statestore.SaveState(hostname, modifiedIndex)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-27 17:47:07 +02:00
|
|
|
//Purge old hosts
|
|
|
|
hosts, _ = statestore.GetStalledHosts(config.HostTimeout)
|
|
|
|
for _, host := range hosts {
|
|
|
|
err := cloudflare.DeleteSSHFPRecordsForHost(host)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Fatalf("Cannot delete records for host: %s", host)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
statestore.PurgeStalledHosts(config.HostTimeout)
|
|
|
|
|
2022-06-24 02:41:58 +02:00
|
|
|
}
|