-
Notifications
You must be signed in to change notification settings - Fork 4
/
db.go
163 lines (151 loc) · 3.77 KB
/
db.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
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
package main
import (
"database/sql"
"log"
"math"
"time"
)
func create_table(db *sql.DB, name string, ddl string) {
row := db.QueryRow("SELECT count(sql) FROM sqlite_master WHERE name=?", name)
var count int32
err := row.Scan(&count)
if err != nil {
log.Fatal("Could not check table status ")
}
if count == 0 {
_, err = db.Exec(ddl)
if err != nil {
log.Fatal("Could not create dests table", err)
}
}
}
func db_init(db *sql.DB) {
create_table(db, "dests", "CREATE TABLE dests (host TEXT PRIMARY KEY)")
create_table(db, "pings", "CREATE TABLE pings (time REAL, host TEXT, ip TEXT, rtt REAL, PRIMARY KEY(time, host))")
}
func get_dests(db *sql.DB) []string {
dests := make([]string, 0)
rows, err := db.Query("SELECT host FROM dests")
if err != nil {
log.Fatal(err)
}
for rows.Next() {
var ip string
err = rows.Scan(&ip)
if err != nil {
log.Fatal(err)
}
dests = append(dests, ip)
}
return dests
}
func db_add_dest(db *sql.DB, host string) {
log.Printf("adding host %s", host)
_, err := db.Exec("insert into dests (host) values (?)", host)
if err != nil {
log.Fatal("could not insert host into destinations", err)
}
}
func db_del_dest(db *sql.DB, host string) {
log.Printf("deleting host %s", host)
_, err := db.Exec("delete from dests where host=?", host)
if err != nil {
log.Fatal("could not delete host from destinations", err)
}
}
func record(db *sql.DB, r *Result) {
if *verbose {
log.Printf("ping %s(%s) = %v\n", r.Host, r.IP, r.Rtt)
}
res, err := db.Exec("insert or replace into pings (time, host, ip, rtt) values (julianday('now'), ?, ?, ?)", r.Host, r.IP, 1e-6*float32(r.Rtt.Nanoseconds()))
if err != nil {
log.Fatal(err)
}
rowCount, err := res.RowsAffected()
if err != nil {
log.Fatal(err)
}
if rowCount != 1 {
log.Fatal("could not insert row in pings table", rowCount)
}
}
func ResultWorker(db *sql.DB) chan *Result {
in := make(chan *Result, 5)
go func() {
for {
r := <-in
if r == nil {
return
}
record(db, r)
}
}()
return in
}
func get_data(db *sql.DB, since float64) (header []string, ordered []int64, points map[int64][]float64) {
rows, err := db.Query("SELECT DISTINCT host FROM dests ORDER BY 1")
var host string
cols := make(map[string]int, 0)
colnum := 0
for rows.Next() {
err = rows.Scan(&host)
if err != nil {
log.Fatal(err)
}
colnum += 1
cols[host] = colnum
}
rows, err = db.Query("SELECT time, host, ip, rtt FROM pings WHERE time > julianday('now')-? AND time > ?ORDER by 1, 2", display.Seconds()/86400.0, 2440587.5+since/86400000.0)
if err != nil {
log.Fatal(err)
}
points = make(map[int64][]float64, 0)
var col int
var ok bool
var rounded time.Time
var row []float64
ordered = make([]int64, 0)
for rows.Next() {
var fts, rtt float64
var ip string
err = rows.Scan(&fts, &host, &ip, &rtt)
if err != nil {
log.Fatal(err)
}
// julian day to Unix seconds
fts = (fts - 2440587.5) * 86400.0
rounded = time.Unix(int64(fts), int64(1e9*(fts-math.Trunc(fts))))
// XXX this rounding down is not very clean, should try to run pings
// XXX at the exact times instead
rounded = rounded.Round(*interval)
ts := rounded.Unix() * 1000
col, ok = cols[host]
if !ok {
log.Printf("ERROR: skipping host=%v not found in dests\n", host)
continue
}
row, ok = points[ts]
if !ok {
row = make([]float64, colnum)
ordered = append(ordered, ts)
}
for len(row) < col {
row = append(row, 0.0)
}
row[col-1] = rtt
points[ts] = row
}
header = make([]string, colnum+1)
header[0] = "Date"
for colname, col := range cols {
header[col] = colname
}
// fill out the first rows that may be missing columns
num_cols := len(cols)
for _, ts := range ordered {
for len(points[ts]) < num_cols {
points[ts] = append(points[ts], 0.0)
}
}
return header, ordered, points
}