-
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbobcat.go
136 lines (120 loc) · 3.45 KB
/
bobcat.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
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
package main
import (
json "encoding/json"
"io/ioutil"
"log"
"net/http"
"net/url"
"strconv"
"time"
)
type Bobcat struct {
Debug bool
ErrorLimit int
periodSeconds int
address string
eventChannel chan<- BobcatStatus
errorCount int
}
type BobcatStatus struct {
Status string `json:"status"`
Gap int64 `json:"gap"`
MinerHeight int64 `json:"miner_height"`
BlockchainHeight int64 `json:"blockchain_height"`
Epoch int64 `json:"epoch"`
LatencyMs int64 `json:"latency_ms"`
ErrorCount int `json:"error_count"`
Valid bool `json:"valid"`
}
type bobcatStatusJson struct {
Status string `json:"status"`
Gap string `json:"gap"`
MinerHeight string `json:"miner_height"`
BlockchainHeight string `json:"blockchain_height"`
Epoch string `json:"epoch"`
}
var HTTP_TIMEOUT_SECONDS = 30
func (bobcat Bobcat) Begin() {
// SANITY CHECK
if bobcat.address == "" {
panic("Empty Bobcat address!")
}
if bobcat.ErrorLimit == 0 {
bobcat.ErrorLimit = 1
}
bobcatUrl := url.URL{Scheme: "http", Host: bobcat.address, Path: "/status.json"}
client := http.Client{
Timeout: time.Duration(HTTP_TIMEOUT_SECONDS) * time.Second,
}
for {
if bobcat.Debug {
log.Printf("Fetching bobcat status from %s", bobcatUrl.String())
}
requestStart := time.Now()
resp, err := client.Get(bobcatUrl.String())
if err != nil {
bobcat.errorCount++
if bobcat.errorCount >= bobcat.ErrorLimit {
bobcatStatus := BobcatStatus{
Status: "Error",
LatencyMs: int64(time.Since(requestStart) / time.Millisecond),
ErrorCount: bobcat.errorCount,
Valid: false,
}
bobcat.eventChannel <- bobcatStatus
}
log.Printf("Error while fetching bobcat status: %s \n", err)
time.Sleep(time.Duration(bobcat.periodSeconds) * time.Second)
continue
}
requestEnd := time.Since(requestStart)
bobcat.errorCount = 0
if bobcat.Debug {
log.Printf("Got response from bobcat! Latency: %v\n", requestEnd)
}
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
log.Printf("Error while reading response from bobcat : %s \n", err)
time.Sleep(time.Duration(bobcat.periodSeconds) * time.Second)
continue
}
jsonResponse := &bobcatStatusJson{}
if err := json.Unmarshal(body, &jsonResponse); err != nil {
log.Printf("Error parsing JSON body from bobcat (\"%s\"): %s \n", string(body), err)
time.Sleep(time.Duration(bobcat.periodSeconds) * time.Second)
continue
}
// PARSE STRINGFUL JSON INTO STRUCT WITH INTS
bobcatStatus := BobcatStatus{
Status: jsonResponse.Status,
LatencyMs: int64(requestEnd / time.Millisecond),
Valid: true,
}
gap, err := strconv.ParseInt(jsonResponse.Gap, 10, 64)
if err != nil {
bobcatStatus.Valid = false
} else {
bobcatStatus.Gap = gap
}
minerHeight, err := strconv.ParseInt(jsonResponse.MinerHeight, 10, 64)
if err != nil {
bobcatStatus.Valid = false
} else {
bobcatStatus.MinerHeight = minerHeight
}
blockchainHeight, err := strconv.ParseInt(jsonResponse.BlockchainHeight, 10, 64)
if err != nil {
bobcatStatus.Valid = false
} else {
bobcatStatus.BlockchainHeight = blockchainHeight
}
epoch, err := strconv.ParseInt(jsonResponse.Epoch, 10, 64)
if err != nil {
bobcatStatus.Valid = false
} else {
bobcatStatus.Epoch = epoch
}
bobcat.eventChannel <- bobcatStatus
time.Sleep(time.Duration(bobcat.periodSeconds) * time.Second)
}
}