Source code for aaa_modules.zone_parser

import re

from core import osgi
from core.jsr223 import scope
from org.eclipse.smarthome.core.items import Metadata
from org.eclipse.smarthome.core.items import MetadataKey
from org.slf4j import Logger, LoggerFactory

from aaa_modules.layout_model.neighbor import Neighbor, NeighborType
from aaa_modules.layout_model.zone import Zone, Level
from aaa_modules.layout_model.astro_sensor import AstroSensor
from aaa_modules.layout_model.dimmer import Dimmer
from aaa_modules.layout_model.illuminance_sensor import IlluminanceSensor
from aaa_modules.layout_model.motion_sensor import MotionSensor
from aaa_modules.layout_model.switch import Fan, Light

META_DIMMING_SETTING = 'dimmable'

# A metadata item to indicate which light to turn off when the current light
# is switched on.
META_TURN_OFF_OTHER_LIGHT = 'turnOff'

# A meta data item to indicate that this light shouldn't be turned on when a
# motion event is triggered, if the other light is already on.
META_DISABLE_MOTION_TRIGGERING_IF_OTHER_LIGHT_IS_ON = 'disableMotionTriggeringIfOtherLightIsOn'

# Indicates that the switch must not be turned on when the associated 
# motion sensor is triggered.
TAG_DISABLE_TRIGGERING_FROM_MOTION_SENSOR = "disable-triggering-from-motion-sensor"

# The light level threshold; if it is below this value, turn on the light.
ILLUMINANCE_THRESHOLD_IN_LUX = 8

TIME_OF_DAY_ITEM_NAME = 'VT_Time_Of_Day'

ITEM_NAME_PATTERN = '([^_]+)_([^_]+)_(.+)' # level_location_deviceName

MetadataRegistry = osgi.get_service("org.eclipse.smarthome.core.items.MetadataRegistry")
logger = LoggerFactory.getLogger("org.eclipse.smarthome.model.script.Rules")

