Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit b4b8259

Browse files
committedNov 23, 2024
Added functionality to get stream destination info (stream url and key)
1 parent a5d7006 commit b4b8259

File tree

4 files changed

+79
-11
lines changed

4 files changed

+79
-11
lines changed
 

‎kick/client.py

+23-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@
1010
from .http import HTTPClient
1111
from .livestream import PartialLivestream
1212
from .message import Message
13-
from .users import ClientUser, PartialUser, User
13+
from .users import ClientUser, PartialUser, User, DestinationInfo
1414
from .utils import MISSING, decorator, setup_logging
1515

1616
if TYPE_CHECKING:
@@ -213,6 +213,28 @@ async def fetch_user(self, name: str, /) -> User:
213213
user = User(data=data, http=self.http)
214214
return user
215215

216+
async def fetch_stream_url_and_key(self) -> DestinationInfo:
217+
"""
218+
|coro|
219+
220+
Fetches your stream URL and stream key from the API.
221+
You must be authenticated to use this endpoint.
222+
223+
Raises
224+
-----------
225+
HTTPException
226+
Fetching Failed
227+
Forbidden
228+
You are not authenticated
229+
230+
Returns
231+
-----------
232+
str
233+
"""
234+
235+
data = await self.http.get_stream_destination_url_and_key()
236+
return DestinationInfo(data=data)
237+
216238
def dispatch(self, event_name: str, *args, **kwargs) -> None:
217239
event_name = f"on_{event_name}"
218240

‎kick/http.py

+24-8
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,12 @@
4343
ReplyOriginalSender,
4444
V1MessageSentPayload,
4545
)
46-
from .types.user import ChatterPayload, ClientUserPayload, UserPayload
46+
from .types.user import (
47+
ChatterPayload,
48+
ClientUserPayload,
49+
UserPayload,
50+
DestinationInfoPayload,
51+
)
4752
from .types.videos import GetVideosPayload
4853

4954
T = TypeVar("T")
@@ -60,10 +65,9 @@
6065
async def json_or_text(response: ClientResponse, /) -> Union[dict[str, Any], str]:
6166
text = await response.text()
6267
try:
63-
try:
64-
return json.loads(text)
65-
except json.JSONDecodeError:
66-
pass
68+
return json.loads(text)
69+
except json.JSONDecodeError:
70+
pass
6771
except KeyError:
6872
pass
6973

@@ -238,9 +242,11 @@ async def request(self, route: Route, **kwargs) -> Any:
238242
try:
239243
res = await self.__session.request(
240244
route.method,
241-
url
242-
if self.whitelisted is True
243-
else f"{self.bypass_host}:{self.bypass_port}/request?url={url}",
245+
(
246+
url
247+
if self.whitelisted is True
248+
else f"{self.bypass_host}:{self.bypass_port}/request?url={url}"
249+
),
244250
headers=headers,
245251
cookies=cookies,
246252
**kwargs,
@@ -488,6 +494,16 @@ def reply_to_message(
488494
def get_me(self) -> Response[ClientUserPayload]:
489495
return self.request(Route.root("GET", "/api/v1/user"))
490496

497+
def get_stream_destination_url_and_key(self) -> Response[DestinationInfoPayload]:
498+
"""Gets the authenticated user's stream URL and key.
499+
500+
Returns
501+
-------
502+
StreamURLKeyPayload
503+
The stream URL and key information containing the publish URL and token
504+
"""
505+
return self.request(Route.root("GET", "/stream/publish_token"))
506+
491507
async def get_asset(self, url: str) -> bytes:
492508
if self.__session is MISSING:
493509
self.__session = ClientSession()

‎kick/types/user.py

+5
Original file line numberDiff line numberDiff line change
@@ -140,3 +140,8 @@ class ClientUserPayload(TypedDict):
140140
streamer_channel: ClientUserStreamerChannelsPayload
141141
roles: list # Unknown
142142
profilepic: str | None
143+
144+
145+
class DestinationInfoPayload(TypedDict):
146+
rtmp_publish_path: str
147+
rtmp_stream_token: str

‎kick/users.py

+27-2
Original file line numberDiff line numberDiff line change
@@ -16,9 +16,34 @@
1616
if TYPE_CHECKING:
1717
from .chatroom import Chatroom
1818
from .http import HTTPClient
19-
from .types.user import ClientUserPayload, InnerUser, UserPayload
19+
from .types.user import ClientUserPayload, InnerUser, UserPayload, DestinationInfoPayload, StreamInfoPayload
20+
21+
__all__ = ("DestinationInfo", "StreamInfo", "Socials", "PartialUser", "User", "ClientUser")
22+
23+
24+
class DestinationInfo(BaseDataclass["DestinationInfoPayload"]):
25+
"""
26+
Information about a user's stream destination
27+
28+
Attributes
29+
-----------
30+
stream_url: str
31+
The URL for streaming
32+
stream_key: str
33+
The stream key
34+
"""
35+
36+
@property
37+
def stream_url(self) -> str:
38+
"""The URL for streaming"""
39+
return self._data["rtmp_publish_path"]
40+
41+
@property
42+
def stream_key(self) -> str:
43+
"""The stream key"""
44+
return self._data["rtmp_stream_token"]
45+
2046

21-
__all__ = ("User", "Socials", "PartialUser", "ClientUser")
2247

2348

2449
class Socials(BaseDataclass["InnerUser | ClientUserPayload"]):

0 commit comments

Comments
 (0)
Please sign in to comment.