Skip to main content

Package for controlling VSSL's range of streaming amplifiers

Project description

vsslctrl

Package for controlling VSSL's range of streaming amplifiers.

vsslctrl is not endorsed or affiliated with VSSL in any manner.

Help

I am looking for testers with any VSSL amplifier models, please get in touch if you interested in helping. vsslcontrolled@proton.me

Tested on:

  • A.1 software version p15265.033.3703 via Home Assistant integration (reported)
  • A.3 software version p12013.141.3703 via Home Assistant integration (reported)
  • A.3x software version p15305.016.3701
  • A.6x software version p15305.017.3701 via Home Assistant integration (reported)

Important

There should not be any VSSL Agent's running on the same network. If you dont know what this is, then you can ignore this notice.

TODOs

  • A.1(x) specific control e.g bluetooth
  • Better test coverage

Basic Usage

vsslctrl needs to be running inside a asyncio event loop.

A.1 Example

import asyncio
from vsslctrl import Vssl, DeviceModels, Zone, ZoneIDs

async def main():
  
  # Represents a physical VSSL amplifier
  vssl = Vssl(DeviceModels.A1)
  a1 = vssl.add_zone(ZoneIDs.A1, '192.168.1.10')

  # Connect and initiate zones.
  await vssl.initialise()

  """Control Examples"""
  # Print device name
  print(a1.settings.name)
  # Set volume to 25%
  a1.volume = 25
  # Pause
  a1.pause()
  # Print track name
  print(a1.track.name)

  # Shutdown and disconnect all zones
  await vssl.shutdown()

asyncio.run(main())

A.3x Example

import asyncio
from vsslctrl import Vssl, DeviceModels, Zone, ZoneIDs

async def main():
	
  # Represents a physical VSSL amplifier
  # If no DeviceModels is passed, vsslctrl will default to the feature set of the X series amps
  vssl = Vssl(DeviceModels.A3X)

  # Add each you wish to control
  zone1 = vssl.add_zone(ZoneIDs.ZONE_1, '192.168.1.10')
  zone2 = vssl.add_zone(ZoneIDs.ZONE_2, '192.168.1.11')
  zone3 = vssl.add_zone(ZoneIDs.ZONE_3, '192.168.1.12')
  #... up to 6 zones

  # Connect and initiate zones.
  await vssl.initialise()

  """Control Examples"""
  # Print zone1 name
  print(zone1.settings.name)
  # Set zone2 volume to 25%
  zone2.volume = 25
  # Pause zone3
  zone3.pause()
  # or zone3.transport.pause()
  # Print zone1 track name
  print(zone1.track.name)


  # Shutdown and disconnect all zones
  await vssl.shutdown()


asyncio.run(main())

API

Most functionality is achived via getters and setters of the two main classes Vssl, Zone.

The classes will update the physical VSSL device when setting a property and once feedback has been received, the classes internal state will be updated. For example:

# Setting the zones name
zone1.settings.name = 'Living Room'
>>> None

# Printing zone name
zone_name = zone1.settings.name
print(zone_name)
>>> 'Living Room'

Important in the above example, zone1.settings.name wont be set to its new value until after the VSSL device has changed the name and the Zone class has received confirmation feedback. If you need to wait for the value change, you can await a [property_name]_CHANGE events.

Vssl

Property Description Type
sw_version Software version str readonly
serial Serial number str readonly
model Device Model int readonly
reboot() Reboot all zones func
factory_reset() Factory reset device func
"""Example"""
# Reboot all zones
vssl.reboot()
# Do a factory reset (reset all settings)
vssl.factory_reset()

Vssl.settings

Property Description Type
name Device name str
optical_input_name Name of the optical input str
bluetooth Bluetooth enabled / disabled bool
bluetooth_toggle() Toggle Bluetooth func
"""Example"""
# Setting device name
vssl.settings.name = 'My House'
# Setting optical input name
vssl.settings.optical_input_name = 'Optical Input 1'
# Enable Bluetooth
vssl.settings.bluetooth = True
# Toggle Bluetooth
vssl.settings.bluetooth_toggle()

Vssl.settings.power