[docs]class ZoneParser: ''' Construct the zones from the existing items in OpenHab, using this naming convention: Floor_Location_ItemType For example, item "Switch FF_Foyer_LightSwitch ..." will create a zone named 'Foyer' at the first floor; and the zone contains a Light object. See :class:`.ZoneManager` and :class:`.Zone`. '''
[docs] @staticmethod def parse(items, itemRegistry): ''' :param scope.items items: :param scope.itemRegistry itemRegistry: :rtype: list(Zone) ''' zoneMap = {} # map from string zoneId to Zone # Each item is a list of 3 items: zone id, zone id, neighbor type. neighbors = [] for itemName in items.keys(): match = re.search(ITEM_NAME_PATTERN, itemName) if not match: continue levelString = match.group(1) location = match.group(2) deviceName = match.group(3) zoneId = ZoneParser._getZoneIdFromItemName(itemName) if zoneId in zoneMap: zone = zoneMap[zoneId] else: zone = Zone(location, [], ZoneParser._getZoneLevel(levelString)) openHabItem = itemRegistry.getItem(itemName) if 'LightSwitch' == deviceName: # open space relationship turnOffMeta = MetadataRegistry.get( MetadataKey(META_TURN_OFF_OTHER_LIGHT, itemName)) if None != turnOffMeta: neighborZoneId = ZoneParser._getZoneIdFromItemName( turnOffMeta.value) neighbor = [zoneId, neighborZoneId, NeighborType.OPEN_SPACE] neighbors.append(neighbor) # master-slave open space relationship masterSlaveMeta = MetadataRegistry.get( MetadataKey(META_DISABLE_MOTION_TRIGGERING_IF_OTHER_LIGHT_IS_ON, itemName)) if None != masterSlaveMeta: masterZoneId = ZoneParser._getZoneIdFromItemName(masterSlaveMeta.value) neighborForward = [masterZoneId, zoneId, NeighborType.OPEN_SPACE_SLAVE] neighbors.append(neighborForward) neighborReverse = [zoneId, masterZoneId, NeighborType.OPEN_SPACE_MASTER] neighbors.append(neighborReverse) timerItem = itemRegistry.getItem(itemName + '_Timer') disableMotionSensorTriggering = openHabItem.hasTag( TAG_DISABLE_TRIGGERING_FROM_MOTION_SENSOR) # dimmer setting meta = MetadataRegistry.get( MetadataKey(META_DIMMING_SETTING, itemName)) if None != meta: config = meta.configuration level = config['level'].intValue() timeRanges = config['timeRanges'] switch = Dimmer(openHabItem, timerItem, level, timeRanges, ILLUMINANCE_THRESHOLD_IN_LUX, disableMotionSensorTriggering) else: switch = Light(openHabItem, timerItem, ILLUMINANCE_THRESHOLD_IN_LUX, disableMotionSensorTriggering) zone = zone.addDevice(switch) elif 'FanSwitch' == deviceName: fan = Fan(openHabItem, itemRegistry.getItem(itemName + '_Timer')) zone = zone.addDevice(fan) elif 'LightSwitch_Illuminance' == deviceName: illuminanceSensor = IlluminanceSensor(openHabItem) zone = zone.addDevice(illuminanceSensor) elif deviceName.endswith('MotionSensor'): motionSensor = MotionSensor(openHabItem) zone = zone.addDevice(motionSensor) if len(zone.getDevices()) > 0: zoneMap[zoneId] = zone # end looping items astroSensor = AstroSensor(itemRegistry.getItem(TIME_OF_DAY_ITEM_NAME)) for z in zoneMap.values(): if len(z.getDevicesByType(Light)) > 0 or \ len(z.getDevicesByType(Dimmer)) > 0: z = z.addDevice(astroSensor) zoneMap[z.getId()] = z for neighborInfo in neighbors: zone = zoneMap[neighborInfo[0]] zone = zone.addNeighbor(Neighbor(neighborInfo[1], neighborInfo[2])) zoneMap[neighborInfo[0]] = zone return [ZoneParser._normalizeNeighbors(z) for z in zoneMap.values()]
@staticmethod def _normalizeNeighbors(zone): ''' If a zone has the same neighbor with more than one OPEN_SPACE type, remove the generic one NeighborType.OPEN_SPACE :rtype: Zone new object ''' zoneIdToType = {} for neighbor in zone.getNeighbors(): zoneId = neighbor.getZoneId() if zoneId in zoneIdToType: types = zoneIdToType[zoneId] else: types = [] zoneIdToType[zoneId] = types types.append(neighbor.getType()) for types in zoneIdToType.values(): if NeighborType.OPEN_SPACE_MASTER in types \ or NeighborType.OPEN_SPACE_SLAVE in types: if NeighborType.OPEN_SPACE in types: types.remove(NeighborType.OPEN_SPACE) zone = Zone(zone.getName(), zone.getDevices(), zone.getLevel(), []) for zoneId in zoneIdToType.keys(): for type in zoneIdToType[zoneId]: zone = zone.addNeighbor(Neighbor(zoneId, type)) return zone @staticmethod def _getZoneIdFromItemName(itemName): ''' :rtype: str ''' match = re.search(ITEM_NAME_PATTERN, itemName) if not match: raise ValueError('Invalid item name pattern: ' + itemName) levelString = match.group(1) location = match.group(2) return str(ZoneParser._getZoneLevel(levelString)) + '_' + location @staticmethod def _getZoneLevel(levelString): ''' :rtype: Level ''' if 'BM' == levelString: return Level.BASEMENT elif 'FF' == levelString: return Level.FIRST_FLOOR elif 'SF' == levelString: return Level.SECOND_FLOOR elif 'TF' == levelString: return Level.THIRD_FLOOR else: return Level.BASEMENT
zones = ZoneParser.parse(scope.items, scope.itemRegistry) logger.info("{} zones".format(len(zones))) output = '' for z in zones: output += '\n' + str(z) logger.info(output)