typecast-python 0.1.7__py3-none-any.whl → 0.1.9__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.
typecast/async_client.py CHANGED
@@ -196,12 +196,11 @@ class AsyncTypecast:
196
196
  params = {}
197
197
  if filter:
198
198
  filter_dict = filter.model_dump(exclude_none=True)
199
- # Convert enum values to strings
199
+ # Convert enum values to their underlying str representation.
200
+ # Every VoicesV2Filter field is an Optional[Enum], so getattr
201
+ # falls back only if a future non-enum field is added.
200
202
  for key, value in filter_dict.items():
201
- if hasattr(value, "value"):
202
- params[key] = value.value
203
- else:
204
- params[key] = value
203
+ params[key] = getattr(value, "value", value)
205
204
 
206
205
  async with self.session.get(
207
206
  f"{self.host}{endpoint}", params=params
typecast/client.py CHANGED
@@ -173,12 +173,11 @@ class Typecast:
173
173
  params = {}
174
174
  if filter:
175
175
  filter_dict = filter.model_dump(exclude_none=True)
176
- # Convert enum values to strings
176
+ # Convert enum values to their underlying str representation.
177
+ # Every VoicesV2Filter field is an Optional[Enum], so getattr
178
+ # falls back only if a future non-enum field is added.
177
179
  for key, value in filter_dict.items():
178
- if hasattr(value, "value"):
179
- params[key] = value.value
180
- else:
181
- params[key] = value
180
+ params[key] = getattr(value, "value", value)
182
181
 
183
182
  response = self.session.get(f"{self.host}{endpoint}", params=params)
184
183
 
typecast/sse.py CHANGED
@@ -7,12 +7,35 @@ from .exceptions import TypecastError
7
7
 
8
8
 
9
9
  class TypecastSSE:
10
- SSE_URL = f"{conf.get_host()}/v1/text-to-speech/sse"
11
-
12
- def __init__(self, api_key: str):
10
+ """Server-Sent Events client for Typecast streaming endpoints."""
11
+
12
+ def __init__(
13
+ self,
14
+ api_key: Optional[str] = None,
15
+ host: Optional[str] = None,
16
+ sse_url: Optional[str] = None,
17
+ ):
18
+ """Initialize the SSE client.
19
+
20
+ Args:
21
+ api_key: API key. Defaults to TYPECAST_API_KEY env var.
22
+ host: API host. Defaults to TYPECAST_API_HOST env var or
23
+ 'https://api.typecast.ai'. Used to derive sse_url when
24
+ sse_url is not provided.
25
+ sse_url: Full SSE base URL override (escape hatch for tests).
26
+ When provided, takes precedence over host.
27
+ """
13
28
  self.api_key = conf.get_api_key(api_key)
29
+ self.host = conf.get_host(host)
30
+ self._sse_url_override = sse_url
14
31
  self.session: Optional[aiohttp.ClientSession] = None
15
32
 
33
+ @property
34
+ def sse_url(self) -> str:
35
+ if self._sse_url_override is not None:
36
+ return self._sse_url_override
37
+ return f"{self.host}/v1/text-to-speech/sse"
38
+
16
39
  async def connect(self, endpoint: str) -> AsyncIterator[str]:
17
40
  if self.session:
18
41
  await self.session.close()
@@ -21,7 +44,7 @@ class TypecastSSE:
21
44
  headers={"X-API-KEY": self.api_key, "Accept": "text/event-stream"}
22
45
  )
23
46
 
24
- async with self.session.get(f"{self.SSE_URL}/{endpoint}") as response:
47
+ async with self.session.get(f"{self.sse_url}/{endpoint}") as response:
25
48
  if response.status != 200:
26
49
  raise TypecastError(f"SSE connection failed: {response.status}")
27
50
 
typecast/websocket.py CHANGED
@@ -9,17 +9,18 @@ from .models import WebSocketMessage
9
9
 
10
10
 
11
11
  class TypecastWebSocket:
12
- WS_URL = "wss://api.typecast.ai/v1/ws"
12
+ """WebSocket client for Typecast streaming TTS."""
13
13
 
14
- def __init__(self, api_key: str):
14
+ DEFAULT_WS_URL = "wss://api.typecast.ai/v1/ws"
15
+
16
+ def __init__(self, api_key: str, ws_url: Optional[str] = None):
15
17
  self.api_key = api_key
18
+ self.ws_url = ws_url or self.DEFAULT_WS_URL
16
19
  self.ws: Optional[websockets.WebSocketClientProtocol] = None
17
20
  self.callbacks: dict[str, Callable] = {}
18
21
 
19
22
  async def connect(self):
20
- self.ws = await websockets.connect(f"{self.WS_URL}?token={self.api_key}")
21
-
22
- # Start message handler
23
+ self.ws = await websockets.connect(f"{self.ws_url}?token={self.api_key}")
23
24
  asyncio.create_task(self._message_handler())
24
25
 
25
26
  async def _message_handler(self):
@@ -34,7 +35,7 @@ class TypecastWebSocket:
34
35
  await self.callbacks[msg.type](msg.payload)
35
36
 
36
37
  def on(self, event_type: str, callback: Callable):
37
- """Register event callback"""
38
+ """Register event callback."""
38
39
  self.callbacks[event_type] = callback
39
40
 
40
41
  async def send(self, message: WebSocketMessage):
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: typecast-python
3
- Version: 0.1.7
3
+ Version: 0.1.9
4
4
  Summary: Official Typecast Python SDK - Convert text to lifelike speech using AI-powered voices
5
5
  Project-URL: Homepage, https://typecast.ai
6
6
  Project-URL: Documentation, https://typecast.ai/docs/overview
@@ -230,11 +230,13 @@ Requires-Dist: sseclient-py>=1.7.2
230
230
  Requires-Dist: typing-extensions>=4.0.0
