python-linkplay 0.0.2__py3-none-any.whl → 0.0.4__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
linkplay/__main__.py CHANGED
@@ -1,14 +1,20 @@
1
1
  import asyncio
2
2
  import aiohttp
3
3
 
4
- from linkplay.discovery import discover_linkplay_bridges, discover_multirooms
4
+ from linkplay.controller import LinkPlayController
5
5
 
6
6
 
7
7
  async def main():
8
8
  async with aiohttp.ClientSession() as session:
9
- bridges = await discover_linkplay_bridges(session)
10
- multirooms = await discover_multirooms(bridges)
11
- return bridges, multirooms
9
+ controller = LinkPlayController(session)
10
+
11
+ await controller.discover_bridges()
12
+ for bridge in controller.bridges:
13
+ print(bridge)
14
+
15
+ await controller.discover_multirooms()
16
+ for multiroom in controller.multirooms:
17
+ print(multiroom.followers)
12
18
 
13
19
  if __name__ == "__main__":
14
20
  asyncio.run(main())
linkplay/__version__.py CHANGED
@@ -1 +1 @@
1
- __version__ = '0.0.2'
1
+ __version__ = '0.0.4'
linkplay/bridge.py CHANGED
@@ -1,5 +1,5 @@
1
1
  from __future__ import annotations
2
-
2
+ from typing import Any
3
3
  from aiohttp import ClientSession
4
4
 