Property Description Type Values
state Power state int readonly VsslPowerSettings.States
adaptive Power adaptive bool
"""Example"""
# Setting power adaptive
vssl.settings.power.adaptive = True

Zone

Property Description Type Values
id Zone number / ID int readonly ZoneIDs
host IP address str readonly
volume Volume int 0...100
volume_raise([step=1]) Raise volume by step func step: int 1...100
volume_lower([step=1]) Lower volume by step func step: int 1...100
mute Volume muted bool
mute_toggle() Mute / Unmute func
play() Play func
stop() Stop func
pause() Pause func
next() Next track func
prev() Begining of track or previous track func
reboot() Reboot zone func
play_url([url], [all_zones]) Play a URL func url: str, all_zones: bool
"""Examples"""
# Set volume to 50%
zone1.volume = 50
# Raise volume by 5%
zone1.volume_raise(5)
# Mute
zone1.mute = True
# Toggle mute
zone1.mute_toggle()
# Pause transport
zone1.pause()
# Next track
zone1.next()
# Play a URL on this zone1
zone1.play_url('http://soundbible.com/grab.php?id=2217&type=mp3')
# Play a URL on all zones
zone1.play_url('http://soundbible.com/grab.php?id=2217&type=mp3', True)

Zone.transport

A VSSL amplifier can not start a stream except for playing a URL directly. This is a limitation of the hardware itself.

Property Description Type Values
state Transport state. i.e Play, Stop, Pause int ZoneTransport.States
play() Play func
stop() Stop func
pause() Pause func
next() Next track func
prev() Begining of track or previous track func
is_playing Is the zone playing bool readonly
is_stopped Is the zone stopped bool readonly
is_pasued Is the zone pasued bool readonly
is_repeat Repeat state. i.e all, one, off int readonly ZoneTransport.Repeat
is_shuffle Is shuffle enabled bool readonly
has_next Is the next button enabled bool readonly
has_prev Is the prev button enabled bool readonly
"""Example"""
# Pause the stream
zone1.transport.pause()
# or
zone1.transport.state = ZoneTransport.States.PAUSE

Zone.track

  • Not all sources have complete metadata - missing value will be set to defaults.
  • Airplay track progress is not available.
Property Description Type Values
title Title str readonly
album Album str readonly
artist Artist str readonly
genre Genre str readonly
duration Length in miliseconds (ms) int readonly
progress Current position in miliseconds (ms) int readonly
cover_art_url URL to cover art str readonly
source Track source e.g Spotify int readonly TrackMetadata.Sources
url URL of file or track str readonly

Zone.input

Property Description Type Values
source Change input source int InputRouter.Sources
priority Change input priority. Stream or analog in higher priority int InputRouter.Priorities

Priority Order:

InputRouter.Priorities Priority Order
STREAM
  1. Stream
  2. Party Zone
  3. Bus 1 In
  4. Bus 2 In
  5. Analog Input
LOCAL
  1. Bus 1 In
  2. Bus 2 In
  3. Analog Input
  4. Stream
  5. Party Zone
"""Example"""
# Change zone 1 to listen to analog input 4
zone1.input.source = InputRouter.Sources.ANALOG_IN_4

# Change zone 1 to perfer analog input (local) over stream
zone1.input.priority = InputRouter.Priorities.LOCAL

Zone.group

Officially unsupported in X series amplifiers.

Property Description Type Values
source Zone ID of group master / source int readonly ZoneIDs
is_master This zone is the group master bool readonly
add_member() Add zone to group / create group func ZoneIDs
remove_member() Remove zone from group func ZoneIDs
dissolve() Dissolve group / remove all members func
leave() Leave the group if a member func
is_party_zone_member Member of Party Zone bool
"""Examples"""
# Add group 2 to a group with zone 1 as master
zone1.group.add_member(ZoneIDs.ZONE_2)
# Remove zone 2 from group
zone2.group.leave() # or
zone1.group.remove_member(ZoneIDs.ZONE_2)
# If zone 1 is a master, remove all members
zone1.group.dissolve()
# Add zone to the party zone group
zone1.group.is_party_zone_member = True
# Toggle Party Zone Membership
zone1.group.is_party_zone_member_toggle()

