Source code for BAC0.core.devices.Device

#!/usr/bin/python
# -*- coding: utf-8 -*-
#
# Copyright (C) 2015-2017 by Christian Tremblay, P.Eng <christian.tremblay@servisysDeviceObject.com>
# Licensed under LGPLv3, see file LICENSE in this source tree.
#
"""
Device.py - describe a BACnet Device

"""
import os.path

# --- standard Python modules ---
from collections import namedtuple

try:
    import pandas as pd

    _PANDAS = True
except ImportError:
    _PANDAS = False

try:
    from xlwings import Chart, Range, Sheet, Workbook  # noqa E401

    _XLWINGS = True
except ImportError:
    _XLWINGS = False


# --- this application's modules ---
from bacpypes.basetypes import ServicesSupported

# from ...bokeh.BokehRenderer import BokehPlot
from ...db.sql import SQLMixin
from ...tasks.DoOnce import DoOnce
from ..io.IOExceptions import (
    BadDeviceDefinition,
    DeviceNotConnected,
    NoResponseFromController,
    RemovedPointException,
    SegmentationNotSupported,
    WritePropertyException,
    WrongParameter,
)
from ..utils.notes import note_and_log
from .mixins.read_mixin import ReadProperty, ReadPropertyMultiple
from .Points import BooleanPoint, EnumPoint, NumericPoint, OfflinePoint
from .Virtuals import VirtualPoint

# ------------------------------------------------------------------------------


