#!/usr/bin/env python3
''' Check on esphome-devices '''
import argparse
import asyncio
import logging
import os
import sys
import common
#import colorlog
import zeroconf
from aioesphomeapi import APIClient, APIConnectionError, ReconnectLogic, SensorState
sleepsec = 60
noise_psk = os.environ['el_esphome_api_psk']
async def main(args):
"""Connect to an ESPHome device and get details."""
log.debug('function main()')
# Establish connection
api = APIClient(
address=args.address,
port=6053,
password='',
client_info="esphomeapi.py",
noise_psk=noise_psk)
log.debug('Connecting')
try:
await api.connect(login=True)
except APIConnectionError as e:
log.error("esphome api connection error - %s", e)
sys.exit(1)
log.info('Connected. Api version: ' + str(api.api_version))
# Show device details
log.debug('Getting device info')
device = vars(await api.device_info())
log.info('Device name: ' + device['name'])
log.debug('Getting sensors')
rawsensors, _ = await api.list_entities_services()
sensors = {}
for sensor in rawsensors:
sensors[sensor.key] = vars(sensor)
if log.isEnabledFor(logging.DEBUG):
from pprint import pformat
log.debug('Sensors: \n' + pformat(sensors))
log.info('Disconnecting')
await api.disconnect()
def callback(state):
if type(state) == SensorState and state.missing_state is False:
log.debug('function callback(' + str(state) + ')')
sensor = sensors[state.key]
value = state.state
if 'accuracy_decimals' in sensor:
decimals = sensor['accuracy_decimals']
value = round(value, decimals) if decimals > 0 else round(value)
unit = sensor['unit_of_measurement'] if 'unit_of_measurement' in sensor else ''
device_class = sensor['device_class'] if 'device_class' in sensor else ''
log.info(sensor['name'] + ' ' + sensor['device_class'] + ' - ' + str(value) + str(unit))
common.statein(device['friendly_name'], value, device_class, unit, verbose=True)
async def on_connect() -> None:
log.debug('function on_connect()')
log.info('Connected to API')
try:
await api.subscribe_states(callback)
except APIConnectionError as e:
log.error("esphome api connection error - %s", e)
await api.disconnect()
except Exception as e:
log.error("catched exception - %s", e)
await api.disconnect()
async def on_disconnect() -> None:
log.debug('function on_disconnect()')
log.warning("Disconnected from API")
await asyncio.sleep(sleepsec)
reconnect = ReconnectLogic(
client=api,
on_connect=on_connect,
on_disconnect=on_disconnect,
zeroconf_instance=zeroconf.Zeroconf()
)
await reconnect.start()
try:
while True:
try:
log.debug('Sleep for ' + str(sleepsec) + 's')
await asyncio.sleep(sleepsec)
except Exception as e:
log.error("catched exception - %s", e)
except KeyboardInterrupt:
log.error("Keyboard interrupt")
pass
except KeyboardInterrupt:
log.error("Keyboard interrupt")
await reconnect.stop()
finally:
log.info("Closing loop.")
if __name__ == "__main__":
# Logging
#handler = colorlog.StreamHandler()
#handler.setFormatter(colorlog.ColoredFormatter(
# "%(log_color)s%(levelname)s - %(message)s {%(filename)s:%(lineno)d}",
# log_colors={
# 'DEBUG': 'light_black',
# 'INFO': 'cyan',
# 'WARNING': 'yellow',
# 'ERROR': 'red',
# 'CRITICAL': 'red,bg_white'
# }))
#log = colorlog.getLogger(__name__)
#log.setLevel(logging.WARNING)
#log.addHandler(handler)
log = logging.getLogger(__name__)
log.setLevel(logging.WARNING)
logging.basicConfig(format="%(levelname)s - %(message)s {%(filename)s:%(lineno)d}")
parser = argparse.ArgumentParser(description="Connect to an esphome-device and access the native api")
parser.add_argument( "-a", "--address", help="Address of esp-device to connect to")
parser.add_argument( "-v", "--verbose", help="Set logging to debug mode", action="store_true")
args = parser.parse_args()
# Verbose logging?
if args.verbose:
log.setLevel(logging.DEBUG)
log.debug('asyncio.run(main(args))')
asyncio.run(main(args))
print('Bottom of script. Exiting.')
sys.exit(0)