-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.go
86 lines (70 loc) · 1.95 KB
/
main.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
package main
import (
"context"
"fmt"
"log"
"net/http"
"sync"
"time"
)
func main() {
// Strip log timestamp to catch repeated messages in main loop
log := log.New(customLogWriter{}, "", 0)
cfg, err := parseConfig()
if err != nil {
log.Panic(err)
}
ctx := context.Background()
httpClient := createHTTPClient()
api, err := createAPIClient(cfg.APIKey)
if err != nil {
log.Panic(err)
}
// Fetch initial DNS states to be kept updated without additional API calls
dnsStates := make(map[string]*DNSState)
for _, record := range cfg.Records {
activeRecord, err := fetchARecords(api, ctx, record)
if err != nil {
log.Panicf("Error fetching DNS record for %v: %v", record.Domain, err)
}
if len(activeRecord) == 0 {
log.Panicf("No DNS records found for %v: %v", record.Domain, err)
}
// Pointer to struct to reflect state across funcs, goroutines, etc.
dnsStates[record.Domain] = &DNSState{
ActiveIP: activeRecord[0].Content,
ID: activeRecord[0].ID,
}
}
lastLogMsgs := make(map[string]string)
var wg sync.WaitGroup
for {
logCh := make(chan logEntry, len(cfg.Records))
for _, r := range cfg.Records {
wg.Add(1)
go func(httpClient *http.Client, r Record, s *DNSState, logCh chan logEntry) {
defer wg.Done()
ip, err := getResponsiveIP(httpClient, r, logCh)
if err != nil {
sendLogEntry(logCh, r.Domain, fmt.Sprintf("%s: Error getting responsive IP: %v", r.Domain, err))
return
}
manageDNS(ctx, api, r, s, ip, logCh)
}(httpClient, r, dnsStates[r.Domain], logCh)
}
// Log without consecutive repetitions since we're running in a loop
go func() {
for entry := range logCh {
if lastLogMsgs[entry.domain] != entry.msg {
log.Printf("[%s]: %s\n", entry.timestamp.Format("2006-01-02 15:04:05"), entry.msg)
lastLogMsgs[entry.domain] = entry.msg
}
}
}()
go func() {
wg.Wait()
close(logCh)
}()
time.Sleep(time.Duration(cfg.CheckInterval) * time.Second)
}
}