-
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.go
113 lines (102 loc) · 4.59 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
package main
import (
"encoding/binary"
"encoding/hex"
"errors"
"fmt"
"github.com/go-kit/log/level"
"github.com/go-playground/validator/v10"
blue "github.com/myoung34/bluey/bluey"
"github.com/myoung34/bluey/cli"
"github.com/myoung34/bluey/emitters"
"github.com/myoung34/gatt"
"github.com/myoung34/gatt/examples/option"
"reflect"
"strings"
"time"
)
var config = blue.Config{}
var validate = validator.New()
var EmittersMap = map[string]interface{}{
"webhook.emit": emitters.WebhookEmit,
"stdout.emit": emitters.StdoutEmit,
"datadog.emit": emitters.DatadogEmit,
"sqlite.emit": emitters.SQLiteEmit,
"mqtt.emit": emitters.MQTTEmit,
}
func main() {
blue.EnableLogging()
config = cli.ParseArgs()
blue.SetLogging(fmt.Sprintf("%s", config.ConfigData.Get("general.logging_level")))
level.Debug(blue.Logger).Log("main", fmt.Sprintf("Using emitter: %s", config.EnabledEmitter))
level.Info(blue.Logger).Log("main", "Scanning...")
device, err := gatt.NewDevice(option.DefaultClientOptions...)
if err != nil {
level.Error(blue.Logger).Log("main", fmt.Sprintf("Failed to open device, err: %s\n", err))
return
}
device.Handle(gatt.PeripheralDiscovered(OnPeripheralDiscovered))
device.Init(OnStateChanged)
select {}
}
func OnStateChanged(device gatt.Device, s gatt.State) {
switch s {
case gatt.StatePoweredOn:
level.Info(blue.Logger).Log("main.OnStateChanged", "Scanning...")
device.Scan([]gatt.UUID{}, true)
return
default:
device.StopScanning()
}
}
func NewBlue(data []byte) (blue.Payload, error) {
// http://www.havlena.net/wp-content/themes/striking/includes/timthumb.php?src=/wp-content/uploads/ibeacon-packet.png&w=600&zc=1
//pkt = b' \x04>* \x02\x01x03\x01w\t \xbc\xd0W\xef\x1e\x02\x01\x04\x1a\xffL\x00\x02\x15 \xa4\x95\xbb0\xc5\xb1KD\xb5\x12\x13p\xf0-t\xde \x00B \x03\xf7 \xc5\xa7' # noqa
// | | | | | | | | | # noqa
// | preamble+header | PDU | # noqa
// | 3 bytes | x bytes (plen) | # noqa
// | | | mac addr | uuid | unused data | major| minor | tx | # noqa
// | | | | |
if len(data) < 25 || binary.BigEndian.Uint32(data) != 0x4c000215 {
return blue.Payload{}, errors.New("not an iBeacon")
}
return blue.Payload{
ID: strings.ToLower(strings.Replace(strings.ToUpper(hex.EncodeToString(data[4:8])+"-"+hex.EncodeToString(data[8:10])+"-"+hex.EncodeToString(data[10:12])+"-"+hex.EncodeToString(data[12:14])+"-"+hex.EncodeToString(data[14:20])), "-", "", -1)),
Major: binary.BigEndian.Uint16(data[20:22]),
Minor: binary.BigEndian.Uint16(data[22:24]),
}, nil
}
func OnPeripheralDiscovered(p gatt.Peripheral, a *gatt.Advertisement, rssi int) {
_blue, err := NewBlue(a.ManufacturerData)
if err == nil {
payload := blue.Payload{
ID: _blue.ID,
Mac: p.ID(),
Major: _blue.Major,
Minor: _blue.Minor,
Rssi: rssi,
Timestamp: time.Now().UTC().Unix(),
}
level.Info(blue.Logger).Log("main.OnPeripheralDiscovered", payload.ID)
err = validate.Struct(payload)
if err == nil {
level.Info(blue.Logger).Log("main.OnPeripheralDiscovered", fmt.Sprintf("%s [%s] major: %d minor: %d rssi: %d", payload.ID, payload.Nickname, payload.Major, payload.Minor, rssi))
returnStr, _ := callEmitter(fmt.Sprintf("%s.emit", config.EnabledEmitter), payload, config.ConfigData.Get(config.EnabledEmitter))
level.Info(blue.Logger).Log("main.OnPeripheralDiscovered", returnStr)
}
}
}
func callEmitter(funcName string, payload blue.Payload, emitterConfig interface{}) (result interface{}, err error) {
level.Info(blue.Logger).Log("main.callEmitter", fmt.Sprintf("Attempting to call %+v", funcName))
_, ok := EmittersMap[funcName]
if !ok {
panic(fmt.Sprintf("Emitter '%s' not found'", strings.Split(funcName, ".")[0]))
}
f := reflect.ValueOf(EmittersMap[funcName])
in := make([]reflect.Value, 2)
in[0] = reflect.ValueOf(payload)
in[1] = reflect.ValueOf(emitterConfig)
var res = f.Call(in)
result = res[0].Interface()
return
}