mirror of
https://github.com/onyx-and-iris/vban-cmd-python.git
synced 2026-03-12 04:59:09 +00:00
Compare commits
4 Commits
98ec9b715f
...
c46ca8a8c8
| Author | SHA1 | Date | |
|---|---|---|---|
| c46ca8a8c8 | |||
| f8b56b4a30 | |||
| 09259269d7 | |||
| 242401e294 |
@ -548,7 +548,7 @@ You may pass the following optional keyword arguments:
|
|||||||
- `channel`: int=0, channel on which to send the UDP requests.
|
- `channel`: int=0, channel on which to send the UDP requests.
|
||||||
- `pdirty`: boolean=False, parameter updates
|
- `pdirty`: boolean=False, parameter updates
|
||||||
- `ldirty`: boolean=False, level updates
|
- `ldirty`: boolean=False, level updates
|
||||||
- `script_ratelimit`: float=0.05, default to 20 script requests per second. This affects vban.sendtext() specifically.
|
- `script_ratelimit`: float | None=None, ratelimit for vban.sendtext() specifically.
|
||||||
- `timeout`: int=5, timeout for socket operations.
|
- `timeout`: int=5, timeout for socket operations.
|
||||||
- `disable_rt_listeners`: boolean=False, set `True` if you don't wish to receive RT packets.
|
- `disable_rt_listeners`: boolean=False, set `True` if you don't wish to receive RT packets.
|
||||||
- You can still send Matrix string requests ending with `?` and receive a response.
|
- You can still send Matrix string requests ending with `?` and receive a response.
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
[project]
|
[project]
|
||||||
name = "vban-cmd"
|
name = "vban-cmd"
|
||||||
version = "2.10.1"
|
version = "2.10.2"
|
||||||
description = "Python interface for the VBAN RT Packet Service (Sendtext)"
|
description = "Python interface for the VBAN RT Packet Service (Sendtext)"
|
||||||
authors = [{ name = "Onyx and Iris", email = "code@onyxandiris.online" }]
|
authors = [{ name = "Onyx and Iris", email = "code@onyxandiris.online" }]
|
||||||
license = { text = "MIT" }
|
license = { text = "MIT" }
|
||||||
|
|||||||
@ -1,6 +1,18 @@
|
|||||||
|
from .packet.enums import ServiceTypes, SubProtocols
|
||||||
|
|
||||||
|
|
||||||
class VBANCMDError(Exception):
|
class VBANCMDError(Exception):
|
||||||
"""Base VBANCMD Exception class."""
|
"""Base VBANCMD Exception class."""
|
||||||
|
|
||||||
|
|
||||||
class VBANCMDConnectionError(VBANCMDError):
|
class VBANCMDConnectionError(VBANCMDError):
|
||||||
"""Exception raised when connection/timeout errors occur"""
|
"""Exception raised when connection/timeout errors occur"""
|
||||||
|
|
||||||
|
|
||||||
|
class VBANCMDPacketError(VBANCMDError):
|
||||||
|
"""Exception raised when packet parsing errors occur"""
|
||||||
|
|
||||||
|
def __init__(self, message: str, protocol: SubProtocols, type_: ServiceTypes):
|
||||||
|
super().__init__(message)
|
||||||
|
self.protocol = protocol
|
||||||
|
self.type = type_
|
||||||
|
|||||||
@ -89,7 +89,7 @@ class FactoryBase(VbanCmd):
|
|||||||
'streamname': 'Command1',
|
'streamname': 'Command1',
|
||||||
'bps': 256000,
|
'bps': 256000,
|
||||||
'channel': 0,
|
'channel': 0,
|
||||||
'script_ratelimit': 0.05, # 20 commands per second, to avoid overloading Voicemeeter
|
'script_ratelimit': None, # if None or 0, no rate limit applied to script commands
|
||||||
'timeout': 5, # timeout on socket operations, in seconds
|
'timeout': 5, # timeout on socket operations, in seconds
|
||||||
'disable_rt_listeners': False,
|
'disable_rt_listeners': False,
|
||||||
'sync': False,
|
'sync': False,
|
||||||
|
|||||||
@ -48,26 +48,18 @@ class IRemote(abc.ABC):
|
|||||||
def apply(self, data):
|
def apply(self, data):
|
||||||
"""Sets all parameters of a dict for the channel."""
|
"""Sets all parameters of a dict for the channel."""
|
||||||
|
|
||||||
script = ''
|
|
||||||
|
|
||||||
def fget(attr, val):
|
def fget(attr, val):
|
||||||
if attr == 'mode':
|
if attr == 'mode':
|
||||||
return (f'mode.{val}', 1)
|
return (getattr(self, attr), val, 1)
|
||||||
elif attr == 'knob':
|
return (self, attr, val)
|
||||||
return ('', val)
|
|
||||||
return (attr, val)
|
|
||||||
|
|
||||||
for attr, val in data.items():
|
for attr, val in data.items():
|
||||||
if not isinstance(val, dict):
|
if not isinstance(val, dict):
|
||||||
if attr in dir(self): # avoid calling getattr (with hasattr)
|
if attr in dir(self): # avoid calling getattr (with hasattr)
|
||||||
attr, val = fget(attr, val)
|
target, attr, val = fget(attr, val)
|
||||||
if isinstance(val, bool):
|
setattr(target, attr, val)
|
||||||
val = 1 if val else 0
|
else:
|
||||||
|
self.logger.error(f'invalid attribute {attr} for {self}')
|
||||||
self._remote.cache[self._cmd(attr)] = val
|
|
||||||
script += f'{self._cmd(attr)}={val};'
|
|
||||||
else:
|
else:
|
||||||
target = getattr(self, attr)
|
target = getattr(self, attr)
|
||||||
target.apply(val)
|
target.apply(val)
|
||||||
|
|
||||||
self._remote.sendtext(script)
|
|
||||||
|
|||||||
@ -2,6 +2,7 @@ import struct
|
|||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
|
|
||||||
from vban_cmd.enums import NBS
|
from vban_cmd.enums import NBS
|
||||||
|
from vban_cmd.error import VBANCMDPacketError
|
||||||
from vban_cmd.kinds import KindMapClass
|
from vban_cmd.kinds import KindMapClass
|
||||||
|
|
||||||
from .enums import ServiceTypes, StreamTypes, SubProtocols
|
from .enums import ServiceTypes, StreamTypes, SubProtocols
|
||||||
@ -10,6 +11,15 @@ PINGPONG_PACKET_SIZE = 704 # Size of the PING/PONG header + payload in bytes
|
|||||||
MAX_PACKET_SIZE = 1436
|
MAX_PACKET_SIZE = 1436
|
||||||
HEADER_SIZE = 4 + 1 + 1 + 1 + 1 + 16
|
HEADER_SIZE = 4 + 1 + 1 + 1 + 1 + 16
|
||||||
|
|
||||||
|
STREAMNAME_MAX_LENGTH = 16
|
||||||
|
# fmt: off
|
||||||
|
BPS_OPTS = [
|
||||||
|
0, 110, 150, 300, 600, 1200, 2400, 4800, 9600, 14400, 19200, 31250,
|
||||||
|
38400, 57600, 115200, 128000, 230400, 250000, 256000, 460800, 921600,
|
||||||
|
1000000, 1500000, 2000000, 3000000,
|
||||||
|
]
|
||||||
|
# fmt: on
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class VbanPingHeader:
|
class VbanPingHeader:
|
||||||
@ -28,7 +38,9 @@ class VbanPingHeader:
|
|||||||
|
|
||||||
@property
|
@property
|
||||||
def streamname(self) -> bytes:
|
def streamname(self) -> bytes:
|
||||||
return self.name.encode('ascii')[:16].ljust(16, b'\x00')
|
return self.name.encode('ascii')[:STREAMNAME_MAX_LENGTH].ljust(
|
||||||
|
STREAMNAME_MAX_LENGTH, b'\x00'
|
||||||
|
)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def to_bytes(cls, framecounter: int = 0) -> bytes:
|
def to_bytes(cls, framecounter: int = 0) -> bytes:
|
||||||
@ -64,7 +76,9 @@ class VbanPongHeader:
|
|||||||
|
|
||||||
@property
|
@property
|
||||||
def streamname(self) -> bytes:
|
def streamname(self) -> bytes:
|
||||||
return self.name.encode('ascii')[:16].ljust(16, b'\x00')
|
return self.name.encode('ascii')[:STREAMNAME_MAX_LENGTH].ljust(
|
||||||
|
STREAMNAME_MAX_LENGTH, b'\x00'
|
||||||
|
)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def from_bytes(cls, data: bytes):
|
def from_bytes(cls, data: bytes):
|
||||||
@ -74,7 +88,11 @@ class VbanPongHeader:
|
|||||||
# PONG responses use the same service type as PING (0x00)
|
# PONG responses use the same service type as PING (0x00)
|
||||||
# and are identified by having payload data
|
# and are identified by having payload data
|
||||||
if parsed['format_nbc'] != ServiceTypes.PONG.value:
|
if parsed['format_nbc'] != ServiceTypes.PONG.value:
|
||||||
raise ValueError(f'Not a PONG response packet: {parsed["format_nbc"]:02x}')
|
raise VBANCMDPacketError(
|
||||||
|
f'Not a PONG response packet: {parsed["format_nbc"]:02x}',
|
||||||
|
protocol=SubProtocols(parsed['format_sr'] & SubProtocols.MASK.value),
|
||||||
|
type_=ServiceTypes(parsed['format_nbc']),
|
||||||
|
)
|
||||||
|
|
||||||
return cls(**parsed)
|
return cls(**parsed)
|
||||||
|
|
||||||
@ -133,7 +151,7 @@ class VbanRTPacket:
|
|||||||
class VbanRTSubscribeHeader:
|
class VbanRTSubscribeHeader:
|
||||||
"""Represents the header of an RT subscription packet"""
|
"""Represents the header of an RT subscription packet"""
|
||||||
|
|
||||||
nbs: NBS = NBS.zero
|
_nbs: NBS = NBS.zero
|
||||||
name: str = 'Register-RTP'
|
name: str = 'Register-RTP'
|
||||||
timeout: int = 15
|
timeout: int = 15
|
||||||
|
|
||||||
@ -142,36 +160,38 @@ class VbanRTSubscribeHeader:
|
|||||||
return b'VBAN'
|
return b'VBAN'
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def format_sr(self) -> bytes:
|
def sr(self) -> int:
|
||||||
return SubProtocols.SERVICE.value.to_bytes(1, 'little')
|
return SubProtocols.SERVICE.value
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def format_nbs(self) -> bytes:
|
def nbs(self) -> int:
|
||||||
return (self.nbs.value & 0xFF).to_bytes(1, 'little')
|
return self._nbs.value & 0xFF
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def format_nbc(self) -> bytes:
|
def nbc(self) -> int:
|
||||||
return ServiceTypes.RTPACKETREGISTER.value.to_bytes(1, 'little')
|
return ServiceTypes.RTPACKETREGISTER.value
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def format_bit(self) -> bytes:
|
def bit(self) -> int:
|
||||||
return (self.timeout & 0xFF).to_bytes(1, 'little')
|
return self.timeout & 0xFF
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def streamname(self) -> bytes:
|
def streamname(self) -> bytes:
|
||||||
return self.name.encode('ascii') + bytes(16 - len(self.name))
|
return self.name.encode()[:STREAMNAME_MAX_LENGTH].ljust(
|
||||||
|
STREAMNAME_MAX_LENGTH, b'\x00'
|
||||||
|
)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def to_bytes(cls, nbs: NBS, framecounter: int) -> bytes:
|
def to_bytes(cls, nbs: NBS, framecounter: int) -> bytes:
|
||||||
header = cls(nbs=nbs)
|
header = cls(_nbs=nbs)
|
||||||
|
|
||||||
return struct.pack(
|
return struct.pack(
|
||||||
'<4s4B16sI',
|
'<4s4B16sI',
|
||||||
header.vban,
|
header.vban,
|
||||||
header.format_sr[0],
|
header.sr,
|
||||||
header.format_nbs[0],
|
header.nbs,
|
||||||
header.format_nbc[0],
|
header.nbc,
|
||||||
header.format_bit[0],
|
header.bit,
|
||||||
header.streamname,
|
header.streamname,
|
||||||
framecounter,
|
framecounter,
|
||||||
)
|
)
|
||||||
@ -182,59 +202,64 @@ class VbanRTRequestHeader:
|
|||||||
"""Represents the header of an RT request packet"""
|
"""Represents the header of an RT request packet"""
|
||||||
|
|
||||||
name: str
|
name: str
|
||||||
bps_index: int
|
bps: int
|
||||||
channel: int
|
channel: int
|
||||||
framecounter: int = 0
|
framecounter: int = 0
|
||||||
|
|
||||||
|
def __post_init__(self):
|
||||||
|
if self.bps not in BPS_OPTS:
|
||||||
|
raise ValueError(
|
||||||
|
f'Invalid bps value: {self.bps}. Must be one of {BPS_OPTS}'
|
||||||
|
)
|
||||||
|
self.bps_index = BPS_OPTS.index(self.bps)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def vban(self) -> bytes:
|
def vban(self) -> bytes:
|
||||||
return b'VBAN'
|
return b'VBAN'
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def sr(self) -> bytes:
|
def sr(self) -> int:
|
||||||
return (self.bps_index | SubProtocols.TEXT.value).to_bytes(1, 'little')
|
return self.bps_index | SubProtocols.TEXT.value
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def nbs(self) -> bytes:
|
def nbs(self) -> int:
|
||||||
return (0).to_bytes(1, 'little')
|
return 0
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def nbc(self) -> bytes:
|
def nbc(self) -> int:
|
||||||
return (self.channel).to_bytes(1, 'little')
|
return self.channel
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def bit(self) -> bytes:
|
def bit(self) -> int:
|
||||||
return (StreamTypes.UTF8.value).to_bytes(1, 'little')
|
return StreamTypes.UTF8.value
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def streamname(self) -> bytes:
|
def streamname(self) -> bytes:
|
||||||
return self.name.encode()[:16].ljust(16, b'\x00')
|
return self.name.encode()[:STREAMNAME_MAX_LENGTH].ljust(
|
||||||
|
STREAMNAME_MAX_LENGTH, b'\x00'
|
||||||
|
)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def to_bytes(
|
def to_bytes(cls, name: str, bps: int, channel: int, framecounter: int) -> bytes:
|
||||||
cls, name: str, bps_index: int, channel: int, framecounter: int
|
header = cls(name=name, bps=bps, channel=channel, framecounter=framecounter)
|
||||||
) -> bytes:
|
|
||||||
header = cls(
|
|
||||||
name=name, bps_index=bps_index, channel=channel, framecounter=framecounter
|
|
||||||
)
|
|
||||||
|
|
||||||
return struct.pack(
|
return struct.pack(
|
||||||
'<4s4B16sI',
|
'<4s4B16sI',
|
||||||
header.vban,
|
header.vban,
|
||||||
header.sr[0],
|
header.sr,
|
||||||
header.nbs[0],
|
header.nbs,
|
||||||
header.nbc[0],
|
header.nbc,
|
||||||
header.bit[0],
|
header.bit,
|
||||||
header.streamname,
|
header.streamname,
|
||||||
header.framecounter,
|
header.framecounter,
|
||||||
)
|
)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def encode_with_payload(
|
def encode_with_payload(
|
||||||
cls, name: str, bps_index: int, channel: int, framecounter: int, payload: str
|
cls, name: str, bps: int, channel: int, framecounter: int, payload: str
|
||||||
) -> bytes:
|
) -> bytes:
|
||||||
"""Creates the complete packet with header and payload."""
|
"""Creates the complete packet with header and payload."""
|
||||||
return cls.to_bytes(name, bps_index, channel, framecounter) + payload.encode()
|
return cls.to_bytes(name, bps, channel, framecounter) + payload.encode()
|
||||||
|
|
||||||
|
|
||||||
def _parse_vban_service_header(data: bytes) -> dict:
|
def _parse_vban_service_header(data: bytes) -> dict:
|
||||||
@ -253,7 +278,10 @@ def _parse_vban_service_header(data: bytes) -> dict:
|
|||||||
# Verify this is a service protocol packet
|
# Verify this is a service protocol packet
|
||||||
protocol = format_sr & SubProtocols.MASK.value
|
protocol = format_sr & SubProtocols.MASK.value
|
||||||
if protocol != SubProtocols.SERVICE.value:
|
if protocol != SubProtocols.SERVICE.value:
|
||||||
raise ValueError(f'Not a service protocol packet: {protocol:02x}')
|
raise VBANCMDPacketError(
|
||||||
|
f'Invalid protocol in service header: {protocol:02x}',
|
||||||
|
protocol=SubProtocols(protocol),
|
||||||
|
)
|
||||||
|
|
||||||
# Extract stream name and frame counter
|
# Extract stream name and frame counter
|
||||||
name = data[8:24].rstrip(b'\x00').decode('utf-8', errors='ignore')
|
name = data[8:24].rstrip(b'\x00').decode('utf-8', errors='ignore')
|
||||||
@ -286,7 +314,9 @@ class VbanRTResponseHeader:
|
|||||||
|
|
||||||
@property
|
@property
|
||||||
def streamname(self) -> bytes:
|
def streamname(self) -> bytes:
|
||||||
return self.name.encode('ascii') + bytes(16 - len(self.name))
|
return self.name.encode()[:STREAMNAME_MAX_LENGTH].ljust(
|
||||||
|
STREAMNAME_MAX_LENGTH, b'\x00'
|
||||||
|
)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def from_bytes(cls, data: bytes):
|
def from_bytes(cls, data: bytes):
|
||||||
@ -295,8 +325,10 @@ class VbanRTResponseHeader:
|
|||||||
|
|
||||||
# Validate this is an RTPacket response
|
# Validate this is an RTPacket response
|
||||||
if parsed['format_nbc'] != ServiceTypes.RTPACKET.value:
|
if parsed['format_nbc'] != ServiceTypes.RTPACKET.value:
|
||||||
raise ValueError(
|
raise VBANCMDPacketError(
|
||||||
f'Not an RTPacket response packet: {parsed["format_nbc"]:02x}'
|
f'Not an RTPacket response packet: {parsed["format_nbc"]:02x}',
|
||||||
|
protocol=SubProtocols(parsed['format_sr'] & SubProtocols.MASK.value),
|
||||||
|
type_=ServiceTypes(parsed['format_nbc']),
|
||||||
)
|
)
|
||||||
|
|
||||||
return cls(**parsed)
|
return cls(**parsed)
|
||||||
@ -319,16 +351,29 @@ class VbanMatrixResponseHeader:
|
|||||||
|
|
||||||
@property
|
@property
|
||||||
def streamname(self) -> bytes:
|
def streamname(self) -> bytes:
|
||||||
return self.name.encode('ascii')[:16].ljust(16, b'\x00')
|
return self.name.encode()[:STREAMNAME_MAX_LENGTH].ljust(
|
||||||
|
STREAMNAME_MAX_LENGTH, b'\x00'
|
||||||
|
)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def from_bytes(cls, data: bytes):
|
def from_bytes(cls, data: bytes):
|
||||||
"""Parse a matrix response packet from bytes."""
|
"""Parse a matrix response packet from bytes."""
|
||||||
parsed = _parse_vban_service_header(data)
|
parsed = _parse_vban_service_header(data)
|
||||||
|
|
||||||
# Validate this is a service reply packet
|
# Validate this is a service reply packet (dual encoding scheme)
|
||||||
if parsed['format_nbs'] != ServiceTypes.FNCT_REPLY.value:
|
if parsed['format_nbs'] != ServiceTypes.FNCT_REPLY.value:
|
||||||
raise ValueError(f'Not a service reply packet: {parsed["format_nbs"]:02x}')
|
raise VBANCMDPacketError(
|
||||||
|
f'Not a service reply packet: {parsed["format_nbs"]:02x}',
|
||||||
|
protocol=SubProtocols(parsed['format_sr'] & SubProtocols.MASK.value),
|
||||||
|
type_=ServiceTypes(parsed['format_nbs']),
|
||||||
|
)
|
||||||
|
|
||||||
|
if parsed['format_nbc'] != ServiceTypes.REQUESTREPLY.value:
|
||||||
|
raise VBANCMDPacketError(
|
||||||
|
f'Not a request reply packet: {parsed["format_nbc"]:02x}',
|
||||||
|
protocol=SubProtocols(parsed['format_sr'] & SubProtocols.MASK.value),
|
||||||
|
type_=ServiceTypes(parsed['format_nbc']),
|
||||||
|
)
|
||||||
|
|
||||||
return cls(**parsed)
|
return cls(**parsed)
|
||||||
|
|
||||||
|
|||||||
@ -5,12 +5,12 @@ from typing import Iterator
|
|||||||
from .error import VBANCMDConnectionError
|
from .error import VBANCMDConnectionError
|
||||||
|
|
||||||
|
|
||||||
def ratelimit(func):
|
def script_ratelimit(func):
|
||||||
"""ratelimit decorator for {VbanCmd}.sendtext, to prevent flooding the network with script requests."""
|
"""script_ratelimit decorator for {VbanCmd}.sendtext, to prevent flooding the network with script requests."""
|
||||||
|
|
||||||
def wrapper(*args, **kwargs):
|
def wrapper(*args, **kwargs):
|
||||||
self, *rem = args
|
self, *rem = args
|
||||||
if self.script_ratelimit > 0:
|
if self.script_ratelimit:
|
||||||
now = time.time()
|
now = time.time()
|
||||||
elapsed = now - self._last_script_request_time
|
elapsed = now - self._last_script_request_time
|
||||||
if elapsed < self.script_ratelimit:
|
if elapsed < self.script_ratelimit:
|
||||||
|
|||||||
@ -17,7 +17,7 @@ from .packet.headers import (
|
|||||||
)
|
)
|
||||||
from .packet.ping0 import VbanPing0Payload, VbanServerType
|
from .packet.ping0 import VbanPing0Payload, VbanServerType
|
||||||
from .subject import Subject
|
from .subject import Subject
|
||||||
from .util import bump_framecounter, deep_merge, pong_timeout, ratelimit
|
from .util import bump_framecounter, deep_merge, pong_timeout, script_ratelimit
|
||||||
from .worker import Producer, Subscriber, Updater
|
from .worker import Producer, Subscriber, Updater
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
@ -27,13 +27,6 @@ class VbanCmd(abc.ABC):
|
|||||||
"""Abstract Base Class for Voicemeeter VBAN Command Interfaces"""
|
"""Abstract Base Class for Voicemeeter VBAN Command Interfaces"""
|
||||||
|
|
||||||
DELAY = 0.001
|
DELAY = 0.001
|
||||||
# fmt: off
|
|
||||||
BPS_OPTS = [
|
|
||||||
0, 110, 150, 300, 600, 1200, 2400, 4800, 9600, 14400, 19200, 31250,
|
|
||||||
38400, 57600, 115200, 128000, 230400, 250000, 256000, 460800, 921600,
|
|
||||||
1000000, 1500000, 2000000, 3000000,
|
|
||||||
]
|
|
||||||
# fmt: on
|
|
||||||
|
|
||||||
def __init__(self, **kwargs):
|
def __init__(self, **kwargs):
|
||||||
self.logger = logger.getChild(self.__class__.__name__)
|
self.logger = logger.getChild(self.__class__.__name__)
|
||||||
@ -203,7 +196,7 @@ class VbanCmd(abc.ABC):
|
|||||||
self.sock.sendto(
|
self.sock.sendto(
|
||||||
VbanRTRequestHeader.encode_with_payload(
|
VbanRTRequestHeader.encode_with_payload(
|
||||||
name=self.streamname,
|
name=self.streamname,
|
||||||
bps_index=self.BPS_OPTS.index(self.bps),
|
bps=self.bps,
|
||||||
channel=self.channel,
|
channel=self.channel,
|
||||||
framecounter=self._get_next_framecounter(),
|
framecounter=self._get_next_framecounter(),
|
||||||
payload=payload,
|
payload=payload,
|
||||||
@ -216,7 +209,7 @@ class VbanCmd(abc.ABC):
|
|||||||
self._send_request(f'{cmd}={val};')
|
self._send_request(f'{cmd}={val};')
|
||||||
self.cache[cmd] = val
|
self.cache[cmd] = val
|
||||||
|
|
||||||
@ratelimit
|
@script_ratelimit
|
||||||
def sendtext(self, script) -> str | None:
|
def sendtext(self, script) -> str | None:
|
||||||
"""Sends a multiple parameter string over a network."""
|
"""Sends a multiple parameter string over a network."""
|
||||||
self._send_request(script)
|
self._send_request(script)
|
||||||
|
|||||||
@ -3,7 +3,8 @@ import threading
|
|||||||
import time
|
import time
|
||||||
|
|
||||||
from .enums import NBS
|
from .enums import NBS
|
||||||
from .error import VBANCMDConnectionError
|
from .error import VBANCMDConnectionError, VBANCMDPacketError
|
||||||
|
from .packet.enums import SubProtocols
|
||||||
from .packet.headers import (
|
from .packet.headers import (
|
||||||
HEADER_SIZE,
|
HEADER_SIZE,
|
||||||
VbanRTPacket,
|
VbanRTPacket,
|
||||||
@ -81,7 +82,12 @@ class Producer(threading.Thread):
|
|||||||
|
|
||||||
try:
|
try:
|
||||||
header = VbanRTResponseHeader.from_bytes(data[:HEADER_SIZE])
|
header = VbanRTResponseHeader.from_bytes(data[:HEADER_SIZE])
|
||||||
except ValueError as e:
|
except VBANCMDPacketError as e:
|
||||||
|
match e.protocol:
|
||||||
|
case SubProtocols.SERVICE:
|
||||||
|
# Silently ignore periodic SERVICE packets unrelated to vban-cmd
|
||||||
|
pass
|
||||||
|
case _:
|
||||||
self.logger.debug(f'Error parsing response packet: {e}')
|
self.logger.debug(f'Error parsing response packet: {e}')
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user