231
231
  Requires-Dist: websockets>=10.0
232
232
  Provides-Extra: dev
233
+ Requires-Dist: aioresponses>=0.7.6; extra == 'dev'
233
234
  Requires-Dist: black>=23.0.0; extra == 'dev'
234
235
  Requires-Dist: flake8>=6.0.0; extra == 'dev'
235
236
  Requires-Dist: isort>=5.0.0; extra == 'dev'
236
237
  Requires-Dist: mypy>=1.0.0; extra == 'dev'
237
238
  Requires-Dist: pytest-asyncio>=0.21.0; extra == 'dev'
239
+ Requires-Dist: pytest-cov>=7.0.0; extra == 'dev'
238
240
  Requires-Dist: pytest-mock>=3.14.0; extra == 'dev'
239
241
  Requires-Dist: pytest>=7.0.0; extra == 'dev'
240
242
  Description-Content-Type: text/markdown
@@ -248,6 +250,7 @@ Description-Content-Type: text/markdown
248
250
  Convert text to lifelike speech using AI-powered voices
249
251
 
250
252
  [![PyPI version](https://img.shields.io/pypi/v/typecast-python.svg?style=flat-square)](https://pypi.org/project/typecast-python/)
253
+ [![coverage](https://img.shields.io/badge/coverage-100%25-brightgreen.svg?style=flat-square)](../docs/coverage-policy.md)
251
254
  [![License](https://img.shields.io/badge/license-Apache--2.0-blue.svg?style=flat-square)](LICENSE)
252
255
  [![Python](https://img.shields.io/badge/Python-3.9+-3776ab.svg?style=flat-square&logo=python&logoColor=white)](https://www.python.org/)
253
256
 
@@ -1,17 +1,17 @@
1
1
  typecast/__init__.py,sha256=8xdL3GnADTIsi2KuqPY938opQtN6GbHWnJI0XwjqHPc,871
2
- typecast/async_client.py,sha256=_v-YHussPMAG4az8XnWdXwb6uh5gcXAhWcpW1Scuf0o,9004
3
- typecast/client.py,sha256=Kj_IqZ_RgDGaPTylsUsYAQfRO_1PG7NZdEuc8nZ0a4A,7431
2
+ typecast/async_client.py,sha256=gSsp8pGxX7twCfnDDohFxazMV0G63UcHqgMRFKv5V_U,9083
3
+ typecast/client.py,sha256=B5EgmAJ6rWGfSsrWaiVoRlB98IW1lKrf2Un8yXE5uUA,7510
4
4
  typecast/conf.py,sha256=Fn_T4XW7BaHRnj0tP11BT5at3Y-db7oGcbBA_E1fmF0,479
5
5
  typecast/exceptions.py,sha256=Y0ZzYebe8zOSOSAHbXfKR0G_RJgdmZXxi15Z7ZxPLIk,1568
6
- typecast/sse.py,sha256=yYO2h5E_1-uiIID0TZM03eRZfvgkCStxu-0J6ej4Bpo,1109
6
+ typecast/sse.py,sha256=3qpNaS7iPU97QIWRFRUoiQFinHcIwloEim8Z_LtD05c,1951
7
7
  typecast/utils.py,sha256=XuNuX7gW8_CGKqZ-cv_tKlPVMPBluAYJBw2clwmjIMI,708
8
- typecast/websocket.py,sha256=SwFmk2efGtMLY3UgEtZljZAhULl5HG6KaA-rqfJ0Jaw,1336
8
+ typecast/websocket.py,sha256=3m5klCYykEOyC_j28rLR2zAt7LHWO4k3-2W-L2rlaGY,1450
9
9
  typecast/models/__init__.py,sha256=sF_0YSHXOPebNpNi_RiYAy-GnoG1D7ugHskszQYDsxk,754
10
10
  typecast/models/error.py,sha256=XomIjx7jvlCjItqzJuCAT4mXC9jwTjxR8lLDUk6P8KA,152
11
11
  typecast/models/tts.py,sha256=51TeFsiCZMzhMWdvP6Q6bKm4KCMjk0OyLPahOPig-yk,5358
12
12
  typecast/models/tts_wss.py,sha256=zYfP3oNs5VuOkxI7bu0uSCHDYfSbQWaenqg0lLLqaSc,101
13
13
  typecast/models/voices.py,sha256=VwmK_Ts17QccAaClGYqqstIGJ5RT9Qgj2kxqEoDr6z4,1659
14
- typecast_python-0.1.7.dist-info/METADATA,sha256=ZU-cAV8CicMjOdxmHiB4_XQQn7W-PwwcFbFEqE12QAA,22726
15
- typecast_python-0.1.7.dist-info/WHEEL,sha256=QccIxa26bgl1E6uMy58deGWi-0aeIkkangHcxk2kWfw,87
16
- typecast_python-0.1.7.dist-info/licenses/LICENSE,sha256=HvtJ-S89uUkuYmt-OvVk4MRxmzwtbn84__qJtSrGU2Q,11348
17
- typecast_python-0.1.7.dist-info/RECORD,,
14
+ typecast_python-0.1.9.dist-info/METADATA,sha256=cYVtKWj-dNPOMqWS_9uBeP0hbWk-CMa2TUM016GAEgE,22948
15
+ typecast_python-0.1.9.dist-info/WHEEL,sha256=QccIxa26bgl1E6uMy58deGWi-0aeIkkangHcxk2kWfw,87
16
+ typecast_python-0.1.9.dist-info/licenses/LICENSE,sha256=HvtJ-S89uUkuYmt-OvVk4MRxmzwtbn84__qJtSrGU2Q,11348
17
+ typecast_python-0.1.9.dist-info/RECORD,,