5
5
  from linkplay.consts import (
@@ -15,8 +15,10 @@ from linkplay.consts import (
15
15
  InputMode,
16
16
  SpeakerType,
17
17
  PlayingMode,
18
- INPUT_MODE_MAP
18
+ INPUT_MODE_MAP,
19
+ MultiroomAttribute
19
20
  )
21
+
20
22
  from linkplay.utils import session_call_api_json, session_call_api_ok, decode_hexstr
21
23
 
22
24
 
@@ -243,9 +245,21 @@ class LinkPlayMultiroom():
243
245
  leader: LinkPlayBridge
244
246
  followers: list[LinkPlayBridge]
245
247
 
246
- def __init__(self, leader: LinkPlayBridge, followers: list[LinkPlayBridge]):
248
+ def __init__(self, leader: LinkPlayBridge):
247
249
  self.leader = leader
248
- self.followers = followers
250
+ self.followers = []
251
+
252
+ async def update_status(self, bridges: list[LinkPlayBridge]) -> None:
253
+ """Updates the multiroom status."""
254
+ properties: dict[Any, Any] = await self.leader.json_request(LinkPlayCommand.MULTIROOM_LIST)
255
+
256
+ self.followers = []
257
+ if int(properties[MultiroomAttribute.NUM_FOLLOWERS]) == 0:
258
+ return
259
+
260
+ follower_uuids = [follower[MultiroomAttribute.UUID] for follower in properties[MultiroomAttribute.FOLLOWER_LIST]]
261
+ new_followers = [bridge for bridge in bridges if bridge.device.uuid in follower_uuids]
262
+ self.followers.extend(new_followers)
249
263
 
250
264
  async def ungroup(self) -> None:
251
265
  """Ungroups the multiroom group."""
linkplay/controller.py ADDED
@@ -0,0 +1,54 @@
1
+ from aiohttp import ClientSession
2
+
3
+ from linkplay.bridge import LinkPlayBridge, LinkPlayMultiroom
4
+ from linkplay.discovery import discover_linkplay_bridges
5
+
6
+
7
+ class LinkPlayController():
8
+ """Represents a LinkPlay controller to manage the devices and multirooms."""
9
+
10
+ session: ClientSession
11
+ bridges: list[LinkPlayBridge]
12
+ multirooms: list[LinkPlayMultiroom]
13
+
14
+ def __init__(self, session: ClientSession):
15
+ self.session = session
16
+ self.bridges = []
17
+ self.multirooms = []
18
+
19
+ async def discover_bridges(self) -> None:
20
+ """Attempts to discover LinkPlay devices on the local network."""
21
+
22
+ # Discover new bridges
23
+ discovered_bridges = await discover_linkplay_bridges(self.session)
24
+ current_bridges = [bridge.device.uuid for bridge in self.bridges]
25
+ new_bridges = [discovered_bridge for discovered_bridge in discovered_bridges if discovered_bridge.device.uuid not in current_bridges]
26
+ self.bridges.extend(new_bridges)
27
+
28
+ async def discover_multirooms(self) -> None:
29
+ """Attempts to discover multirooms on the local network."""
30
+
31
+ # Create new multirooms from new bridges
32
+ new_multirooms = []
33
+ for bridge in self.bridges:
34
+ has_multiroom = any(multiroom for multiroom in self.multirooms if multiroom.leader == bridge)
35
+
36
+ if has_multiroom:
37
+ continue
38
+
39
+ multiroom = LinkPlayMultiroom(bridge)
40
+ await multiroom.update_status(self.bridges)
41
+ if len(multiroom.followers) > 0:
42
+ new_multirooms.append(multiroom)
43
+
44
+ # Update existing multirooms
45
+ for multiroom in self.multirooms:
46
+ await multiroom.update_status(self.bridges)
47
+
48
+ # Remove multirooms if they have no followers
49
+ empty_multirooms = [multiroom for multiroom in self.multirooms if not multiroom.followers]
50
+ for empty_multiroom in empty_multirooms:
51
+ self.multirooms.remove(empty_multiroom)
52
+
53
+ # Add new multirooms
54
+ self.multirooms.extend(new_multirooms)
linkplay/discovery.py CHANGED
@@ -5,7 +5,7 @@ from async_upnp_client.search import async_search
5
5
  from async_upnp_client.utils import CaseInsensitiveDict
6
6
 
7
7
  from linkplay.consts import UPNP_DEVICE_TYPE, LinkPlayCommand, MultiroomAttribute
8
- from linkplay.bridge import LinkPlayBridge, LinkPlayMultiroom
8
+ from linkplay.bridge import LinkPlayBridge
9
9
  from linkplay.exceptions import LinkPlayRequestException
10
10
 
11
11
 
@@ -21,7 +21,7 @@ async def linkplay_factory_bridge(ip_address: str, session: ClientSession) -> Li
21
21
  return bridge
22
22
 
23
23
 
24
- async def discover_linkplay_bridges(session: ClientSession) -> list[LinkPlayBridge]:
24
+ async def discover_linkplay_bridges(session: ClientSession, discovery_through_multiroom: bool = True) -> list[LinkPlayBridge]:
25
25
  """Attempts to discover LinkPlay devices on the local network."""
26
26
  bridges: dict[str, LinkPlayBridge] = {}
27
27
 
@@ -40,34 +40,15 @@ async def discover_linkplay_bridges(session: ClientSession) -> list[LinkPlayBrid
40
40
  )
41
41
 
42
42
  # Discover additional bridges through grouped multirooms
43
- multiroom_discovered_bridges: dict[str, LinkPlayBridge] = {}
44
- for bridge in bridges.values():
45
- for new_bridge in await discover_bridges_through_multiroom(bridge, session):
46
- multiroom_discovered_bridges[new_bridge.device.uuid] = new_bridge
43
+ if discovery_through_multiroom:
44
+ multiroom_discovered_bridges: dict[str, LinkPlayBridge] = {}
45
+ for bridge in bridges.values():
46
+ for new_bridge in await discover_bridges_through_multiroom(bridge, session):
47
+ multiroom_discovered_bridges[new_bridge.device.uuid] = new_bridge
47
48
 
48
- bridges = bridges | multiroom_discovered_bridges
49
- return list(bridges.values())
50
-
51
-
52
- async def discover_multirooms(bridges: list[LinkPlayBridge]) -> list[LinkPlayMultiroom]:
53
- """Discovers multirooms through the list of provided bridges."""
54
- multirooms: list[LinkPlayMultiroom] = []
55
- bridges_dict: dict[str, LinkPlayBridge] = {bridge.device.uuid: bridge for bridge in bridges}
56
-
57
- for bridge in bridges:
58
- properties: dict[Any, Any] = await bridge.json_request(LinkPlayCommand.MULTIROOM_LIST)
49
+ bridges = bridges | multiroom_discovered_bridges
59
50
 
60
- if int(properties[MultiroomAttribute.NUM_FOLLOWERS]) == 0:
61
- continue
62
-
63
- followers: list[LinkPlayBridge] = []
64
- for follower in properties[MultiroomAttribute.FOLLOWER_LIST]:
65
- if follower[MultiroomAttribute.UUID] in bridges_dict:
66
- followers.append(bridges_dict[follower[MultiroomAttribute.UUID]])
67
-
68
- multirooms.append(LinkPlayMultiroom(bridge, followers))
69
-
70
- return multirooms
51
+ return list(bridges.values())
71
52
 
72
53
 
73
54
  async def discover_bridges_through_multiroom(bridge: LinkPlayBridge,
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: python_linkplay
3
- Version: 0.0.2
3
+ Version: 0.0.4
4
4
  Summary: A Python Library for Seamless LinkPlay Device Control
5
5
  Author: Velleman Group nv
6
6
  License: MIT
@@ -0,0 +1,14 @@
1
+ linkplay/__init__.py,sha256=y9ZehEq-KhS3cwn-PUpwVSJGfDUx7e5wf_G6guODcTk,56
2
+ linkplay/__main__.py,sha256=eBEvuF8rDs0vuexX3Mcd5O02GRPQtOOOAVNEWqZhhdg,502
3
+ linkplay/__version__.py,sha256=-Wqu9UyJphp-2_EahEsqQvkcE0VBhm1-N7D6e5rANdY,22
4
+ linkplay/bridge.py,sha256=NX02ZBcbaz7yZv1EeFvQ9CAUh5jVcuSABtpKIY4aj74,10948
5
+ linkplay/consts.py,sha256=KIBGrRQqxd1B4kRO0Vl0e5-UzbMLzJGC_ECohzkSRwQ,7750
6
+ linkplay/controller.py,sha256=f0-fxURcts4j_p5prW-EW5IkAcdsR5Zl8nT1aL_587U,2050
7
+ linkplay/discovery.py,sha256=NIUC3FVPH-FcHvMmojJw0s0w2Q-WCuulGrQA-PDLcsw,2642
8
+ linkplay/exceptions.py,sha256=tWJWHsKVkUEq3Yet1Z739IxcaQT8YamDeSp0tqHde9c,107
9
+ linkplay/utils.py,sha256=E_SjIyeK76ishhwuU24m28y1FDAMEj1QbRNt-aHIMdA,2137
10
+ python_linkplay-0.0.4.dist-info/LICENSE,sha256=bgEtxMyjEHX_4uwaAY3GCFTm234D4AOZ5dM15sk26ms,1073
11
+ python_linkplay-0.0.4.dist-info/METADATA,sha256=j_sNnigaFbClz5uTawxWeggHzuPRxGu_U7xttzRRHKs,2859
12
+ python_linkplay-0.0.4.dist-info/WHEEL,sha256=GJ7t_kWBFywbagK5eo9IoUwLW6oyOeTKmQ-9iHFVNxQ,92
13
+ python_linkplay-0.0.4.dist-info/top_level.txt,sha256=CpSaOVPTzJf5TVIL7MrotSCR34gcIOQy-11l4zGmxxM,9
14
+ python_linkplay-0.0.4.dist-info/RECORD,,
@@ -1,13 +0,0 @@
1
- linkplay/__init__.py,sha256=y9ZehEq-KhS3cwn-PUpwVSJGfDUx7e5wf_G6guODcTk,56
2
- linkplay/__main__.py,sha256=AiwpQKJIV9HrSqj0qU-a8MKE-AUjbfTrQ6st45U78rE,382
3
- linkplay/__version__.py,sha256=miSsqPy2PFJ18TW9kocQJLcv4HFpkfQhcTzgAO-GTVE,22
4
- linkplay/bridge.py,sha256=l5Vvtp2SHF2FYMa67Yf-daRGsErMRXB1GMPzHhwwG6k,10344
5
- linkplay/consts.py,sha256=KIBGrRQqxd1B4kRO0Vl0e5-UzbMLzJGC_ECohzkSRwQ,7750
6
- linkplay/discovery.py,sha256=l4vRkkFYwdD8iqbSUHGZ16yBV8RqylkhBNL2zAqXVP0,3428
7
- linkplay/exceptions.py,sha256=tWJWHsKVkUEq3Yet1Z739IxcaQT8YamDeSp0tqHde9c,107
8
- linkplay/utils.py,sha256=E_SjIyeK76ishhwuU24m28y1FDAMEj1QbRNt-aHIMdA,2137
9
- python_linkplay-0.0.2.dist-info/LICENSE,sha256=bgEtxMyjEHX_4uwaAY3GCFTm234D4AOZ5dM15sk26ms,1073
10
- python_linkplay-0.0.2.dist-info/METADATA,sha256=esFb5Hu0jcGykawjsfEqJk7YpQxU4paUQhwOuCg6eBU,2859
11
- python_linkplay-0.0.2.dist-info/WHEEL,sha256=GJ7t_kWBFywbagK5eo9IoUwLW6oyOeTKmQ-9iHFVNxQ,92
12
- python_linkplay-0.0.2.dist-info/top_level.txt,sha256=CpSaOVPTzJf5TVIL7MrotSCR34gcIOQy-11l4zGmxxM,9
13
- python_linkplay-0.0.2.dist-info/RECORD,,