solana-agent 31.2.3__tar.gz → 31.2.4__tar.gz

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.
Files changed (47) hide show
  1. {solana_agent-31.2.3 → solana_agent-31.2.4}/PKG-INFO +4 -4
  2. {solana_agent-31.2.3 → solana_agent-31.2.4}/README.md +3 -3
  3. {solana_agent-31.2.3 → solana_agent-31.2.4}/pyproject.toml +1 -1
  4. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/adapters/ffmpeg_transcoder.py +40 -4
  5. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/adapters/openai_realtime_ws.py +9 -0
  6. {solana_agent-31.2.3 → solana_agent-31.2.4}/LICENSE +0 -0
  7. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/__init__.py +0 -0
  8. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/adapters/__init__.py +0 -0
  9. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/adapters/mongodb_adapter.py +0 -0
  10. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/adapters/openai_adapter.py +0 -0
  11. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/adapters/pinecone_adapter.py +0 -0
  12. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/cli.py +0 -0
  13. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/client/__init__.py +0 -0
  14. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/client/solana_agent.py +0 -0
  15. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/domains/__init__.py +0 -0
  16. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/domains/agent.py +0 -0
  17. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/domains/routing.py +0 -0
  18. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/factories/__init__.py +0 -0
  19. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/factories/agent_factory.py +0 -0
  20. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/guardrails/pii.py +0 -0
  21. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/interfaces/__init__.py +0 -0
  22. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/interfaces/client/client.py +0 -0
  23. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/interfaces/guardrails/guardrails.py +0 -0
  24. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/interfaces/plugins/plugins.py +0 -0
  25. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/interfaces/providers/audio.py +0 -0
  26. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/interfaces/providers/data_storage.py +0 -0
  27. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/interfaces/providers/llm.py +0 -0
  28. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/interfaces/providers/memory.py +0 -0
  29. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/interfaces/providers/realtime.py +0 -0
  30. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/interfaces/providers/vector_storage.py +0 -0
  31. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/interfaces/services/agent.py +0 -0
  32. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/interfaces/services/knowledge_base.py +0 -0
  33. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/interfaces/services/query.py +0 -0
  34. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/interfaces/services/routing.py +0 -0
  35. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/plugins/__init__.py +0 -0
  36. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/plugins/manager.py +0 -0
  37. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/plugins/registry.py +0 -0
  38. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/plugins/tools/__init__.py +0 -0
  39. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/plugins/tools/auto_tool.py +0 -0
  40. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/repositories/__init__.py +0 -0
  41. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/repositories/memory.py +0 -0
  42. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/services/__init__.py +0 -0
  43. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/services/agent.py +0 -0
  44. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/services/knowledge_base.py +0 -0
  45. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/services/query.py +0 -0
  46. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/services/realtime.py +0 -0
  47. {solana_agent-31.2.3 → solana_agent-31.2.4}/solana_agent/services/routing.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.3
2
2
  Name: solana-agent
3
- Version: 31.2.3
3
+ Version: 31.2.4
4
4
  Summary: AI Agents for Solana
5
5
  License: MIT
6
6
  Keywords: solana,solana ai,solana agent,ai,ai agent,ai agents
@@ -335,18 +335,18 @@ async def generate():
335
335
  rt_encode_output=True,
336
336
  rt_voice="marin",
337
337
  output_format="audio",
338
- audio_output_format="m4a",
338
+ audio_output_format="mp3",
339
339
  audio_input_format="mp4",
340
340
  ):
341
341
  yield chunk
342
342
 
343
343
  return StreamingResponse(
344
344
  content=generate(),
345
- media_type="audio/mp4",
345
+ media_type="audio/mp3",
346
346
  headers={
347
347
  "Cache-Control": "no-store",
348
348
  "Pragma": "no-cache",
349
- "Content-Disposition": "inline; filename=stream.m4a",
349
+ "Content-Disposition": "inline; filename=stream.mp3",
350
350
  "X-Accel-Buffering": "no",
351
351
  },
352
352
  )
@@ -299,18 +299,18 @@ async def generate():
299
299
  rt_encode_output=True,
300
300
  rt_voice="marin",
301
301
  output_format="audio",
302
- audio_output_format="m4a",
302
+ audio_output_format="mp3",
303
303
  audio_input_format="mp4",
