mirror of
https://github.com/LukeSpad/BeoGateway.git
synced 2024-12-23 21:51:51 +00:00
298 lines
15 KiB
Python
298 lines
15 KiB
Python
import indigo
|
|
import asyncore
|
|
import json
|
|
import requests
|
|
import logging
|
|
from requests.auth import HTTPDigestAuth, HTTPBasicAuth
|
|
from collections import OrderedDict
|
|
|
|
import Resources.CONSTANTS as CONST
|
|
|
|
|
|
class MLConfig:
|
|
|
|
def __init__(self, host_address='blgw.local', user='admin', pwd='admin', debug=False):
|
|
self.debug = debug
|
|
|
|
self._host = host_address
|
|
self._user = user
|
|
self._pwd = pwd
|
|
|
|
self._download_data()
|
|
|
|
def _download_data(self):
|
|
try:
|
|
indigo.server.log('Downloading configuration data from Gateway...', level=logging.WARNING)
|
|
url = 'http://' + self._host + '/mlgwpservices.json'
|
|
# try Basic Auth next (this is needed for the BLGW)
|
|
response = requests.get(url, auth=HTTPBasicAuth(self._user, self._pwd))
|
|
|
|
if response.status_code == 401:
|
|
# try Digest Auth first (this is needed for the MLGW)
|
|
response = requests.get(url, auth=HTTPDigestAuth(self._user, self._pwd))
|
|
|
|
if response.status_code == 401:
|
|
return
|
|
else:
|
|
# Once logged in successfully download and process the configuration data
|
|
configurationdata = json.loads(response.text)
|
|
self.configure_mlgw(configurationdata)
|
|
except ValueError:
|
|
pass
|
|
|
|
def configure_mlgw(self, data):
|
|
if "BeoGateway" not in indigo.devices.folders:
|
|
indigo.devices.folder.create("BeoGateway")
|
|
folder_id = indigo.devices.folders.getId("BeoGateway")
|
|
|
|
# Check to see if any devices already exist to avoid duplication
|
|
_nodes = []
|
|
for node in indigo.devices.iter('uk.co.lukes_plugins.BeoGateway.plugin.AVrenderer'):
|
|
_nodes.append(int(node.address))
|
|
|
|
indigo.server.log('Processing Gateway configuration data...\n', level=logging.WARNING)
|
|
# Check to see if gateway device exists and create one if not
|
|
try:
|
|
gw = indigo.device.create(
|
|
protocol=indigo.kProtocol.Plugin,
|
|
name="Bang and Olufsen Gateway",
|
|
description="Automatically generated device for BeoGateway plugin:\n"
|
|
" - Please do not delete or rename!\n"
|
|
" - Editing device properties for advanced users only!",
|
|
deviceTypeId="BOGateway",
|
|
pluginId='uk.co.lukes_plugins.BeoGateway.plugin',
|
|
folder=folder_id,
|
|
address=1
|
|
)
|
|
except ValueError:
|
|
gw = indigo.devices['Bang and Olufsen Gateway']
|
|
|
|
try:
|
|
gateway_type = 'blgw'
|
|
gw.replacePluginPropsOnServer(
|
|
{
|
|
'serial_no': data['sn'],
|
|
'project': data['project'],
|
|
'installer': str(data['installer']['name']),
|
|
'contact': str(data['installer']['contact']),
|
|
'isBLGW': 'BLGW'
|
|
}
|
|
)
|
|
except KeyError:
|
|
gateway_type = 'mlgw'
|
|
gw.replacePluginPropsOnServer(
|
|
{
|
|
'serial_no': data['sn'],
|
|
'project': data['project'],
|
|
'isBLGW': 'MLGW'
|
|
}
|
|
)
|
|
|
|
# Replace States
|
|
gw.updateStatesOnServer(CONST.gw_all_stb)
|
|
|
|
# Loop over the config data to find the rooms, devices and sources in the installation
|
|
for zone in data["zones"]:
|
|
# Get rooms
|
|
if int(zone['number']) == 240:
|
|
continue
|
|
room = OrderedDict()
|
|
room['Room_Number'] = zone['number']
|
|
if gateway_type == 'blgw':
|
|
# BLGW arranges rooms within zones
|
|
room['Zone'] = str(zone['name']).split('/')[0]
|
|
room['Room_Name'] = str(zone['name']).split('/')[1]
|
|
elif gateway_type == 'mlgw':
|
|
# MLGW has no zoning concept - devices are arranged in rooms only
|
|
room['Room_Name'] = str(zone['name'])
|
|
|
|
# Get products
|
|
for product in zone["products"]:
|
|
device = OrderedDict()
|
|
|
|
# Device identification
|
|
device['Device'] = str(product["name"])
|
|
device['MLN'] = product["MLN"]
|
|
device['ML_ID'] = ''
|
|
try:
|
|
device['Serial_num'] = str(product["sn"])
|
|
except KeyError:
|
|
device['Serial_num'] = 'NA'
|
|
|
|
# Physical location
|
|
if gateway_type == 'blgw':
|
|
# BLGW arranges rooms within zones
|
|
device['Zone'] = str(zone['name']).split('/')[0]
|
|
device['Room'] = str(zone['name']).split('/')[1]
|
|
elif gateway_type == 'mlgw':
|
|
# MLGW has no zoning concept - devices are arranged in rooms only
|
|
device['Room'] = str(zone['name'])
|
|
device['Room_Number'] = str(zone["number"])
|
|
|
|
# Source information
|
|
device['Sources'] = dict()
|
|
|
|
for source in product["sources"]:
|
|
src_name = str(source["name"]).strip().replace(' ', '_')
|
|
device['Sources'][src_name] = dict()
|
|
for selectCmd in source["selectCmds"]:
|
|
if gateway_type == 'blgw':
|
|
# get source information from the BLGW config file
|
|
if str(source['sourceId']) == '':
|
|
source_id = self._srcdictsanitize(CONST.beo4_commanddict, source['selectID']).upper()
|
|
else:
|
|
source_id = str(source['sourceId'].split(':')[0])
|
|
source_id = self._srcdictsanitize(CONST.blgw_srcdict, source_id).upper()
|
|
device['Sources'][src_name]['source'] = source_id
|
|
device['Sources'][src_name]['uniqueID'] = str(source['sourceId'])
|
|
else:
|
|
# MLGW config file is structured differently
|
|
source_id = self._srcdictsanitize(CONST.beo4_commanddict, source['selectID']).upper()
|
|
device['Sources'][src_name]['source'] = source_id
|
|
source_tuple = (str(source_id), str(source["name"]))
|
|
device['Sources'][src_name]['BR1_cmd'] = \
|
|
dict([('command', int(selectCmd["cmd"])), ('unit', int(selectCmd["unit"]))])
|
|
|
|
# Establish the channel list for sources with favourites lists
|
|
if 'channels' in source:
|
|
device['Sources'][src_name]['channels'] = []
|
|
for channel in source['channels']:
|
|
c_num = 'c'
|
|
if gateway_type == 'blgw':
|
|
num = channel['selectSEQ'][::2]
|
|
else:
|
|
num = channel['selectSEQ'][:-1]
|
|
for n in num:
|
|
c_num += str(n)
|
|
c = (c_num, str(channel['name']))
|
|
device['Sources'][src_name]['channels'].append(c)
|
|
|
|
if source_tuple not in CONST.available_sources:
|
|
CONST.available_sources.append(source_tuple)
|
|
|
|
# Create indigo devices to represent the B&O AV renderers in the installation
|
|
if int(device['MLN']) not in _nodes:
|
|
if self.debug:
|
|
indigo.server.log("New Device! Creating Indigo Device " + device['Device'],
|
|
level=logging.DEBUG)
|
|
|
|
node = indigo.device.create(
|
|
protocol=indigo.kProtocol.Plugin,
|
|
name=device['Device'],
|
|
description="Automatically generated device for BeoGateway plugin:\n"
|
|
"- Device data sourced from gateway config:\n"
|
|
"- Please do not delete or rename!\n"
|
|
"- Editing device properties for advanced users only!",
|
|
deviceTypeId="AVrenderer",
|
|
pluginId='uk.co.lukes_plugins.BeoGateway.plugin',
|
|
folder=folder_id,
|
|
address=device['MLN'],
|
|
props={
|
|
'serial_no': device['Serial_num'],
|
|
'mlid': 'NA',
|
|
'zone': room['Zone'],
|
|
'room': device['Room'],
|
|
'roomnum': device['Room_Number'],
|
|
'sources': device['Sources']
|
|
}
|
|
)
|
|
|
|
# Update the device states
|
|
node.updateStatesOnServer(CONST.standby_state)
|
|
node.updateStateImageOnServer(indigo.kStateImageSel.PowerOff)
|
|
else:
|
|
# if node already exists, update the properties in case they have been updated
|
|
for node in indigo.devices.iter('uk.co.lukes_plugins.BeoGateway.plugin.AVrenderer'):
|
|
if int(node.address) == int(device['MLN']):
|
|
if self.debug:
|
|
indigo.server.log("Old Device! Updating Properties for " + device['Device'] + "\n",
|
|
level=logging.DEBUG)
|
|
# Update the name of the device
|
|
node.name = device['Device']
|
|
node.description = "Automatically generated device for BeoGateway plugin:\n" \
|
|
" - Device data sourced from gateway config:\n" \
|
|
" - Please do not delete or rename!\n" \
|
|
" - Editing device properties for advanced users only!"
|
|
node.replaceOnServer()
|
|
# Update the properties of the device
|
|
node_props = node.pluginProps
|
|
node_props.update(
|
|
{
|
|
'serial_no': device['Serial_num'],
|
|
'zone': room['Zone'],
|
|
'room': device['Room'],
|
|
'roomnum': device['Room_Number'],
|
|
'sources': device['Sources']
|
|
}
|
|
)
|
|
node.replacePluginPropsOnServer(node_props)
|
|
|
|
# Update the device states
|
|
node.stateListOrDisplayStateIdChanged()
|
|
node.updateStatesOnServer(CONST.standby_state)
|
|
node.updateStateImageOnServer(indigo.kStateImageSel.PowerOff)
|
|
indigo.device.moveToFolder(node.id, value=folder_id)
|
|
break
|
|
|
|
# Keep track of the room data
|
|
CONST.rooms.append(room)
|
|
|
|
# Report details of the configuration
|
|
n_devices = indigo.devices.len(filter="uk.co.lukes_plugins.BeoGateway.plugin.AVrenderer") - 1
|
|
indigo.server.log('Found ' + str(n_devices) + ' AV Renderers!', level=logging.DEBUG)
|
|
for node in indigo.devices.iter('uk.co.lukes_plugins.BeoGateway.plugin.AVrenderer'):
|
|
indigo.server.log('\tMLN ' + str(node.address) + ': ' + str(node.name), level=logging.INFO)
|
|
indigo.server.log('\tFound ' + str(len(CONST.available_sources)) + ' Available Sources [Type, Name]:',
|
|
level=logging.DEBUG)
|
|
for i in range(len(CONST.available_sources)):
|
|
indigo.server.log('\t\t' + str(list(CONST.available_sources[i])), level=logging.INFO)
|
|
indigo.server.log('\tDone!\n', level=logging.DEBUG)
|
|
|
|
@staticmethod
|
|
def get_masterlink_id(mlgw, mlcli):
|
|
# Identify the MasterLink ID of products
|
|
indigo.server.log("Finding MasterLink ID of products:", level=logging.WARNING)
|
|
if mlgw.is_connected and mlcli.is_connected:
|
|
for node in indigo.devices.iter('uk.co.lukes_plugins.BeoGateway.plugin.AVrenderer'):
|
|
indigo.server.log("Finding MasterLink ID of product " + node.name, level=logging.WARNING)
|
|
# Ping the device with a light timeout to elicit a ML telegram containing its ML_ID
|
|
mlgw.send_beo4_cmd(int(node.address),
|
|
CONST.CMDS_DEST.get("AUDIO SOURCE"),
|
|
CONST.BEO4_CMDS.get("LIGHT TIMEOUT"))
|
|
node_props = node.pluginProps
|
|
if node_props['serial_no'] in [None, 'NA', '']:
|
|
# If this is a MasterLink product it has no serial number...
|
|
# loop to until expected response received from ML Command Line Interface
|
|
test = True
|
|
while test:
|
|
try:
|
|
if mlcli.last_message['from_device'] == "MLGW" and \
|
|
mlcli.last_message['payload_type'] == "MLGW_REMOTE_BEO4" and \
|
|
mlcli.last_message['State_Update']['command'] == "Light Timeout":
|
|
|
|
if node_props['mlid'] == 'NA':
|
|
node_props['mlid'] = mlcli.last_message.get('to_device')
|
|
node_props['serial_no'] = 'NA'
|
|
node.replacePluginPropsOnServer(node_props)
|
|
|
|
indigo.server.log("\tMasterLink ID of product " +
|
|
node.name + " is " + node_props['mlid'] + ".\n",
|
|
level=logging.DEBUG)
|
|
test = False
|
|
except KeyError:
|
|
asyncore.loop(count=1, timeout=0.2)
|
|
|
|
else:
|
|
# If this is a NetLink product then it has a serial number and no ML_ID
|
|
indigo.server.log("\tNetworkLink ID of product " + node.name + " is " +
|
|
node_props['serial_no'] + ". No MasterLink ID assigned.\n",
|
|
level=logging.DEBUG)
|
|
|
|
# ########################################################################################
|
|
# Utility Functions
|
|
@staticmethod
|
|
def _srcdictsanitize(d, s):
|
|
result = d.get(s)
|
|
if result is None:
|
|
result = s
|
|
return str(result)
|