-
Notifications
You must be signed in to change notification settings - Fork 1
/
main.go
119 lines (90 loc) · 2.19 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
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
package main
import (
"flag"
"fmt"
"net/http"
"time"
"github.com/rcrowley/go-metrics"
)
var addr = flag.String("addr", "0.0.0.0:8080", "WebSocket server address")
var minRange = flag.Float64("min-range", 500, "The minimum range")
var alpha = flag.Float64("alpha", 0.1, "The histogram decay alpha")
var threshold = flag.Float64("threshold", 0, "The threshold for on/off values")
var freq = flag.Int("freq", 60, "Sample publish frequency")
var debug = flag.String("debug", "", `Debug server address e.g. ":1234"`)
var usbPrefix = flag.String("usb-prefix", "cu.usbmodem", "The prefix of usb devices to consider")
func main() {
// parse flags
flag.Parse()
// run debug
if *debug != "" {
go func() {
panic(http.ListenAndServe(*debug, uiHandler()))
}()
}
// open stream
go stream(*addr)
// manage devices
go manage()
// process samples
process()
}
func process() {
// timeout
timeout := time.Second / time.Duration(*freq)
// prepare histograms
var histogram []metrics.Sample
for i := 0; i < 12; i++ {
histogram = append(histogram, metrics.NewExpDecaySample(80, *alpha))
}
for {
// get samples
l := left.Load().(sample)
r := right.Load().(sample)
// merge samples
s := make(sample, 12)
copy(s, l)
copy(s[6:], r)
// prepare result
result := make(sample, 12)
minimums := make(sample, 12)
maximums := make(sample, 12)
// calculate result
for i, v := range s {
// add to histogram
histogram[i].Update(int64(v))
// get quantiles
min := histogram[i].Percentile(0.15)
max := histogram[i].Percentile(0.95)
// adjust range
if max-min < +*minRange {
max = min + *minRange
}
// get value
v := clamp(scale(v, min, max, 0, 1), 0, 1)
// apply threshold if set
if *threshold > 0 {
if v > *threshold {
v = 1
} else {
v = 0
}
}
// set result
result[i] = v
minimums[i] = min
maximums[i] = max
}
// emit
emit(result)
// get min and max
min, _ := minimums.minMax()
_, max := maximums.minMax()
// debug
if *debug != "" {
fmt.Printf("Values: %s | Range %.2f - %.2f | Devices: %d | Clients: %d\n", result.String(), min, max, numDevices, numClients)
}
// sleep
time.Sleep(timeout)
}
}