Zone.analog_output

Property Description Type Values Default
source Where the AO is routed from. i.e a zone, optical input or off int AnalogOutput.Sources Off
is_fixed_volume Fix the output volume. Output wont respond to volume control bool False
is_fixed_volume_toggle() Toggle fixed volume func
"""Examples"""
# Change analog output of zone1 to be outputting the optical input
zone1.analog_output.source = AnalogOutput.Sources.OPTICAL_IN

# Change analog output of zone1 to be outputting the zone 2 source (whatever zone 2 is using as a source)
zone1.analog_output.source = AnalogOutput.Sources.ZONE_2

# Fix the analog output volume. 
zone1.analog_output.is_fixed_volume = True

Zone.settings

Property Description Type Values Default
name Name str
disabled Disable the zone bool False
disabled_toggle() disable / enable func
mono Set output to mono or stereo int ZoneSettings.StereoMono Stereo
mono_toggle() Toggle mono or stereo func
"""Examples"""
# Set name
zone1.settings.name = 'Living Room'
# Disable Zone
zone1.disabled = True
# Toggle mono output
zone1.mono_toggle()

Zone.settings.analog_input

Property Description Type Values Default
name Name str
fixed_gain Fix the input gain to a specific value int 0...100 0 is disabled or variable gain
"""Examples"""
# Change zone1 analog input name
zone1.settings.analog_input.name = 'BluRay Player'

# Fix zone1 analog input gain to 50%.
zone1.settings.analog_input.fixed_gain = 50

Zone.settings.volume

Property Description Type Values Default
default_on Default on volume int 0...100 0 is disabled
max_left Max volume left channel int 0...100 75
max_right Max volume right channel int 0...100 75
"""Examples"""
# Set default on volume to 50%
zone1.settings.volume.default_on = 50
# Set maximum volume for left channel to 75%
zone1.settings.volume.default_on = 75

Zone.settings.eq

Property Description Type Values Default
enabled Enable / disable EQ bool False

EQ to be set in decibel using a range -10dB to +10dB

Property Description Type Values Default
hz60_db 60Hz int -10...10 0
hz200_db 200Hz int -10...10 0
hz500_db 500Hz int -10...10 0
khz1_db 1kHz int -10...10 0
khz4_db 4kHz int -10...10 0
khz8_db 8kHz int -10...10 0
khz15_db 15kHz int -10...10 0
"""Examples"""
# Set 1kHz EQ to -2
zone1.settings.eq.khz1_db = -2

Zone.settings.subwoofer

  • A.1 and A.1x only
  • Set 0 for full frequency range
Property Description Type Values Default
crossover Set "sub out" crossover frequency from 50-200Hz. int 0 or 50...200 0
"""Examples"""
# Set subwoofer ouput crossover to 100hz
zone1.settings.subwoofer.crossover = 100

Another (Lite) Way

If you perfer to not run the complete intergration, you can send basic HEX commands to the VSSL device using Netcat (or any network tool) on port 50002.

HEX Description
\x10\x05\x03\x0{Zone Number}\xff\x03 Volume Up
\x10\x05\x03\x0{Zone Number}\xfe\x03 Volume Down
\x10\x11\x02\x0{Zone Number}\x01 Mute
All commands can be found by looking here

The volume up HEX command for Zone 2 would be \x10\x05\x03\x02\xff\x03

Now send the raw HEX using Netcat to the device using this syntax:

echo -e "{HEX Command}" | nc {IP Address} 50002

For example to send volume up to Zone 2:

echo -e "\x10\x05\x03\x02\xff\x03" | nc 192.168.1.11 50002

Home Assistant Configuration.yaml example:

...

