mirror of
https://github.com/home-assistant/core.git
synced 2025-07-14 00:37:13 +00:00
media_player/denon: refactor
* connect only if necessary * do not throw errors if offline
This commit is contained in:
parent
853a9fd4cd
commit
b6e6512367
@ -6,8 +6,7 @@ Developed for a Denon DRA-N5, see
|
|||||||
http://www.denon.co.uk/chg/product/compactsystems/networkmusicsystems/ceolpiccolo
|
http://www.denon.co.uk/chg/product/compactsystems/networkmusicsystems/ceolpiccolo
|
||||||
|
|
||||||
A few notes:
|
A few notes:
|
||||||
- As long as this module is active and connected, the receiver does
|
- The receiver handles only one telnet connection and refuses others.
|
||||||
not seem to accept additional telnet connections.
|
|
||||||
|
|
||||||
- Be careful with the volume. 50% or even 100% are very loud.
|
- Be careful with the volume. 50% or even 100% are very loud.
|
||||||
|
|
||||||
@ -67,13 +66,15 @@ def setup_platform(hass, config, add_devices, discovery_info=None):
|
|||||||
CONF_HOST)
|
CONF_HOST)
|
||||||
return False
|
return False
|
||||||
|
|
||||||
add_devices([
|
denon = DenonDevice(
|
||||||
DenonDevice(
|
config.get("name", "Music station"),
|
||||||
config.get('name', 'Music station'),
|
config.get("host")
|
||||||
config.get('host'))
|
)
|
||||||
])
|
if denon.update():
|
||||||
|
add_devices([denon])
|
||||||
return True
|
return True
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
class DenonDevice(MediaPlayerDevice):
|
class DenonDevice(MediaPlayerDevice):
|
||||||
@ -84,28 +85,41 @@ class DenonDevice(MediaPlayerDevice):
|
|||||||
def __init__(self, name, host):
|
def __init__(self, name, host):
|
||||||
self._name = name
|
self._name = name
|
||||||
self._host = host
|
self._host = host
|
||||||
self._telnet = telnetlib.Telnet(self._host)
|
self._pwstate = "PWSTANDBY"
|
||||||
|
self._volume = 0
|
||||||
|
self._muted = False
|
||||||
|
self._mediasource = ""
|
||||||
|
|
||||||
def query(self, message):
|
@classmethod
|
||||||
""" Send request and await response from server """
|
def telnet_request(cls, telnet, command):
|
||||||
|
""" Executes `command` and returns the response. """
|
||||||
|
telnet.write(command.encode("ASCII") + b"\r")
|
||||||
|
return telnet.read_until(b"\r", timeout=0.2).decode("ASCII").strip()
|
||||||
|
|
||||||
|
def telnet_command(self, command):
|
||||||
|
""" Establishes a telnet connection and sends `command`. """
|
||||||
|
telnet = telnetlib.Telnet(self._host)
|
||||||
|
telnet.write(command.encode("ASCII") + b"\r")
|
||||||
|
telnet.read_very_eager() # skip response
|
||||||
|
telnet.close()
|
||||||
|
|
||||||
|
def update(self):
|
||||||
try:
|
try:
|
||||||
# unspecified command, should be ignored
|
telnet = telnetlib.Telnet(self._host)
|
||||||
self._telnet.write("?".encode('UTF-8') + b'\r')
|
except ConnectionRefusedError:
|
||||||
except (EOFError, BrokenPipeError, ConnectionResetError):
|
return False
|
||||||
self._telnet.open(self._host)
|
|
||||||
|
|
||||||
self._telnet.read_very_eager() # skip what is not requested
|
self._pwstate = self.telnet_request(telnet, "PW?")
|
||||||
|
# PW? sends also SISTATUS, which is not interesting
|
||||||
|
telnet.read_until(b"\r", timeout=0.2)
|
||||||
|
|
||||||
self._telnet.write(message.encode('ASCII') + b'\r')
|
volume_str = self.telnet_request(telnet, "MV?")[len("MV"):]
|
||||||
# timeout 200ms, defined by protocol
|
self._volume = int(volume_str) / 60
|
||||||
resp = self._telnet.read_until(b'\r', timeout=0.2)\
|
self._muted = (self.telnet_request(telnet, "MU?") == "MUON")
|
||||||
.decode('UTF-8').strip()
|
self._mediasource = self.telnet_request(telnet, "SI?")[len("SI"):]
|
||||||
|
|
||||||
if message == "PW?":
|
telnet.close()
|
||||||
# workaround; PW? sends also SISTATUS
|
return True
|
||||||
self._telnet.read_until(b'\r', timeout=0.2)
|
|
||||||
|
|
||||||
return resp
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def name(self):
|
def name(self):
|
||||||
@ -115,10 +129,9 @@ class DenonDevice(MediaPlayerDevice):
|
|||||||
@property
|
@property
|
||||||
def state(self):
|
def state(self):
|
||||||
""" Returns the state of the device. """
|
""" Returns the state of the device. """
|
||||||
pwstate = self.query('PW?')
|
if self._pwstate == "PWSTANDBY":
|
||||||
if pwstate == "PWSTANDBY":
|
|
||||||
return STATE_OFF
|
return STATE_OFF
|
||||||
if pwstate == "PWON":
|
if self._pwstate == "PWON":
|
||||||
return STATE_ON
|
return STATE_ON
|
||||||
|
|
||||||
return STATE_UNKNOWN
|
return STATE_UNKNOWN
|
||||||
@ -126,17 +139,17 @@ class DenonDevice(MediaPlayerDevice):
|
|||||||
@property
|
@property
|
||||||
def volume_level(self):
|
def volume_level(self):
|
||||||
""" Volume level of the media player (0..1). """
|
""" Volume level of the media player (0..1). """
|
||||||
return int(self.query('MV?')[len('MV'):]) / 60
|
return self._volume
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def is_volume_muted(self):
|
def is_volume_muted(self):
|
||||||
""" Boolean if volume is currently muted. """
|
""" Boolean if volume is currently muted. """
|
||||||
return self.query('MU?') == "MUON"
|
return self._muted
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def media_title(self):
|
def media_title(self):
|
||||||
""" Current media source. """
|
""" Current media source. """
|
||||||
return self.query('SI?')[len('SI'):]
|
return self._mediasource
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def supported_media_commands(self):
|
def supported_media_commands(self):
|
||||||
@ -145,24 +158,24 @@ class DenonDevice(MediaPlayerDevice):
|
|||||||
|
|
||||||
def turn_off(self):
|
def turn_off(self):
|
||||||
""" turn_off media player. """
|
""" turn_off media player. """
|
||||||
self.query('PWSTANDBY')
|
self.telnet_command("PWSTANDBY")
|
||||||
|
|
||||||
def volume_up(self):
|
def volume_up(self):
|
||||||
""" volume_up media player. """
|
""" volume_up media player. """
|
||||||
self.query('MVUP')
|
self.telnet_command("MVUP")
|
||||||
|
|
||||||
def volume_down(self):
|
def volume_down(self):
|
||||||
""" volume_down media player. """
|
""" volume_down media player. """
|
||||||
self.query('MVDOWN')
|
self.telnet_command("MVDOWN")
|
||||||
|
|
||||||
def set_volume_level(self, volume):
|
def set_volume_level(self, volume):
|
||||||
""" set volume level, range 0..1. """
|
""" set volume level, range 0..1. """
|
||||||
# 60dB max
|
# 60dB max
|
||||||
self.query('MV' + str(round(volume * 60)).zfill(2))
|
self.telnet_command("MV" + str(round(volume * 60)).zfill(2))
|
||||||
|
|
||||||
def mute_volume(self, mute):
|
def mute_volume(self, mute):
|
||||||
""" mute (true) or unmute (false) media player. """
|
""" mute (true) or unmute (false) media player. """
|
||||||
self.query('MU' + ('ON' if mute else 'OFF'))
|
self.telnet_command("MU" + ("ON" if mute else "OFF"))
|
||||||
|
|
||||||
def media_play_pause(self):
|
def media_play_pause(self):
|
||||||
""" media_play_pause media player. """
|
""" media_play_pause media player. """
|
||||||
@ -170,22 +183,22 @@ class DenonDevice(MediaPlayerDevice):
|
|||||||
|
|
||||||
def media_play(self):
|
def media_play(self):
|
||||||
""" media_play media player. """
|
""" media_play media player. """
|
||||||
self.query('NS9A')
|
self.telnet_command("NS9A")
|
||||||
|
|
||||||
def media_pause(self):
|
def media_pause(self):
|
||||||
""" media_pause media player. """
|
""" media_pause media player. """
|
||||||
self.query('NS9B')
|
self.telnet_command("NS9B")
|
||||||
|
|
||||||
def media_next_track(self):
|
def media_next_track(self):
|
||||||
""" Send next track command. """
|
""" Send next track command. """
|
||||||
self.query('NS9D')
|
self.telnet_command("NS9D")
|
||||||
|
|
||||||
def media_previous_track(self):
|
def media_previous_track(self):
|
||||||
self.query('NS9E')
|
self.telnet_command("NS9E")
|
||||||
|
|
||||||
def media_seek(self, position):
|
def media_seek(self, position):
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
def turn_on(self):
|
def turn_on(self):
|
||||||
""" turn the media player on. """
|
""" turn the media player on. """
|
||||||
self.query('PWON')
|
self.telnet_command("PWON")
|
||||||
|
Loading…
x
Reference in New Issue
Block a user