Topic - volsync.py - A Python Script to sync Devialet Phantom Systems volume in a group using a Phantom Hardware Remote
What is it? : A Python script to sync the volume of a Devialet Phantom System 1 to System 2 in a group
Scenario: I use a Phantom Remote connected to System1, which acts as the group leader for multiroom with two systems. The remote only changes the volume on the system the remote is paired with. I want it to change the volume on both systems in a multiroom group. To synchronize volume across both systems, you can use the provided Python script. This script listens for volume changes on System1 and applies the same changes to System2 with a specified offset. When you change the volume on System1 using the remote, the script automatically adjusts the volume on System2 to match.
The Volsync.py script:
How to use:
volsync.py
VOLUME_OFFSET
optionally used this to adjust the volume difference desired between the Leader and Follower systemExample output:
Using Verbose = 0
you should see something like this:
2025-02-02 21:10:23,983 - INFO - Syncing volume: System 1 (7) -> System 2 (12) 2025-02-02 21:11:34,134 - INFO - Syncing volume: System 1 (17) -> System 2 (22) 2025-02-02 21:12:14,221 - INFO - Syncing volume: System 1 (20) -> System 2 (25)
or (depending on the verbose flag, it will print more info)
The remote is paired with System 1, also the GROUP leader.
Example Python script: volsync.py
import requests
import time
import logging
# Define Phantom system IPs and settings
SYSTEM_1_IP = "192.168.0.31" # Leader system
SYSTEM_2_IP = "192.168.0.32" # Follower system
VOLUME_OFFSET = 5 # System 2 will be 5 units louder than System 1
VOLUME_ENDPOINT = "/ipcontrol/v1/systems/current/sources/current/soundControl/volume"
VERBOSE = 0 # 0 for less output, 1 for more output
# Set up logging
logging.basicConfig(level=logging.DEBUG if VERBOSE else logging.INFO,
format='%(asctime)s - %(levelname)s - %(message)s')
def get_volume(ip):
"""Retrieve current volume from a Phantom system."""
try:
response = requests.get(f"http://{ip}{VOLUME_ENDPOINT}", timeout=5)
response.raise_for_status()
return response.json().get("volume")
logging.debug(f"get_volume - {ip}: {volume}")
except requests.exceptions.RequestException as e:
logging.error(f"Error retrieving volume from {ip}: {e}")
return None
def set_volume(ip, volume):
"""Set the volume on a Phantom system."""
try:
response = requests.post(f"http://{ip}{VOLUME_ENDPOINT}", json={"volume": volume}, timeout=5)
response.raise_for_status()
logging.debug(f"set_volume to {volume} on {ip}")
except requests.exceptions.RequestException as e:
logging.error(f"Error setting volume on {ip}: {e}")
def clamp_volume(vol1, vol2, offset, min_vol=0, max_vol=100):
"""Clamp System 2's volume based on System 1's volume and the offset."""
if vol1 == 0:
logging.debug(f"clamp_volume found vol1 is zero return zero")
return 0
return max(min_vol, min(max_vol, vol1 + offset))
def sync_volume():
"""Sync volume from System 1 to System 2 with an offset."""
last_volume = None
polling_interval = 30
while True:
vol1, vol2 = get_volume(SYSTEM_1_IP), get_volume(SYSTEM_2_IP)
if vol1 is None:
time.sleep(polling_interval)
continue
if vol1 != last_volume:
logging.debug(f"Volume changed on System 1: {vol1}")
last_volume = vol1
polling_interval = 10
else:
polling_interval = 30
new_vol2 = clamp_volume(vol1, vol2, VOLUME_OFFSET)
if new_vol2 != vol2:
logging.info(f"Syncing volume: System 1 ({vol1}) -> System 2 ({new_vol2})")
set_volume(SYSTEM_2_IP, new_vol2)
time.sleep(polling_interval)
if __name__ == "__main__":
sync_volume()