shell_command:
  #Zone 1
  vssl_zone_1_volume_up: 'echo -e "\x10\x05\x03\x01\xff\x03" | nc 192.168.1.10 50002'
  vssl_zone_1_volume_down: 'echo -e "\x10\x05\x03\x01\xfe\x03" | nc 192.168.1.10 50002'
  vssl_zone_1_mute: 'echo -e "\x10\x11\x02\x01\x01" | nc 192.168.1.10 50002'
  vssl_zone_1_unmute: 'echo -e "\x10\x11\x02\x01\x00" | nc 192.168.1.10 50002'
  #Zone 2
  vssl_zone_2_volume_up: 'echo -e "\x10\x05\x03\x02\xff\x03" | nc 192.168.1.11 50002'
  vssl_zone_2_volume_down: 'echo -e "\x10\x05\x03\x02\xfe\x03" | nc 192.168.1.11 50002'
  vssl_zone_2_mute: 'echo -e "\x10\x11\x02\x02\x01" | nc 192.168.1.11 50002'
  vssl_zone_2_unmute: 'echo -e "\x10\x11\x02\x02\x00" | nc 192.168.1.11 50002'
  #Zone 3
  vssl_zone_3_volume_up: 'echo -e "\x10\x05\x03\x03\xff\x03" | nc 192.168.1.12 50002'
  vssl_zone_3_volume_down: 'echo -e "\x10\x05\x03\x03\xfe\x03" | nc 192.168.1.12 50002'
  vssl_zone_3_mute: 'echo -e "\x10\x11\x02\x03\x01" | nc 192.168.1.12 50002'
  vssl_zone_3_unmute: 'echo -e "\x10\x11\x02\x03\x00" | nc 192.168.1.12 50002'

 ...

Credit

Thanks to @dj-jam for the continued testing.

The VSSL API was reverse engineered using Wireshark, VSSLs native "legacy" iOS app and their deprecated vsslagent.

Motivation for this project was to integrate VSSLs amplifiers into Home Assistant and have control over different subnets (not mDNS dependant)

Known Issues & Limitiations

  • Not tested on A.1x or original A series range of amplifiers (testers welcome)
  • VSSL can not start a stream except for playing a URL directly. This is a limitation of the hardware itself.
  • Not all sources set the volume to 0 when the zone is muted
  • Grouping feedback is flaky on the X series amplifiers
  • Airplay Zone.track.progress is not available.
  • Cant stop a URL playback, feedback is worng at least
  • VSSL likes to cache old track metadata. For example when playing a URL after Spotify, often the device will respond with the previous (Spotify) tracks metadata
  • stop() is intended to disconnect the client and pause the stream. Doesn’t always function this way, depending on stream source
  • Occasionally a zones might stop responding to certain commands, issuing the reboot command generally corrects

Future

  • A.1(x) coverage i.e Bluetooth
  • REST API / Web App
  • Save and recall EQ
  • IR Control

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

vsslctrl-0.1.10.dev1.tar.gz (53.3 kB view details)

Uploaded Source

Built Distribution

vsslctrl-0.1.10.dev1-py3-none-any.whl (51.0 kB view details)

Uploaded Python 3

File details

Details for the file vsslctrl-0.1.10.dev1.tar.gz.

File metadata

  • Download URL: vsslctrl-0.1.10.dev1.tar.gz
  • Upload date:
  • Size: 53.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/5.1.0 CPython/3.9.6

File hashes

Hashes for vsslctrl-0.1.10.dev1.tar.gz
Algorithm Hash digest
SHA256 478feb77b31ff9466b24676e3721fc7f8012af8bc2bd3b431fe917f61c91f541
MD5 7bdebee03aacc549a86359cf9b2b7ad4
BLAKE2b-256 6e6bf677ae7a95dd99b3eb8eeb36e6055d1089dc3d41c7a24d57f426b88c03cc

See more details on using hashes here.

File details

Details for the file vsslctrl-0.1.10.dev1-py3-none-any.whl.

File metadata

File hashes

Hashes for vsslctrl-0.1.10.dev1-py3-none-any.whl
Algorithm Hash digest
SHA256 6a5f138b7e693702f742462fdbe4535165974c16fd8f895ee3f56c94a9ddf88a
MD5 5ace5b56adc0d7521f1ca7bc3bcddcad
BLAKE2b-256 5e670a426bd951442376a70bbdafeea3081921f944a1da4f39d9d17ee0596405

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page