304
304
  ):
305
305
  yield chunk
306
306
 
307
307
  return StreamingResponse(
308
308
  content=generate(),
309
- media_type="audio/mp4",
309
+ media_type="audio/mp3",
310
310
  headers={
311
311
  "Cache-Control": "no-store",
312
312
  "Pragma": "no-cache",
313
- "Content-Disposition": "inline; filename=stream.m4a",
313
+ "Content-Disposition": "inline; filename=stream.mp3",
314
314
  "X-Accel-Buffering": "no",
315
315
  },
316
316
  )
@@ -1,6 +1,6 @@
1
1
  [tool.poetry]
2
2
  name = "solana-agent"
3
- version = "31.2.3"
3
+ version = "31.2.4"
4
4
  description = "AI Agents for Solana"
5
5
  authors = ["Bevan Hunt <bevan@bevanhunt.com>"]
6
6
  license = "MIT"
@@ -4,6 +4,8 @@ import asyncio
4
4
  import contextlib
5
5
  import logging
6
6
  from typing import List, AsyncGenerator
7
+ import tempfile
8
+ import os
7
9
 
8
10
  from solana_agent.interfaces.providers.audio import AudioTranscoder
9
11
 
@@ -49,11 +51,45 @@ class FFmpegTranscoder(AudioTranscoder):
49
51
  rate_hz,
50
52
  len(audio_bytes),
51
53
  )
52
- # Prefer to hint format for common containers/codecs; ffmpeg can still autodetect if hint is wrong.
53
- hinted_format = None
54
+ # iOS-recorded MP4/M4A often requires a seekable input for reliable demuxing.
55
+ # Decode from a temporary file instead of stdin for MP4/M4A.
54
56
  if input_mime in ("audio/mp4", "audio/m4a"):
55
- hinted_format = "mp4"
56
- elif input_mime in ("audio/aac",):
57
+ suffix = ".m4a" if input_mime == "audio/m4a" else ".mp4"
58
+ tmp_path = None
59
+ try:
60
+ with tempfile.NamedTemporaryFile(delete=False, suffix=suffix) as tf:
61
+ tmp_path = tf.name
62
+ tf.write(audio_bytes)
63
+ args = [
64
+ "-hide_banner",
65
+ "-loglevel",
66
+ "error",
67
+ "-i",
68
+ tmp_path,
69
+ "-vn", # ignore any video tracks
70
+ "-acodec",
71
+ "pcm_s16le",
72
+ "-ac",
73
+ "1",
74
+ "-ar",
75
+ str(rate_hz),
76
+ "-f",
77
+ "s16le",
78
+ "pipe:1",
79
+ ]
80
+ out = await self._run_ffmpeg(args, b"")
81
+ logger.info(
82
+ "Transcoded (MP4/M4A temp-file) to PCM16: output_len=%d", len(out)
83
+ )
84
+ return out
85
+ finally:
86
+ if tmp_path:
87
+ with contextlib.suppress(Exception):
88
+ os.remove(tmp_path)
89
+
90
+ # For other formats, prefer a format hint when helpful and decode from stdin.
91
+ hinted_format = None
92
+ if input_mime in ("audio/aac",):
57
93
  # Raw AAC is typically in ADTS stream format
58
94
  hinted_format = "adts"
59
95
  elif input_mime in ("audio/ogg", "audio/webm"):
@@ -1037,6 +1037,15 @@ class OpenAIRealtimeWebSocketSession(BaseRealtimeSession):
1037
1037
  if "tools" in patch:
1038
1038
  patch["tools"] = _strip_tool_strict(patch["tools"]) # idempotent
1039
1039
 
1040
+ # Per server requirements, always include session.type and output_modalities
1041
+ try:
1042
+ patch["type"] = "realtime"
1043
+ # Preserve caller-provided output_modalities if present, otherwise default to audio
1044
+ if "output_modalities" not in patch:
1045
+ patch["output_modalities"] = ["audio"]
1046
+ except Exception:
1047
+ pass
1048
+
1040
1049
  payload = {"type": "session.update", "session": patch}
1041
1050
  # Mark awaiting updated and store last patch
1042
1051
  self._last_session_patch = patch or {}
File without changes