BAPSicle/helpers/device_manager.py
2021-04-17 12:27:18 +01:00

80 lines
2.5 KiB
Python

from typing import Any, Dict, List, Optional, Tuple
import sounddevice as sd
from helpers.os_environment import isLinux, isMacOS, isWindows
import glob
if isWindows():
from serial.tools.list_ports_windows import comports
# TODO: https://wiki.libsdl.org/FAQUsingSDL maybe try setting some of these env variables for choosing different host APIs?
WINDOWS_APIS = ["Windows DirectSound"]
class DeviceManager:
@classmethod
def _isOutput(cls, device: Dict[str, Any]) -> bool:
return device["max_output_channels"] > 0
@classmethod
def _isHostAPI(cls, host_api) -> bool:
return host_api
@classmethod
def _getAudioDevices(cls) -> sd.DeviceList:
# To update the list of devices
# Sadly this doesn't work on MacOS.
if not isMacOS():
sd._terminate()
sd._initialize()
devices: sd.DeviceList = sd.query_devices()
return devices
@classmethod
def getAudioOutputs(cls) -> Tuple[List[Dict]]:
host_apis = sd.query_hostapis()
devices: sd.DeviceList = cls._getAudioDevices()
for host_api_id in range(len(host_apis)):
if isWindows() and host_apis[host_api_id]["name"] not in WINDOWS_APIS:
host_apis[host_api_id]["usable"] = False
else:
host_apis[host_api_id]["usable"] = True
host_api_devices = (device for device in devices if device["hostapi"] == host_api_id)
outputs: List[Dict] = list(filter(cls._isOutput, host_api_devices))
outputs = sorted(outputs, key=lambda k: k["name"])
host_apis[host_api_id]["output_devices"] = outputs
return host_apis
@classmethod
def getSerialPorts(cls) -> List[Optional[str]]:
"""Lists serial port names
:raises EnvironmentError:
On unsupported or unknown platforms
:returns:
A list of the serial ports available on the system
"""
if isWindows():
ports = [port.device for port in comports()]
elif isLinux():
# this excludes your current terminal "/dev/tty"
ports = glob.glob("/dev/tty[A-Za-z]*")
elif isMacOS():
ports = glob.glob("/dev/tty.*")
else:
raise EnvironmentError("Unsupported platform")
valid: List[str] = ports
result: List[Optional[str]] = []
if len(valid) > 0:
valid.sort()
result.append(None) # Add the None option
result.extend(valid)
return result