BeoGateway/Resources/MLCONFIG.py

132 lines
6.3 KiB
Python
Raw Normal View History

2021-11-23 15:15:08 +00:00
import logging
import requests
import asyncore
import json
from collections import OrderedDict
2021-11-25 19:18:52 +00:00
import Resources.CONSTANTS as CONST
2021-11-23 15:15:08 +00:00
class MLConfig:
def __init__(self, host_address='blgw.local', user='admin', pwd='admin'):
2021-11-25 19:18:52 +00:00
self.log = logging.getLogger('Config')
2021-11-23 15:15:08 +00:00
self.log.setLevel('INFO')
self._host = host_address
self._user = user
self._pwd = pwd
2021-11-25 19:18:52 +00:00
self._download_data()
2021-11-23 15:15:08 +00:00
2021-11-25 19:18:52 +00:00
def _download_data(self):
2021-11-23 15:15:08 +00:00
self.log.info('Downloading configuration data from Gateway...')
url = 'http://' + self._host + '/mlgwpservices.json'
auth = (self._user, self._pwd)
response = requests.get(url, auth=auth)
configurationdata = json.loads(response.text)
2021-11-25 19:18:52 +00:00
self.configure_mlgw(configurationdata)
2021-11-23 15:15:08 +00:00
2021-11-25 19:18:52 +00:00
def configure_mlgw(self, data):
2021-11-23 15:15:08 +00:00
self.log.info('Processing Gateway configuration data...\n')
2021-11-25 19:18:52 +00:00
CONST.gateway['Serial_Number'] = data['sn']
CONST.gateway['Project'] = data['project']
CONST.gateway['Installer'] = str(data['installer']['name'])
CONST.gateway['Contact'] = str(data['installer']['contact'])
2021-11-23 15:15:08 +00:00
for zone in data["zones"]:
if int(zone['number']) == 240:
continue
room = OrderedDict()
room['Room_Number'] = zone['number']
room['Zone'] = str(zone['name']).split('/')[0]
room['Room_Name'] = str(zone['name']).split('/')[1]
room['Products'] = []
for product in zone["products"]:
device = OrderedDict()
room['Products'].append(str(product["name"]))
device['Device'] = str(product["name"])
device['MLN'] = product["MLN"]
device['ML_ID'] = ''
device['Serial_num'] = str(product["sn"])
device['Zone'] = str(zone["name"]).split('/')[0]
device['Room'] = str(zone["name"]).split('/')[1]
device['Room_Number'] = str(zone["number"])
device['Sources'] = OrderedDict()
for source in product["sources"]:
device['Sources'][str(source["name"])] = OrderedDict()
for selectCmd in source["selectCmds"]:
source_id = str(source['sourceId'].split(':')[0])
2021-11-25 19:18:52 +00:00
source_id = self._srcdictsanitize(CONST.blgw_srcdict, source_id).upper()
2021-11-23 15:15:08 +00:00
device['Sources'][str(source["name"])]['source'] = source_id
device['Sources'][str(source["name"])]['uniqueID'] = str(source['sourceId'])
source_tuple = (str(source["name"]), source_id)
2021-11-25 19:18:52 +00:00
cmd_tuple = (source_id, (int(selectCmd["cmd"]), int(selectCmd["unit"])))
2021-11-23 15:15:08 +00:00
device['Sources'][str(source["name"])]['BR1_cmd'] = cmd_tuple
2021-11-25 19:18:52 +00:00
if 'channels' in source:
device['Sources'][str(source["name"])]['channels'] = []
2021-11-23 15:15:08 +00:00
for channel in source['channels']:
2021-11-25 19:18:52 +00:00
c = OrderedDict()
2021-11-23 15:15:08 +00:00
c_num = ''
num = channel['selectSEQ'][::2]
for n in num:
c_num += str(n)
2021-11-25 19:18:52 +00:00
c['number'] = int(c_num)
c['name'] = channel['name']
c['icon'] = channel['icon']
device['Sources'][str(source["name"])]['channels'].append(c)
if source_tuple not in CONST.available_sources:
CONST.available_sources.append(source_tuple)
CONST.devices.append(device)
CONST.rooms.append(room)
self.log.info('Found ' + str(len(CONST.devices)) + ' AV Renderers!')
for i in range(len(CONST.devices)):
self.log.info('\tMLN ' + str(CONST.devices[i].get('MLN')) + ': ' + str(CONST.devices[i].get('Device')))
self.log.info('\tFound ' + str(len(CONST.available_sources)) + ' Available Sources [Name, Type]:')
for i in range(len(CONST.available_sources)):
self.log.info('\t\t' + str(list(CONST.available_sources[i])))
2021-11-23 15:15:08 +00:00
self.log.info('Done!\n')
2021-11-25 19:18:52 +00:00
self.log.debug(json.dumps(CONST.gateway, indent=4))
self.log.debug(json.dumps(CONST.rooms, indent=4))
self.log.debug(json.dumps(CONST.devices, indent=4))
2021-11-23 15:15:08 +00:00
def get_masterlink_id(self, mlgw, mlcli):
self.log.info("Finding MasterLink ID of products:")
if mlgw.is_connected and mlcli.is_connected:
2021-11-25 19:18:52 +00:00
for device in CONST.devices:
2021-11-23 15:15:08 +00:00
self.log.info("Finding MasterLink ID of product " + device.get('Device'))
# Ping the device with a light timeout to elicit a ML telegram containing its ML_ID
mlgw.send_beo4_cmd(int(device.get('MLN')),
2021-11-25 19:18:52 +00:00
CONST.CMDS_DEST.get("AUDIO SOURCE"),
CONST.BEO4_CMDS.get("LIGHT TIMEOUT"))
2021-11-23 15:15:08 +00:00
if device.get('Serial_num') is None:
# If this is a MasterLink product it has no serial number...
# loop to until expected response received from ML Command Line Interface
2021-11-25 19:18:52 +00:00
while 'to_device' not in mlcli.last_message and mlcli.last_message['from_device'] == \
"MLGW" and mlcli.last_message['payload_type'] == \
"MLGW_REMOTE_BEO4" and mlcli.last_message['payload']['command'] == "LIGHT TIMEOUT":
2021-11-23 15:15:08 +00:00
asyncore.loop(count=1, timeout=0.2)
device['ML_ID'] = mlcli.last_message.get('to_device')
2021-11-25 19:18:52 +00:00
self.log.info("\tMasterLink ID of product " +
device.get('Device') + " is " + device.get('ML_ID') + ".\n")
2021-11-23 15:15:08 +00:00
else:
# If this is a NetLink product then it has a serial number and no ML_ID
device['ML_ID'] = 'NA'
2021-11-25 19:18:52 +00:00
self.log.info("\tNetworkLink ID of product " + device.get('Device') + " is " +
device.get('Serial_num') + ". No MasterLink ID assigned.\n")
2021-11-23 15:15:08 +00:00
2021-11-25 19:18:52 +00:00
@staticmethod
def _srcdictsanitize(d, s):
2021-11-23 15:15:08 +00:00
result = d.get(s)
2021-11-25 19:18:52 +00:00
if result is None:
2021-11-23 15:15:08 +00:00
result = s
return str(result)