|
@@ -1,3 +1,7 @@ |
|
|
|
|
|
""" |
|
|
|
|
|
Provides the full AD2USB class and factory. |
|
|
|
|
|
""" |
|
|
|
|
|
|
|
|
import time |
|
|
import time |
|
|
import threading |
|
|
import threading |
|
|
from .event import event |
|
|
from .event import event |
|
@@ -5,6 +9,10 @@ from . import devices |
|
|
from . import util |
|
|
from . import util |
|
|
|
|
|
|
|
|
class Overseer(object): |
|
|
class Overseer(object): |
|
|
|
|
|
""" |
|
|
|
|
|
Factory for creation of AD2USB devices as well as provide4s attach/detach events." |
|
|
|
|
|
""" |
|
|
|
|
|
|
|
|
on_attached = event.Event('Called when an AD2USB device has been detected.') |
|
|
on_attached = event.Event('Called when an AD2USB device has been detected.') |
|
|
on_detached = event.Event('Called when an AD2USB device has been removed.') |
|
|
on_detached = event.Event('Called when an AD2USB device has been removed.') |
|
|
|
|
|
|
|
@@ -12,16 +20,25 @@ class Overseer(object): |
|
|
|
|
|
|
|
|
@classmethod |
|
|
@classmethod |
|
|
def find_all(cls): |
|
|
def find_all(cls): |
|
|
|
|
|
""" |
|
|
|
|
|
Returns all AD2USB devices located on the system. |
|
|
|
|
|
""" |
|
|
cls.__devices = devices.USBDevice.find_all() |
|
|
cls.__devices = devices.USBDevice.find_all() |
|
|
|
|
|
|
|
|
return cls.__devices |
|
|
return cls.__devices |
|
|
|
|
|
|
|
|
@classmethod |
|
|
@classmethod |
|
|
def devices(cls): |
|
|
def devices(cls): |
|
|
|
|
|
""" |
|
|
|
|
|
Returns a cached list of AD2USB devices located on the system. |
|
|
|
|
|
""" |
|
|
return cls.__devices |
|
|
return cls.__devices |
|
|
|
|
|
|
|
|
@classmethod |
|
|
@classmethod |
|
|
def create(cls, device=None): |
|
|
def create(cls, device=None): |
|
|
|
|
|
""" |
|
|
|
|
|
Factory method that returns the requested AD2USB device, or the first device. |
|
|
|
|
|
""" |
|
|
cls.find_all() |
|
|
cls.find_all() |
|
|
|
|
|
|
|
|
if len(cls.__devices) == 0: |
|
|
if len(cls.__devices) == 0: |
|
@@ -36,6 +53,9 @@ class Overseer(object): |
|
|
return AD2USB(device) |
|
|
return AD2USB(device) |
|
|
|
|
|
|
|
|
def __init__(self, attached_event=None, detached_event=None): |
|
|
def __init__(self, attached_event=None, detached_event=None): |
|
|
|
|
|
""" |
|
|
|
|
|
Constructor |
|
|
|
|
|
""" |
|
|
self._detect_thread = Overseer.DetectThread(self) |
|
|
self._detect_thread = Overseer.DetectThread(self) |
|
|
|
|
|
|
|
|
if attached_event: |
|
|
if attached_event: |
|
@@ -49,33 +69,60 @@ class Overseer(object): |
|
|
self.start() |
|
|
self.start() |
|
|
|
|
|
|
|
|
def __del__(self): |
|
|
def __del__(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Destructor |
|
|
|
|
|
""" |
|
|
pass |
|
|
pass |
|
|
|
|
|
|
|
|
def close(self): |
|
|
def close(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Clean up and shut down. |
|
|
|
|
|
""" |
|
|
self.stop() |
|
|
self.stop() |
|
|
|
|
|
|
|
|
def start(self): |
|
|
def start(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Starts the detection thread, if not already running. |
|
|
|
|
|
""" |
|
|
if not self._detect_thread.is_alive(): |
|
|
if not self._detect_thread.is_alive(): |
|
|
self._detect_thread.start() |
|
|
self._detect_thread.start() |
|
|
|
|
|
|
|
|
def stop(self): |
|
|
def stop(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Stops the detection thread. |
|
|
|
|
|
""" |
|
|
self._detect_thread.stop() |
|
|
self._detect_thread.stop() |
|
|
|
|
|
|
|
|
def get_device(self, device=None): |
|
|
def get_device(self, device=None): |
|
|
|
|
|
""" |
|
|
|
|
|
Factory method that returns the requested AD2USB device, or the first device. |
|
|
|
|
|
""" |
|
|
return Overseer.create(device) |
|
|
return Overseer.create(device) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class DetectThread(threading.Thread): |
|
|
class DetectThread(threading.Thread): |
|
|
|
|
|
""" |
|
|
|
|
|
Thread that handles detection of added/removed devices. |
|
|
|
|
|
""" |
|
|
def __init__(self, overseer): |
|
|
def __init__(self, overseer): |
|
|
|
|
|
""" |
|
|
|
|
|
Constructor |
|
|
|
|
|
""" |
|
|
threading.Thread.__init__(self) |
|
|
threading.Thread.__init__(self) |
|
|
|
|
|
|
|
|
self._overseer = overseer |
|
|
self._overseer = overseer |
|
|
self._running = False |
|
|
self._running = False |
|
|
|
|
|
|
|
|
def stop(self): |
|
|
def stop(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Stops the thread. |
|
|
|
|
|
""" |
|
|
self._running = False |
|
|
self._running = False |
|
|
|
|
|
|
|
|
def run(self): |
|
|
def run(self): |
|
|
|
|
|
""" |
|
|
|
|
|
The actual detection process. |
|
|
|
|
|
""" |
|
|
self._running = True |
|
|
self._running = True |
|
|
|
|
|
|
|
|
last_devices = set() |
|
|
last_devices = set() |
|
@@ -101,6 +148,10 @@ class Overseer(object): |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class AD2USB(object): |
|
|
class AD2USB(object): |
|
|
|
|
|
""" |
|
|
|
|
|
High-level wrapper around AD2USB/AD2SERIAL devices. |
|
|
|
|
|
""" |
|
|
|
|
|
|
|
|
on_open = event.Event('Called when the device has been opened') |
|
|
on_open = event.Event('Called when the device has been opened') |
|
|
on_close = event.Event('Called when the device has been closed') |
|
|
on_close = event.Event('Called when the device has been closed') |
|
|
on_read = event.Event('Called when a line has been read from the device') |
|
|
on_read = event.Event('Called when a line has been read from the device') |
|
@@ -109,26 +160,44 @@ class AD2USB(object): |
|
|
on_message = event.Event('Called when a message has been received from the device.') |
|
|
on_message = event.Event('Called when a message has been received from the device.') |
|
|
|
|
|
|
|
|
def __init__(self, device): |
|
|
def __init__(self, device): |
|
|
|
|
|
""" |
|
|
|
|
|
Constructor |
|
|
|
|
|
""" |
|
|
self._device = device |
|
|
self._device = device |
|
|
|
|
|
|
|
|
def __del__(self): |
|
|
def __del__(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Destructor |
|
|
|
|
|
""" |
|
|
pass |
|
|
pass |
|
|
|
|
|
|
|
|
def open(self, baudrate=None, interface=None, index=None): |
|
|
def open(self, baudrate=None, interface=None, index=None): |
|
|
|
|
|
""" |
|
|
|
|
|
Opens the device. |
|
|
|
|
|
""" |
|
|
self._wire_events() |
|
|
self._wire_events() |
|
|
self._device.open(baudrate=baudrate, interface=interface, index=index) |
|
|
self._device.open(baudrate=baudrate, interface=interface, index=index) |
|
|
|
|
|
|
|
|
def close(self): |
|
|
def close(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Closes the device. |
|
|
|
|
|
""" |
|
|
self._device.close() |
|
|
self._device.close() |
|
|
self._device = None |
|
|
self._device = None |
|
|
|
|
|
|
|
|
def _wire_events(self): |
|
|
def _wire_events(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Wires up the internal device events. |
|
|
|
|
|
""" |
|
|
self._device.on_open += self._on_open |
|
|
self._device.on_open += self._on_open |
|
|
self._device.on_close += self._on_close |
|
|
self._device.on_close += self._on_close |
|
|
self._device.on_read += self._on_read |
|
|
self._device.on_read += self._on_read |
|
|
self._device.on_write += self._on_write |
|
|
self._device.on_write += self._on_write |
|
|
|
|
|
|
|
|
def _handle_message(self, data): |
|
|
def _handle_message(self, data): |
|
|
|
|
|
""" |
|
|
|
|
|
Parses messages from the panel. |
|
|
|
|
|
""" |
|
|
if data[0] == '!': |
|
|
if data[0] == '!': |
|
|
return None |
|
|
return None |
|
|
|
|
|
|
|
@@ -138,12 +207,21 @@ class AD2USB(object): |
|
|
print msg.ignore_packet |
|
|
print msg.ignore_packet |
|
|
|
|
|
|
|
|
def _on_open(self, sender, args): |
|
|
def _on_open(self, sender, args): |
|
|
|
|
|
""" |
|
|
|
|
|
Internal handler for opening the device. |
|
|
|
|
|
""" |
|
|
self.on_open(args) |
|
|
self.on_open(args) |
|
|
|
|
|
|
|
|
def _on_close(self, sender, args): |
|
|
def _on_close(self, sender, args): |
|
|
|
|
|
""" |
|
|
|
|
|
Internal handler for closing the device. |
|
|
|
|
|
""" |
|
|
self.on_close(args) |
|
|
self.on_close(args) |
|
|
|
|
|
|
|
|
def _on_read(self, sender, args): |
|
|
def _on_read(self, sender, args): |
|
|
|
|
|
""" |
|
|
|
|
|
Internal handler for reading from the device. |
|
|
|
|
|
""" |
|
|
msg = self._handle_message(args) |
|
|
msg = self._handle_message(args) |
|
|
if msg: |
|
|
if msg: |
|
|
self.on_message(msg) |
|
|
self.on_message(msg) |
|
@@ -151,10 +229,20 @@ class AD2USB(object): |
|
|
self.on_read(args) |
|
|
self.on_read(args) |
|
|
|
|
|
|
|
|
def _on_write(self, sender, args): |
|
|
def _on_write(self, sender, args): |
|
|
|
|
|
""" |
|
|
|
|
|
Internal handler for writing to the device. |
|
|
|
|
|
""" |
|
|
self.on_write(args) |
|
|
self.on_write(args) |
|
|
|
|
|
|
|
|
class Message(object): |
|
|
class Message(object): |
|
|
|
|
|
""" |
|
|
|
|
|
Represents a message from the alarm panel. |
|
|
|
|
|
""" |
|
|
|
|
|
|
|
|
def __init__(self): |
|
|
def __init__(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Constructor |
|
|
|
|
|
""" |
|
|
self._ignore_packet = False |
|
|
self._ignore_packet = False |
|
|
self._ready = False |
|
|
self._ready = False |
|
|
self._armed_away = False |
|
|
self._armed_away = False |
|
@@ -170,132 +258,228 @@ class Message(object): |
|
|
self._numeric = "" |
|
|
self._numeric = "" |
|
|
self._text = "" |
|
|
self._text = "" |
|
|
self._cursor = -1 |
|
|
self._cursor = -1 |
|
|
self._raw_text = "" |
|
|
|
|
|
|
|
|
self._raw = "" |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def ignore_packet(self): |
|
|
def ignore_packet(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Indicates whether or not this message should be ignored. |
|
|
|
|
|
""" |
|
|
return self._ignore_packet |
|
|
return self._ignore_packet |
|
|
|
|
|
|
|
|
@ignore_packet.setter |
|
|
@ignore_packet.setter |
|
|
def ignore_packet(self, value): |
|
|
def ignore_packet(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the value indicating whether or not this packet should be ignored. |
|
|
|
|
|
""" |
|
|
self._ignore_packet = value |
|
|
self._ignore_packet = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def ready(self): |
|
|
def ready(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Indicates whether or not the panel is ready. |
|
|
|
|
|
""" |
|
|
return self._ready |
|
|
return self._ready |
|
|
|
|
|
|
|
|
@ready.setter |
|
|
@ready.setter |
|
|
def ready(self, value): |
|
|
def ready(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the value indicating whether or not the panel is ready. |
|
|
|
|
|
""" |
|
|
self._ready = value |
|
|
self._ready = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def armed_away(self): |
|
|
def armed_away(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Indicates whether or not the panel is armed in away mode. |
|
|
|
|
|
""" |
|
|
return self._armed_away |
|
|
return self._armed_away |
|
|
|
|
|
|
|
|
@armed_away.setter |
|
|
@armed_away.setter |
|
|
def armed_away(self, value): |
|
|
def armed_away(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the value indicating whether or not the panel is armed in away mode. |
|
|
|
|
|
""" |
|
|
self._armed_away = value |
|
|
self._armed_away = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def armed_home(self): |
|
|
def armed_home(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Indicates whether or not the panel is armed in home/stay mode. |
|
|
|
|
|
""" |
|
|
return self._armed_home |
|
|
return self._armed_home |
|
|
|
|
|
|
|
|
@armed_home.setter |
|
|
@armed_home.setter |
|
|
def armed_home(self, value): |
|
|
def armed_home(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the value indicating whether or not the panel is armed in home/stay mode. |
|
|
|
|
|
""" |
|
|
self._armed_home = value |
|
|
self._armed_home = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def backlight(self): |
|
|
def backlight(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Indicates whether or not the panel backlight is on. |
|
|
|
|
|
""" |
|
|
return self._backlight |
|
|
return self._backlight |
|
|
|
|
|
|
|
|
@backlight.setter |
|
|
@backlight.setter |
|
|
def backlight(self, value): |
|
|
def backlight(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the value indicating whether or not the panel backlight is on. |
|
|
|
|
|
""" |
|
|
self._backlight = value |
|
|
self._backlight = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def programming_mode(self): |
|
|
def programming_mode(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Indicates whether or not the panel is in programming mode. |
|
|
|
|
|
""" |
|
|
return self._programming_mode |
|
|
return self._programming_mode |
|
|
|
|
|
|
|
|
@programming_mode.setter |
|
|
@programming_mode.setter |
|
|
def programming_mode(self, value): |
|
|
def programming_mode(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the value indicating whether or not the panel is in programming mode. |
|
|
|
|
|
""" |
|
|
self._programming_mode = value |
|
|
self._programming_mode = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def beeps(self): |
|
|
def beeps(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Returns the number of beeps associated with this message. |
|
|
|
|
|
""" |
|
|
return self._beeps |
|
|
return self._beeps |
|
|
|
|
|
|
|
|
@beeps.setter |
|
|
@beeps.setter |
|
|
def beeps(self, value): |
|
|
def beeps(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the number of beeps associated with this message. |
|
|
|
|
|
""" |
|
|
self._beeps = value |
|
|
self._beeps = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def bypass(self): |
|
|
def bypass(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Indicates whether or not zones have been bypassed. |
|
|
|
|
|
""" |
|
|
return self._bypass |
|
|
return self._bypass |
|
|
|
|
|
|
|
|
@bypass.setter |
|
|
@bypass.setter |
|
|
def bypass(self, value): |
|
|
def bypass(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the value indicating whether or not zones have been bypassed. |
|
|
|
|
|
""" |
|
|
self._bypass = value |
|
|
self._bypass = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def ac(self): |
|
|
def ac(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Indicates whether or not the system is on AC power. |
|
|
|
|
|
""" |
|
|
return self._ac |
|
|
return self._ac |
|
|
|
|
|
|
|
|
@ac.setter |
|
|
@ac.setter |
|
|
def ac(self, value): |
|
|
def ac(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the value indicating whether or not the system is on AC power. |
|
|
|
|
|
""" |
|
|
self._ac = value |
|
|
self._ac = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def chime_mode(self): |
|
|
def chime_mode(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Indicates whether or not panel chimes are enabled. |
|
|
|
|
|
""" |
|
|
return self._chime_mode |
|
|
return self._chime_mode |
|
|
|
|
|
|
|
|
@chime_mode.setter |
|
|
@chime_mode.setter |
|
|
def chime_mode(self, value): |
|
|
def chime_mode(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the value indicating whether or not the panel chimes are enabled. |
|
|
|
|
|
""" |
|
|
self._chime_mode = value |
|
|
self._chime_mode = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def alarm_event_occurred(self): |
|
|
def alarm_event_occurred(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Indicates whether or not an alarm event has occurred. |
|
|
|
|
|
""" |
|
|
return self._alarm_event_occurred |
|
|
return self._alarm_event_occurred |
|
|
|
|
|
|
|
|
@alarm_event_occurred.setter |
|
|
@alarm_event_occurred.setter |
|
|
def alarm_event_occurred(self, value): |
|
|
def alarm_event_occurred(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the value indicating whether or not an alarm event has occurred. |
|
|
|
|
|
""" |
|
|
self._alarm_event_occurred = value |
|
|
self._alarm_event_occurred = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def alarm_bell(self): |
|
|
def alarm_bell(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Indicates whether or not an alarm is currently sounding. |
|
|
|
|
|
""" |
|
|
return self._alarm_bell |
|
|
return self._alarm_bell |
|
|
|
|
|
|
|
|
@alarm_bell.setter |
|
|
@alarm_bell.setter |
|
|
def alarm_bell(self, value): |
|
|
def alarm_bell(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the value indicating whether or not an alarm is currently sounding. |
|
|
|
|
|
""" |
|
|
self._alarm_bell = value |
|
|
self._alarm_bell = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def numeric(self): |
|
|
def numeric(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Numeric indicator of associated with message. For example: If zone #3 is faulted, this value is 003. |
|
|
|
|
|
""" |
|
|
return self._numeric |
|
|
return self._numeric |
|
|
|
|
|
|
|
|
@numeric.setter |
|
|
@numeric.setter |
|
|
def numeric(self, value): |
|
|
def numeric(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the numeric indicator associated with this message. |
|
|
|
|
|
""" |
|
|
self._numeric = value |
|
|
self._numeric = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def text(self): |
|
|
def text(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Alphanumeric text associated with this message. |
|
|
|
|
|
""" |
|
|
return self._text |
|
|
return self._text |
|
|
|
|
|
|
|
|
@text.setter |
|
|
@text.setter |
|
|
def text(self, value): |
|
|
def text(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the alphanumeric text associated with this message. |
|
|
|
|
|
""" |
|
|
self._text = value |
|
|
self._text = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def cursor(self): |
|
|
def cursor(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Indicates which text position has the cursor underneath it. |
|
|
|
|
|
""" |
|
|
return self._cursor |
|
|
return self._cursor |
|
|
|
|
|
|
|
|
@cursor.setter |
|
|
@cursor.setter |
|
|
def cursor(self, value): |
|
|
def cursor(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the value indicating which text position has the cursor underneath it. |
|
|
|
|
|
""" |
|
|
self._cursor = value |
|
|
self._cursor = value |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
|
def raw_text(self): |
|
|
|
|
|
return self._raw_text |
|
|
|
|
|
|
|
|
def raw(self): |
|
|
|
|
|
""" |
|
|
|
|
|
Raw representation of the message data from the panel. |
|
|
|
|
|
""" |
|
|
|
|
|
return self._raw |
|
|
|
|
|
|
|
|
@raw_text.setter |
|
|
@raw_text.setter |
|
|
def raw_text(self, value): |
|
|
|
|
|
self._raw_text = value |
|
|
|
|
|
|
|
|
def raw(self, value): |
|
|
|
|
|
""" |
|
|
|
|
|
Sets the raw representation of the message data from the panel. |
|
|
|
|
|
""" |
|
|
|
|
|
self._raw = value |