-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdecoder.go
150 lines (131 loc) · 4.23 KB
/
decoder.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
package alarmdecoder
import (
"bufio"
"io"
"strconv"
"strings"
"github.com/pkg/errors"
)
// ParseMessage parses a message from AD2PI.
func ParseMessage(s string) (Message, error) {
m := Message{
UnparsedMessage: s,
}
parts := strings.Split(s, ",")
if len(parts) != 4 {
return Message{}, errors.Errorf("expected 4 parts got: %#v", parts)
}
bits := parts[0]
m.Ready = bits[1] == '1'
m.ArmedAway = bits[2] == '1'
m.ArmedHome = bits[3] == '1'
m.BacklightOn = bits[4] == '1'
m.ProgrammingMode = bits[5] == '1'
var err error
m.Beeps, err = strconv.Atoi(bits[6:7])
if err != nil {
return Message{}, err
}
m.ZoneBypassed = bits[7] == '1'
m.ACPower = bits[8] == '1'
m.ChimeEnabled = bits[9] == '1'
m.AlarmHasOccured = bits[10] == '1'
m.AlarmSounding = bits[11] == '1'
m.BatteryLow = bits[12] == '1'
m.EntryDelayDisabled = bits[13] == '1'
m.Fire = bits[14] == '1'
m.SystemIssue = bits[15] == '1'
m.PerimeterOnly = bits[16] == '1'
m.Mode = bits[18:19]
m.Zone = parts[1]
m.RawData = parts[2]
msg := parts[3]
m.KeypadMessage = strings.TrimSpace(msg[1 : len(msg)-1])
return m, nil
}
// Message contains
type Message struct {
UnparsedMessage string
// Bit field
// The bit field present on the keypad messages is where you're going to get
// most of the information on your alarm system's current state. These are
// all represented by a zero or one aside from the one exception. (beep)
// 1 Indicates if the panel is READY
Ready bool
// 2 Indicates if the panel is ARMED AWAY
ArmedAway bool
// 3 Indicates if the panel is ARMED HOME
ArmedHome bool
// 4 Indicates if the keypad backlight is on
BacklightOn bool
// 5 Indicates if the keypad is in programming mode
ProgrammingMode bool
// 6 Number (1-7) indicating how many beeps are associated with the message
Beeps int
// 7 Indicates that a zone has been bypassed
ZoneBypassed bool
// 8 Indicates if the panel is on AC power
ACPower bool
// 9 Indicates if the chime is enabled
ChimeEnabled bool
// 10 Indicates that an alarm has occurred. This is sticky and will be cleared after a second disarm.
AlarmHasOccured bool
// 11 Indicates that an alarm is currently sounding. This is cleared after the first disarm.
AlarmSounding bool
// 12 Indicates that the battery is low
BatteryLow bool
// 13 Indicates that entry delay is off (ARMED INSTANT/MAX)
EntryDelayDisabled bool
// 14 Indicates that there is a fire
Fire bool
// 15 Indicates a system issue
SystemIssue bool
// 16 Indicates that the panel is only watching the perimeter (ARMED STAY/NIGHT)
PerimeterOnly bool
// 17 System specific bits. 4 bits packed into a HEX Nibble [0-9,A-F]
// 18 Ademco or DSC Mode A or D
Mode string
// 19 Unused
// 20 Unused
// Numeric code
// This number specifies which zone is affected by the message. For example,
// if this message is for CHECK ZONE 22 then the numeric code would be 022.
// Most of the time this is zero-padded base10, but there are rare occurrences
// where this may be base16, such as ECP bus failures.
Zone string
// Raw data
// This is the binary data associated with the message. It includes all of the
// bit field entries that were separated out for you in the first field, as
// well as the rest of the message for debugging and exploratory purposes.
// There is one important piece of data included only in this field: the
// keypad address mask. The four bytes starting at position 2 (zero-indexed)
// indicate which keypads this message is intended for.
RawData string
// Alphanumeric Keypad Message
// This section is the data that would be displayed on your keypad's screen.
KeypadMessage string
}
// AlarmDecoder allows for interacting with an AlarmDecoder device over serial.
type AlarmDecoder struct {
rw io.ReadWriter
scanner *bufio.Scanner
}
// New returns a new AlarmDecoder. Context can be cancelled to terminate the
// background goroutine.
func New(rw io.ReadWriter) *AlarmDecoder {
return &AlarmDecoder{
rw: rw,
scanner: bufio.NewScanner(rw),
}
}
// Read returns a single message from the stream.
func (ad *AlarmDecoder) Read() (Message, error) {
hasMsg := ad.scanner.Scan()
if err := ad.scanner.Err(); err != nil {
return Message{}, err
}
if hasMsg {
return ParseMessage(ad.scanner.Text())
}
return Message{}, io.EOF
}