Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Improvements to 'group' objects #231

Merged
merged 2 commits into from
Feb 11, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion src/aiovantage/_config_client/client.py
Original file line number Diff line number Diff line change
Expand Up @@ -131,7 +131,7 @@ async def raw_request(self, request: str, separator: str) -> str:

return response

async def rpc_call(
async def rpc(
self,
interface_cls: type[Interface],
method_cls: type[Method[Call, Return]],
Expand Down
53 changes: 15 additions & 38 deletions src/aiovantage/_config_client/interfaces/configuration.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
"""IConfiguration interfaces."""

from collections.abc import AsyncIterator
from contextlib import suppress
from dataclasses import dataclass, field
Expand All @@ -13,13 +11,16 @@


@dataclass
class OpenFilter:
"""IConfiguration.OpenFilter method definition."""
class WrappedObject:
# Wildcard type that can be used to represent any object.
vid: int = field(metadata={"name": "VID", "type": "Attribute"})
obj: object = field(metadata={"type": "Wildcard"})


@dataclass
class OpenFilter:
@dataclass
class Params:
"""Method parameters."""

object_types: list[str] | None = field(
default=None,
metadata={"wrapper": "Objects", "name": "ObjectType", "type": "Element"},
Expand All @@ -32,64 +33,40 @@ class Params:

@dataclass
class GetFilterResults:
"""IConfiguration.GetFilterResults method definition."""

@dataclass
class Params:
"""Method parameters."""

h_filter: int = field(metadata={"name": "hFilter"})
count: int = 50
whole_object: bool = True

@dataclass
class Object:
"""Wildcard type that can be used to represent any object."""

vid: int = field(metadata={"name": "VID", "type": "Attribute"})
obj: object = field(metadata={"type": "Wildcard"})

call: Params | None = field(default=None, metadata={"name": "call"})
result: list[Object] | None = field(
result: list[WrappedObject] | None = field(
default_factory=list,
metadata={"wrapper": "return", "name": "Object", "type": "Element"},
)


@dataclass
class CloseFilter:
"""IConfiguration.CloseFilter method definition."""

call: int | None = field(default=None, metadata={"name": "call"})
result: bool | None = field(default=None, metadata={"name": "return"})


@dataclass
class GetObject:
"""IConfiguration.GetObject method definition."""

@dataclass
class Object:
"""Wildcard type that can be used to represent any object."""

vid: int = field(metadata={"name": "VID", "type": "Attribute"})
obj: object = field(metadata={"type": "Wildcard"})

call: list[int] | None = field(
default_factory=list,
metadata={"wrapper": "call", "name": "VID", "type": "Element"},
)

result: list[Object] | None = field(
result: list[WrappedObject] | None = field(
default_factory=list,
metadata={"wrapper": "return", "name": "Object", "type": "Element"},
)


@dataclass(kw_only=True)
class IConfiguration:
"""IConfiguration interface."""

open_filter: OpenFilter | None = None
get_filter_results: GetFilterResults | None = None
close_filter: CloseFilter | None = None
Expand All @@ -113,7 +90,7 @@ async def open_filter(
Returns:
The handle of the opened filter
"""
return await client.rpc_call(
return await client.rpc(
IConfiguration,
OpenFilter,
OpenFilter.Params(object_types=list(object_types), xpath=xpath),
Expand All @@ -122,7 +99,7 @@ async def open_filter(
@staticmethod
async def get_filter_results(
client: ConfigClient, h_filter: int, count: int = 50, whole_object: bool = True
) -> list[GetFilterResults.Object]:
) -> list[WrappedObject]:
"""Get results from a filter handle previously opened with open_filter.

Args:
Expand All @@ -134,7 +111,7 @@ async def get_filter_results(
Returns:
A list of Vantage objects
"""
return await client.rpc_call(
return await client.rpc(
IConfiguration, GetFilterResults, GetFilterResults.Params(h_filter)
)

Expand All @@ -149,10 +126,10 @@ async def close_filter(client: ConfigClient, h_filter: int) -> bool:
Returns:
True if the filter was closed successfully, False otherwise
"""
return await client.rpc_call(IConfiguration, CloseFilter, h_filter)
return await client.rpc(IConfiguration, CloseFilter, h_filter)

@staticmethod
async def get_object(client: ConfigClient, *vids: int) -> list[GetObject.Object]:
async def get_object(client: ConfigClient, *vids: int) -> list[WrappedObject]:
"""Get one or more Vantage objects by their VIDs.

Args:
Expand All @@ -162,7 +139,7 @@ async def get_object(client: ConfigClient, *vids: int) -> list[GetObject.Object]
Returns:
A list of Vantage objects
"""
return await client.rpc_call(IConfiguration, GetObject, list(vids))
return await client.rpc(IConfiguration, GetObject, list(vids))

# Convenience functions, not part of the interface
@overload
Expand Down
28 changes: 4 additions & 24 deletions src/aiovantage/_config_client/interfaces/introspection.py
Original file line number Diff line number Diff line change
@@ -1,18 +1,12 @@
"""IIntrospection interface."""

from dataclasses import dataclass, field

from ..client import ConfigClient


@dataclass
class GetInterfaces:
"""IIntrospection.GetInterfaces method definition."""

@dataclass
class Interface:
"""Object interface definition."""

name: str
version: str
iid: int = field(metadata={"name": "IID"})
Expand All @@ -26,12 +20,8 @@ class Interface:

@dataclass
class GetSysInfo:
"""IIntrospection.GetSysInfo method definition."""

@dataclass
class SysInfo:
"""SysInfo class."""

master_number: int
serial_number: int

Expand All @@ -43,12 +33,8 @@ class SysInfo:

@dataclass
class GetTypes:
"""IIntrospection.GetTypes method definition."""

@dataclass
class Type:
"""Object type definition."""

name: str
version: str

Expand All @@ -60,12 +46,8 @@ class Type:

@dataclass
class GetVersion:
"""IIntrospection.GetVersion method definition."""

@dataclass
class Version:
"""Method return value."""

kernel: str | None = field(default=None, metadata={"name": "kernel"})
rootfs: str | None = field(default=None, metadata={"name": "rootfs"})
app: str | None = field(default=None, metadata={"name": "app"})
Expand All @@ -76,8 +58,6 @@ class Version:

@dataclass(kw_only=True)
class IIntrospection:
"""IIntrospection interface."""

get_interfaces: GetInterfaces | None = None
get_sys_info: GetSysInfo | None = None
get_types: GetTypes | None = None
Expand All @@ -97,7 +77,7 @@ async def get_interfaces(client: ConfigClient) -> list[GetInterfaces.Interface]:
Returns:
A list of interfaces.
"""
return await client.rpc_call(IIntrospection, GetInterfaces)
return await client.rpc(IIntrospection, GetInterfaces)

@staticmethod
async def get_sys_info(client: ConfigClient) -> GetSysInfo.SysInfo:
Expand All @@ -109,7 +89,7 @@ async def get_sys_info(client: ConfigClient) -> GetSysInfo.SysInfo:
Returns:
A system information object.
"""
return await client.rpc_call(IIntrospection, GetSysInfo)
return await client.rpc(IIntrospection, GetSysInfo)

@staticmethod
async def get_types(client: ConfigClient) -> list[GetTypes.Type]:
Expand All @@ -121,7 +101,7 @@ async def get_types(client: ConfigClient) -> list[GetTypes.Type]:
Returns:
A list of object types.
"""
return await client.rpc_call(IIntrospection, GetTypes)
return await client.rpc(IIntrospection, GetTypes)

@staticmethod
async def get_version(client: ConfigClient) -> GetVersion.Version:
Expand All @@ -133,4 +113,4 @@ async def get_version(client: ConfigClient) -> GetVersion.Version:
Returns:
A version information object.
"""
return await client.rpc_call(IIntrospection, GetVersion)
return await client.rpc(IIntrospection, GetVersion)
10 changes: 1 addition & 9 deletions src/aiovantage/_config_client/interfaces/login.py
Original file line number Diff line number Diff line change
@@ -1,18 +1,12 @@
"""ILogin.Login method definition."""

from dataclasses import dataclass, field

from ..client import ConfigClient


@dataclass
class Login:
"""ILogin.Login method definition."""

@dataclass
class Params:
"""Method parameters."""

user: str
password: str

Expand All @@ -22,8 +16,6 @@ class Params:

@dataclass(kw_only=True)
class ILogin:
"""ILogin interface."""

login: Login | None = None


Expand All @@ -42,6 +34,6 @@ async def login(client: ConfigClient, user: str, password: str) -> bool:
Returns:
True if the login was successful, False otherwise
"""
return await client.rpc_call(
return await client.rpc(
ILogin, Login, Login.Params(user=user, password=password)
)
7 changes: 4 additions & 3 deletions src/aiovantage/_controllers/blind_groups.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,11 @@

from .base import BaseController

BlindGroupTypes = BlindGroup | SomfyRS485GroupChild | SomfyURTSI2GroupChild
"""Types managed by the blind groups controller."""

class BlindGroupsController(
BaseController[BlindGroup | SomfyRS485GroupChild | SomfyURTSI2GroupChild]
):

class BlindGroupsController(BaseController[BlindGroupTypes]):
"""Blind groups controller."""

vantage_types = (
Expand Down
6 changes: 6 additions & 0 deletions src/aiovantage/_controllers/blinds.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
from aiovantage.objects import (
BlindGroup,
QISBlind,
QubeBlind,
RelayBlind,
Expand All @@ -7,6 +8,7 @@
)

from .base import BaseController
from .query import QuerySet

BlindTypes = (
QISBlind | QubeBlind | RelayBlind | SomfyRS485ShadeChild | SomfyURTSI2ShadeChild
Expand All @@ -24,3 +26,7 @@ class BlindsController(BaseController[BlindTypes]):
"Somfy.RS-485_Shade_CHILD",
"Somfy.URTSI_2_Shade_CHILD",
)

def in_blind_group(self, blind_group: BlindGroup) -> QuerySet[BlindTypes]:
"""Return a queryset of all loads in the given blind group."""
return self.filter(lambda load: load.vid in blind_group.blind_table)
9 changes: 1 addition & 8 deletions src/aiovantage/_controllers/load_groups.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,4 @@
from aiovantage._controllers.query import QuerySet
from aiovantage.objects import Load, LoadGroup
from aiovantage.objects import LoadGroup

from .base import BaseController

Expand All @@ -8,9 +7,3 @@ class LoadGroupsController(BaseController[LoadGroup]):
"""Load groups controller."""

vantage_types = ("LoadGroup",)

def loads(self, vid: int) -> QuerySet[Load]:
"""Return a queryset of all loads in this load group."""
load_group = self[vid]

return self._vantage.loads.filter(lambda load: load.id in load_group.load_table)
8 changes: 6 additions & 2 deletions src/aiovantage/_controllers/loads.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
from aiovantage._controllers.query import QuerySet
from aiovantage.objects import Load
from aiovantage.objects import Load, LoadGroup

from .base import BaseController
from .query import QuerySet


class LoadsController(BaseController[Load]):
Expand Down Expand Up @@ -33,3 +33,7 @@ def motors(self) -> QuerySet[Load]:
def lights(self) -> QuerySet[Load]:
"""Return a queryset of all loads that are lights."""
return self.filter(lambda load: load.is_light)

def in_load_group(self, load_group: LoadGroup) -> QuerySet[Load]:
"""Return a queryset of all loads in the given load group."""
return self.filter(lambda load: load.vid in load_group.load_table)
2 changes: 1 addition & 1 deletion src/aiovantage/_controllers/rgb_loads.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
from aiovantage._controllers.query import QuerySet
from aiovantage.objects import VantageDDGColorLoad, VantageDGColorLoad

from .base import BaseController
from .query import QuerySet

RGBLoadTypes = VantageDDGColorLoad | VantageDGColorLoad
"""Types managed by the RGB loads controller."""
Expand Down
Loading