voicemeeter-api-python/voicemeeterlib/util.py

121 lines
3.3 KiB
Python
Raw Normal View History

2022-06-16 14:07:12 +01:00
import functools
import time
from itertools import zip_longest
from typing import Iterator
2022-06-16 14:07:12 +01:00
from .error import CAPIError, VMError
def timeout(func):
"""
Times out the login function once time elapsed exceeds remote.timeout.
"""
@functools.wraps(func)
def wrapper(*args, **kwargs):
remote, *_ = args
func(*args, **kwargs)
err = None
start = time.time()
while time.time() < start + remote.timeout:
try:
time.sleep(0.1) # ensure at least 0.1 delay before clearing dirty
remote.logger.info(
2025-01-15 12:40:31 +00:00
f'{type(remote).__name__}: Successfully logged into {remote} version {remote.version}'
)
2025-01-15 12:40:31 +00:00
remote.logger.debug(f'login time: {round(time.time() - start, 2)}')
err = None
break
except CAPIError as e:
err = e
continue
if err:
2025-01-15 12:40:31 +00:00
raise VMError('Timeout logging into the api')
remote.clear_dirty()
return wrapper
2022-06-16 14:07:12 +01:00
def polling(func):
"""
Offers memoization for a set into get operation.
If sync clear dirty parameters before fetching new value.
Useful for loop getting if not running callbacks
"""
@functools.wraps(func)
def wrapper(*args, **kwargs):
2025-01-15 12:40:31 +00:00
get = func.__name__ == 'get'
mb_get = func.__name__ == 'get_buttonstatus'
2022-06-16 14:07:12 +01:00
remote, *remaining = args
if get:
param, *rem = remaining
elif mb_get:
id, mode, *rem = remaining
2025-01-15 12:40:31 +00:00
param = f'mb_{id}_{mode}'
2022-06-16 14:07:12 +01:00
if param in remote.cache:
return remote.cache.pop(param)
if remote.sync:
remote.clear_dirty()
return func(*args, **kwargs)
return wrapper
def script(func):
"""Convert dictionary to script"""
def wrapper(*args):
remote, script = args
if isinstance(script, dict):
2025-01-15 12:40:31 +00:00
params = ''
2022-06-16 14:07:12 +01:00
for key, val in script.items():
2025-01-15 12:40:31 +00:00
obj, m2, *rem = key.split('-')
2022-06-16 14:07:12 +01:00
index = int(m2) if m2.isnumeric() else int(*rem)
2025-01-15 12:40:31 +00:00
params += ';'.join(
2022-06-16 14:07:12 +01:00
f"{obj}{f'.{m2}stream' if not m2.isnumeric() else ''}[{index}].{k}={int(v) if isinstance(v, bool) else v}"
for k, v in val.items()
)
2025-01-15 12:40:31 +00:00
params += ';'
2022-06-16 14:07:12 +01:00
script = params
return func(remote, script)
return wrapper
def comp(t0: tuple, t1: tuple) -> Iterator[bool]:
"""
Generator function, accepts two tuples.
Evaluates equality of each member in both tuples.
"""
for a, b in zip(t0, t1):
yield a == b
def grouper(n, iterable, fillvalue=None):
"""
Group elements of an iterable by sets of n length
"""
args = [iter(iterable)] * n
return zip_longest(fillvalue=fillvalue, *args)
def deep_merge(dict1, dict2):
"""Generator function for deep merging two dicts"""
for k in set(dict1) | set(dict2):
if k in dict1 and k in dict2:
if isinstance(dict1[k], dict) and isinstance(dict2[k], dict):
yield k, dict(deep_merge(dict1[k], dict2[k]))
else:
yield k, dict2[k]
elif k in dict1:
yield k, dict1[k]
else:
yield k, dict2[k]