[docs]class DeviceProperties(object): """ This serves as a container for device properties """ def __init__(self): self.name = "Unknown" self.address = None self.device_id = None self.network = None self.pollDelay = None self.objects_list = None self.pss = ServicesSupported() self.multistates = None self.db_name = None self.segmentation_supported = True self.history_size = None self.save_resampling = "1s" self.clear_history_on_save = None self.bacnet_properties = {} self.auto_save = None self.fast_polling = False self.vendor_id = 0 self.ping_failures = 0 def __repr__(self): return "{}".format(self.asdict) @property def asdict(self): return self.__dict__
[docs]@note_and_log class Device(SQLMixin): """ Represent a BACnet device. Once defined, it allows use of read, write, sim, release functions to communicate with the device on the network. :param address: address of the device (ex. '2:5') :param device_id: bacnet device ID (boid) :param network: defined by BAC0.connect() :param poll: (int) if > 0, will poll every points each x seconds. :from_backup: sqlite backup file :segmentation_supported: (boolean) When segmentation is not supported, BAC0 will not use read property multiple to poll the device. :object_list: (list) Use can provide a custom object_list to use for the the creation of the device. the object list must be built using the same pattern returned by bacpypes when polling the objectList property example :: my_obj_list = [('file', 1), ('analogInput', 2), ('analogInput', 3), ('analogInput', 5), ('analogInput', 4), ('analogInput', 0), ('analogInput', 1)] :auto_save: (False or int) If False or 0, auto_save is disabled. To Activate, pass an integer representing the number of polls before auto_save is called. Will write the histories to SQLite db locally. :clear_history_on_save: (boolean) Will clear device history :type address: (str) :type device_id: int :type network: BAC0.scripts.ReadWriteScript.ReadWriteScript """ def __init__( self, address=None, device_id=None, network=None, *, poll=10, from_backup=None, # filename of backup segmentation_supported=True, object_list=None, auto_save=False, save_resampling="1s", clear_history_on_save=False, history_size=None, reconnect_on_failure=True ): self.properties = DeviceProperties() self.properties.address = address self.properties.device_id = device_id self.properties.network = network self.properties.pollDelay = poll self.properties.fast_polling = True if poll < 10 else False self.properties.name = "" self.properties.vendor_id = 0 self.properties.objects_list = [] self.properties.pss = ServicesSupported() self.properties.multistates = {} self.properties.auto_save = auto_save self.properties.save_resampling = save_resampling self.properties.clear_history_on_save = clear_history_on_save self.properties.history_size = history_size self._reconnect_on_failure = reconnect_on_failure self.segmentation_supported = segmentation_supported self.custom_object_list = object_list # self.db = None # Todo : find a way to normalize the name of the db self.properties.db_name = None self.points = [] self._list_of_trendlogs = {} self._polling_task = namedtuple("_polling_task", ["task", "running"]) self._polling_task.task = None self._polling_task.running = False self._find_overrides_progress = 0.0 self._find_overrides_running = False self._release_overrides_progress = 0.0 self._release_overrides_running = False self.note("Controller initialized") if from_backup: filename = from_backup db_name = filename.split(".")[0] self.properties.network = None if os.path.isfile(filename): self.properties.db_name = db_name self.new_state(DeviceDisconnected) else: raise FileNotFoundError("Can't find {} on drive".format(filename)) else: if ( self.properties.network and self.properties.address and self.properties.device_id is not None ): self.new_state(DeviceDisconnected) else: raise BadDeviceDefinition( "Please provide address, device id and network or specify from_backup argument" )
[docs] def new_state(self, newstate): """ Base of the state machine mechanism. Used to make transitions between device states. Take care to call the state init function. """ self._log.info( "Changing device state to {}".format(str(newstate).split(".")[-1]) ) self.__class__ = newstate self._init_state()
def _init_state(self): """ Execute additional code upon state modification """ raise NotImplementedError()
[docs] def connect(self): """ Connect the device to the network """ raise NotImplementedError()
[docs] def disconnect(self): raise NotImplementedError()
[docs] def initialize_device_from_db(self): raise NotImplementedError()
[docs] def df(self, list_of_points, force_read=True): """ Build a pandas DataFrame from a list of points. DataFrames are used to present and analyze data. :param list_of_points: a list of point names as str :returns: pd.DataFrame """ raise NotImplementedError()
@property def simulated_points(self): """ iterate over simulated points :returns: points if simulated (out_of_service == True) :rtype: BAC0.core.devices.Points.Point """ for each in self.points: if each.properties.simulated[0]: yield each def _buildPointList(self): """ Read all points from a device into a (Pandas) dataframe (Pandas). Items are accessible by point name. """ raise NotImplementedError() def __getitem__(self, point_name): """ Get a point from its name. If a list is passed - a dataframe is returned. :param point_name: (str) name of the point or list of point_names :type point_name: str :returns: (Point) the point (can be Numeric, Boolean or Enum) or pd.DataFrame """ raise NotImplementedError() def __iter__(self): """ When iterating a device, iterate points of it. """ raise NotImplementedError() def __contains__(self, value): "When using in..." raise NotImplementedError() @property def points_name(self): """ When iterating a device, iterate points of it. """ raise NotImplementedError()
[docs] def to_excel(self): """ Using xlwings, make a dataframe of all histories and save it """ raise NotImplementedError()
def __setitem__(self, point_name, value): """ Write, sim or ovr value :param point_name: Name of the point to set :param value: value to write to the point :type point_name: str :type value: float """ raise NotImplementedError() def __len__(self): """ Will return number of points available """ raise NotImplementedError() def _parseArgs(self, arg): """ Given a string, interpret the last word as the value, everything else is considered to be the point name. """ args = arg.split() pointName = " ".join(args[:-1]) value = args[-1] return (pointName, value)
[docs] def clear_histories(self): for point in self.points: point.clear_history()
[docs] def update_history_size(self, size=None): for point in self.points: point.properties.history_size = size
@property def analog_units(self): raise NotImplementedError() @property def temperatures(self): raise NotImplementedError() @property def percent(self): raise NotImplementedError() @property def multi_states(self): raise NotImplementedError() @property def binary_states(self): raise NotImplementedError() def _findPoint(self, name, force_read=True): """ Helper that retrieve point based on its name. :param name: (str) name of the point :param force_read: (bool) read value of the point each time the function is called. :returns: Point object :rtype: BAC0.core.devices.Point.Point (NumericPoint, EnumPoint or BooleanPoint) """ raise NotImplementedError()
[docs] def find_point(self, objectType, objectAddress): """ Find point based on type and address """ for point in self.points: if ( point.properties.type == objectType and float(point.properties.address) == objectAddress ): return point raise ValueError( "{} {} doesn't exist in controller".format(objectType, objectAddress) )
[docs] def find_overrides(self, force=False): if self._find_overrides_running and not force: self._log.warning( "Already running ({:.1%})... please wait.".format( self._find_overrides_progress ) ) return lst = [] self._find_overrides_progress = 0.0 self._find_overrides_running = True total = len(self.points) def _find_overrides(): self._log.warning( "Overrides are being checked, wait for completion message." ) for idx, point in enumerate(self.points): if point.is_overridden: lst.append(point) self._find_overrides_progress = idx / total self._log.warning( "Override check ready, results available in device.properties.points_overridden" ) self.properties.points_overridden = lst self._find_overrides_running = False self._find_overrides_progress = 1.0 self.do(_find_overrides)
[docs] def find_overrides_progress(self) -> float: return self._find_overrides_progress
[docs] def release_all_overrides(self, force=False): if self._release_overrides_running and not force: self._log.warning( "Already running ({:.1%})... please wait.".format( self._release_overrides_progress ) ) return self._release_overrides_running = True self._release_overrides_progress = 0.0 def _release_all_overrides(): self.find_overrides() while self._find_overrides_running: self._release_overrides_progress = self._find_overrides_progress * 0.5 if self.properties.points_overridden: total = len(self.properties.points_overridden) self._log.info("=================================") self._log.info("Overrides found... releasing them") self._log.info("=================================") for idx, point in enumerate(self.properties.points_overridden): self._log.info("Releasing {}".format(point)) point.release_ovr() self._release_overrides_progress = (idx / total) / 2 + 0.5 else: self._log.info("No override found") self._release_overrides_running = False self._release_overrides_progress = 1 self.do(_release_all_overrides)
[docs] def do(self, func): DoOnce(func).start()
def __repr__(self): return "{} / Undefined".format(self.properties.name)
# @fix_docs
[docs]class DeviceConnected(Device): """ Find a device on the BACnet network. Set its state to 'connected'. Once connected, all subsequent commands use this BACnet connection. """ def _init_state(self): self._buildPointList() self.properties.network.register_device(self)
[docs] def disconnect(self, save_on_disconnect=True, unregister=True): self._log.info("Wait while stopping polling") self.poll(command="stop") if unregister: self.properties.network.unregister_device(self) self.properties.network = None if save_on_disconnect: self.save() if self.properties.db_name: self.new_state(DeviceFromDB) else: self.new_state(DeviceDisconnected)
[docs] def connect(self, *, db=None): """ A connected device can be switched to 'database mode' where the device will not use the BACnet network but instead obtain its contents from a previously stored database. """ if db: self.poll(command="stop") self.properties.db_name = db.split(".")[0] self.new_state(DeviceFromDB) else: self._log.warning( "Already connected, provide db arg if you want to connect to db" )
[docs] def df(self, list_of_points, force_read=True): """ When connected, calling DF should force a reading on the network. """ his = [] for point in list_of_points: try: his.append(self._findPoint(point, force_read=force_read).history) except ValueError as ve: self._log.error("{}".format(ve)) continue if not _PANDAS: return dict(zip(list_of_points, his)) return pd.DataFrame(dict(zip(list_of_points, his)))
def _buildPointList(self): """ Upon connection to build the device point list and properties. """ try: self.properties.pss.value = self.properties.network.read( "{} device {} protocolServicesSupported".format( self.properties.address, self.properties.device_id ) ) except NoResponseFromController as error: self._log.error("Controller not found, aborting. ({})".format(error)) return ("Not Found", "", [], []) except SegmentationNotSupported: self._log.warning("Segmentation not supported") self.segmentation_supported = False self.new_state(DeviceDisconnected) self.properties.name = self.properties.network.read( "{} device {} objectName".format( self.properties.address, self.properties.device_id ) ) self.properties.vendor_id = self.properties.network.read( "{} device {} vendorIdentifier".format( self.properties.address, self.properties.device_id ) ) self._log.info( "Device {}:[{}] found... building points list".format( self.properties.device_id, self.properties.name ) ) try: ( self.properties.objects_list, self.points, self._list_of_trendlogs, ) = self._discoverPoints(self.custom_object_list) if self.properties.pollDelay is not None and self.properties.pollDelay > 0: self.poll(delay=self.properties.pollDelay) self.update_history_size(size=self.properties.history_size) # self.clear_histories() except NoResponseFromController: self._log.error("Cannot retrieve object list, disconnecting...") self.segmentation_supported = False self.new_state(DeviceDisconnected) except IndexError: if self._reconnect_on_failure: self._log.error("Device creation failed... re-connecting") self.new_state(DeviceDisconnected) else: self._log.error("Device creation failed... disconnecting") def __getitem__(self, point_name): """ Allows the syntax: device['point_name'] or device[list_of_points] If calling a list, last value will be used (won't read on the network) for performance reasons. If calling a simple point, point will be read via BACnet. """ try: if isinstance(point_name, list): return self.df(point_name, force_read=False) elif isinstance(point_name, tuple): _type, _address = point_name for point in self.points: if point.properties.type == _type and str( point.properties.address ) == str(_address): return point else: try: return self._findPoint(point_name, force_read=False) except ValueError: try: return self._findTrend(point_name) except ValueError: try: if "@prop_" in point_name: point_name = point_name.split("prop_")[1] return self.read_property( ("device", self.properties.device_id, point_name) ) else: raise ValueError() except ValueError: raise ValueError() except ValueError as ve: self._log.error("{}".format(ve)) def __iter__(self): yield from self.points def __contains__(self, value): """ Allows the syntax: if "point_name" in device: """ return value in self.points_name @property def pollable_points_name(self): for each in self.points: if not isinstance(each, VirtualPoint): yield each.properties.name else: continue @property def points_name(self): for each in self.points: yield each.properties.name def __setitem__(self, point_name, value): """ Allows the syntax: device['point_name'] = value """ try: self._findPoint(point_name)._set(value) except WritePropertyException as ve: self._log.error("{}".format(ve)) def __len__(self): """ Length of a device = number of points """ return len(self.points) def _parseArgs(self, arg): args = arg.split() pointName = " ".join(args[:-1]) value = args[-1] return (pointName, value) @property def analog_units(self): """ Shortcut to retrieve all analog points units [Used by Bokeh trending feature] """ au = [] us = [] for each in self.points: if isinstance(each, NumericPoint): au.append(each.properties.name) us.append(each.properties.units_state) return dict(zip(au, us)) @property def temperatures(self): for each in self.analog_units.items(): if "deg" in each[1]: yield each @property def percent(self): for each in self.analog_units.items(): if "percent" in each[1]: yield each @property def multi_states(self): ms = [] us = [] for each in self.points: if isinstance(each, EnumPoint): ms.append(each.properties.name) us.append(each.properties.units_state) return dict(zip(ms, us)) @property def binary_states(self): bs = [] us = [] for each in self.points: if isinstance(each, BooleanPoint): bs.append(each.properties.name) us.append(each.properties.units_state) return dict(zip(bs, us)) def _findPoint(self, name, force_read=False): """ Used by getter and setter functions """ for point in self.points: if point.properties.name == name: if force_read: point.value return point raise ValueError("{} doesn't exist in controller".format(name)) def _trendlogs(self): for k, v in self._list_of_trendlogs.items(): name, trendlog = v yield trendlog @property def trendlogs_names(self): for each in self._trendlogs(): yield each.properties.object_name @property def trendlogs(self): return list(self._trendlogs()) def _findTrend(self, name): for trend in self._trendlogs(): if trend.properties.object_name == name: return trend raise ValueError("{} doesn't exist in controller".format(name))
[docs] def read_property(self, prop): # if instance == -1: # pass if isinstance(prop, tuple): _obj, _instance, _prop = prop elif isinstance(prop, str): _obj = "device" _instance = self.properties.device_id _prop = prop else: raise ValueError( "Please provide property using tuple with object, instance and property" ) try: request = "{} {} {} {}".format( self.properties.address, _obj, _instance, _prop ) val = self.properties.network.read( request, vendor_id=self.properties.vendor_id ) except KeyError as error: raise Exception("Unknown property : {}".format(error)) return val
[docs] def write_property(self, prop, value, priority=None): if prop == "description": self.update_description(value) else: if priority is not None: priority = "- {}".format(priority) if isinstance(prop, tuple): _obj, _instance, _prop = prop else: raise ValueError( "Please provide property using tuple with object, instance and property" ) try: request = "{} {} {} {} {} {}".format( self.properties.address, _obj, _instance, _prop, value, priority ) val = self.properties.network.write( request, vendor_id=self.properties.vendor_id ) except KeyError as error: raise Exception("Unknown property : {}".format(error)) return val
[docs] def update_bacnet_properties(self): """ Retrieve bacnet properties for this device """ try: res = self.properties.network.readMultiple( "{} device {} all".format( self.properties.address, str(self.properties.device_id) ), vendor_id=self.properties.vendor_id, show_property_name=True, ) for each in res: if not each: continue v, prop = each self.properties.bacnet_properties[prop] = v except Exception as e: raise Exception("Problem reading : {} | {}".format(self.properties.name, e))
def _bacnet_properties(self, update=False): if not self.properties.bacnet_properties or update: self.update_bacnet_properties() return self.properties.bacnet_properties @property def bacnet_properties(self): return self._bacnet_properties(update=True)
[docs] def update_description(self, value): self.properties.network.send_text_write_request( addr=self.properties.address, obj_type="device", obj_inst=int(self.device_id), value=value, prop_id="description", ) self.properties.description = self.read_property("description")
[docs] def ping(self): try: if self.read_property("objectName") == self.properties.name: self.properties.ping_failures = 0 return True else: self.properties.ping_failures += 1 return False except NoResponseFromController as e: self._log.error( "{} ({})| Ping failure ({}).".format( self.properties.name, self.properties.address, e ) ) self.properties.ping_failures += 1 return False
def __repr__(self): return "{} / Connected".format(self.properties.name)
# ------------------------------------------------------------------------------
[docs]class RPDeviceConnected(DeviceConnected, ReadProperty): """ [Device state] If device is connected but doesn't support ReadPropertyMultiple BAC0 will not poll such points automatically (since it would cause excessive network traffic). Instead manual polling must be used as needed via the poll() function. """ def __str__(self): return "connected [for ReadProperty]"
[docs]class RPMDeviceConnected(DeviceConnected, ReadPropertyMultiple): """ [Device state] If device is connected and supports ReadPropertyMultiple """ def __str__(self): return "connected [for ReadPropertyMultiple]"
# @fix_docs
[docs]class DeviceDisconnected(Device): """ [Device state] Initial state of a device. Disconnected from BACnet. """ def _init_state(self): self.connect()
[docs] def connect(self, *, db=None, network=None): """ Attempt to connect to device. If unable, attempt to connect to a controller database (so the user can use previously saved data). """ if network: self.properties.network = network if not self.properties.network: self._log.debug("No network...calling DeviceFromDB") if db: self.new_state(DeviceFromDB) self._log.info( 'You can reconnect to network using : "device.connect(network=bacnet)"' ) else: try: self.properties.network.read( "{} device {} objectName".format( self.properties.address, self.properties.device_id ) ) segmentation = self.properties.network.read( "{} device {} segmentationSupported".format( self.properties.address, self.properties.device_id ) ) if not self.segmentation_supported or segmentation not in ( "segmentedTransmit", "segmentedBoth", ): segmentation_supported = False self._log.debug("Segmentation not supported") else: segmentation_supported = True if segmentation_supported: self.new_state(RPMDeviceConnected) else: self.new_state(RPDeviceConnected) except SegmentationNotSupported: self.segmentation_supported = False self._log.warning( "Segmentation not supported.... expect slow responses." ) self.new_state(RPDeviceConnected) except (NoResponseFromController, AttributeError) as error: self._log.warning("Error connecting: %s", error) if self.properties.db_name: self.new_state(DeviceFromDB) else: self._log.warning( "Offline: provide database name to load stored data." ) self._log.warning("Ex. controller.connect(db = 'backup')")
[docs] def df(self, list_of_points, force_read=True): raise DeviceNotConnected("Must connect to BACnet or database")
@property def simulated_points(self): for each in self.points: if each.properties.simulated: yield each def _buildPointList(self): raise DeviceNotConnected("Must connect to BACnet or database") # This should be a "read" function and rpm defined in state rpm
[docs] def read_multiple( self, points_list, *, points_per_request=25, discover_request=(None, 6) ): raise DeviceNotConnected("Must connect to BACnet or database")
[docs] def poll(self, command="start", *, delay=10): raise DeviceNotConnected("Must connect to BACnet or database")
def __getitem__(self, point_name): raise DeviceNotConnected("Must connect to BACnet or database") def __iter__(self): raise DeviceNotConnected("Must connect to BACnet or database") def __contains__(self, value): raise DeviceNotConnected("Must connect to BACnet or database") @property def points_name(self): raise DeviceNotConnected("Must connect to BACnet or database")
[docs] def to_excel(self): raise DeviceNotConnected("Must connect to BACnet or database")
def __setitem__(self, point_name, value): raise DeviceNotConnected("Must connect to BACnet or database") def __len__(self): raise DeviceNotConnected("Must connect to BACnet or database") @property def analog_units(self): raise DeviceNotConnected("Must connect to BACnet or database") @property def temperatures(self): raise DeviceNotConnected("Must connect to BACnet or database") @property def percent(self): raise DeviceNotConnected("Must connect to BACnet or database") @property def multi_states(self): raise DeviceNotConnected("Must connect to bacnet or database") @property def binary_states(self): raise DeviceNotConnected("Must connect to BACnet or database") def _discoverPoints(self, custom_object_list=None): raise DeviceNotConnected("Must connect to BACnet or database") def _findPoint(self, name, force_read=True): raise DeviceNotConnected("Must connect to BACnet or database") def __repr__(self): return "{} / Disconnected".format(self.properties.name)
# ------------------------------------------------------------------------------ # @fix_docs
[docs]class DeviceFromDB(DeviceConnected): """ [Device state] Where requests for a point's present value returns the last valid value from the point's history. """ def _init_state(self): try: self.initialize_device_from_db() except ValueError as e: self._log.error("Problem with DB initialization : {}".format(e)) # self.new_state(DeviceDisconnected) raise
[docs] def connect(self, *, network=None, from_backup=None): """ In DBState, a device can be reconnected to BACnet using: device.connect(network=bacnet) (bacnet = BAC0.connect()) """ if network and from_backup: raise WrongParameter("Please provide network OR from_backup") elif network: self._log.debug("Network provided... trying to connect") self.properties.network = network try: name = self.properties.network.read( "{} device {} objectName".format( self.properties.address, self.properties.device_id ) ) segmentation = self.properties.network.read( "{} device {} segmentationSupported".format( self.properties.address, self.properties.device_id ) ) if not self.segmentation_supported or segmentation not in ( "segmentedTransmit", "segmentedBoth", ): segmentation_supported = False self._log.debug("Segmentation not supported") else: segmentation_supported = True if name: if segmentation_supported: self._log.debug("Segmentation supported, connecting...") self.new_state(RPMDeviceConnected) else: self._log.debug("Segmentation not supported, connecting...") self.new_state(RPDeviceConnected) # self.db.close() except NoResponseFromController: self._log.error("Unable to connect, keeping DB mode active") else: self._log.debug("Not connected, open DB") if from_backup: self.properties.db_name = from_backup.split(".")[0] self._init_state()
[docs] def initialize_device_from_db(self): self._log.info("Initializing DB") # Save important properties for reuse if self.properties.db_name: dbname = self.properties.db_name else: self._log.info("Missing argument DB") raise ValueError("Please provide db name using device.load_db('name')") # network = self.properties.network pss = self.properties.pss self._props = self.read_dev_prop(self.properties.db_name) self.points = [] for point in self.points_from_sql(self.properties.db_name): try: self.points.append(OfflinePoint(self, point)) except RemovedPointException: continue self.properties = DeviceProperties() self.properties.db_name = dbname self.properties.address = self._props["address"] self.properties.device_id = self._props["device_id"] self.properties.network = None self.properties.pollDelay = self._props["pollDelay"] self.properties.name = self._props["name"] self.properties.objects_list = self._props["objects_list"] self.properties.pss = pss self.properties.serving_chart = {} self.properties.charts = [] self.properties.multistates = self._props["multistates"] self.properties.auto_save = self._props["auto_save"] self.properties.save_resampling = self._props["save_resampling"] self.properties.clear_history_on_save = self._props["clear_history_on_save"] self.properties.default_history_size = self._props["history_size"] self._log.info("Device restored from db") self._log.info( 'You can reconnect to network using : "device.connect(network=bacnet)"' )
@property def simulated_points(self): raise DeviceNotConnected("Must connect to BACnet or database") def _buildPointList(self): raise DeviceNotConnected("Must connect to BACnet or database") # This should be a "read" function and rpm defined in state rpm
[docs] def read_multiple( self, points_list, *, points_per_request=25, discover_request=(None, 6) ): raise DeviceNotConnected("Must connect to BACnet or database")
[docs] def poll(self, command="start", *, delay=10): raise DeviceNotConnected("Must connect to BACnet or database")
def __contains__(self, value): raise DeviceNotConnected("Must connect to BACnet or database")
[docs] def to_excel(self): raise DeviceNotConnected("Must connect to BACnet or database")
def __setitem__(self, point_name, value): raise DeviceNotConnected("Must connect to BACnet or database") def _discoverPoints(self, custom_object_list=None): raise DeviceNotConnected("Must connect to BACnet or database") def __repr__(self): return "{} / Disconnected".format(self.properties.name)
# ------------------------------------------------------------------------------
[docs]class DeviceLoad(DeviceFromDB): def __init__(self, filename=None): if filename: Device.__init__(self, None, None, None, from_backup=filename) else: raise Exception("Please provide backup file as argument")