forked from muka/go-bluetooth
/
props.go
116 lines (94 loc) · 2.31 KB
/
props.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
package props
import (
"reflect"
"strings"
"github.com/fatih/structs"
"github.com/godbus/dbus"
"github.com/godbus/dbus/prop"
"github.com/muka/go-bluetooth/bluez"
log "github.com/sirupsen/logrus"
)
type PropInfo struct {
prop.Prop
Skip bool
}
func ParseProperties(propertyVal bluez.Properties) map[string]*PropInfo {
t := structs.New(propertyVal)
res := map[string]*PropInfo{}
for _, field := range t.Fields() {
if !field.IsExported() {
continue
}
// if _, ok := field.Value().(dbus.ObjectPath); ok && field.IsZero() {
// log.Debugf("parseProperties: skip empty ObjectPath %s", field.Name())
// continue
// }
propInfo := new(PropInfo)
propInfo.Value = field.Value()
res[field.Name()] = propInfo
tag := field.Tag("dbus")
if tag == "" {
continue
}
parts := strings.Split(tag, ",")
for i := 0; i < len(parts); i++ {
tagKey := parts[i]
tagValue := ""
if strings.Contains(parts[i], "=") {
subpts := strings.Split(parts[i], "=")
tagKey = subpts[0]
tagValue = strings.Join(subpts[1:], "=")
}
if tagKey == "ignore" {
if tagValue == "" {
propInfo.Skip = true
} else {
checkField, ok := t.FieldOk(tagValue)
if !ok {
log.Warnf("%s: field not found, is it avaialable?", tagValue)
continue
}
if !checkField.IsExported() {
log.Warnf("%s: field must be exported. (add a tag `ignore` to avoid exposing it as property)", tagValue)
continue
}
varKind := checkField.Kind()
if varKind != reflect.Bool {
log.Warnf("%s: ignore tag expect a bool property to check, %s given", tagValue, varKind)
continue
}
if checkField.Value().(bool) {
propInfo.Skip = true
}
}
}
// check if empty
if tagKey == "omitEmpty" {
if field.IsZero() {
propInfo.Skip = true
}
}
switch tagKey {
case "emit":
propInfo.Emit = prop.EmitTrue
propInfo.Writable = true
break
case "invalidates":
propInfo.Emit = prop.EmitInvalidates
propInfo.Writable = true
break
case "writable":
propInfo.Writable = true
break
default:
t := reflect.TypeOf(propertyVal)
m, ok := t.MethodByName(tagKey)
if ok {
propInfo.Writable = true
propInfo.Callback = m.Func.Interface().(func(*prop.Change) *dbus.Error)
}
}
}
}
return res
}