devcopilot 0.2.0__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.
Files changed (189) hide show
  1. api/__init__.py +17 -0
  2. api/admin_config.py +1303 -0
  3. api/admin_routes.py +287 -0
  4. api/admin_static/admin.css +459 -0
  5. api/admin_static/admin.js +497 -0
  6. api/admin_static/index.html +77 -0
  7. api/admin_urls.py +34 -0
  8. api/app.py +194 -0
  9. api/command_utils.py +164 -0
  10. api/dependencies.py +144 -0
  11. api/detection.py +152 -0
  12. api/gateway_model_ids.py +54 -0
  13. api/model_catalog.py +133 -0
  14. api/model_router.py +125 -0
  15. api/models/__init__.py +45 -0
  16. api/models/anthropic.py +234 -0
  17. api/models/openai_responses.py +28 -0
  18. api/models/responses.py +60 -0
  19. api/optimization_handlers.py +154 -0
  20. api/request_pipeline.py +424 -0
  21. api/routes.py +156 -0
  22. api/runtime.py +334 -0
  23. api/validation_log.py +48 -0
  24. api/web_server_tools.py +22 -0
  25. api/web_tools/__init__.py +17 -0
  26. api/web_tools/constants.py +15 -0
  27. api/web_tools/egress.py +99 -0
  28. api/web_tools/outbound.py +278 -0
  29. api/web_tools/parsers.py +104 -0
  30. api/web_tools/request.py +87 -0
  31. api/web_tools/streaming.py +206 -0
  32. cli/__init__.py +5 -0
  33. cli/claude_env.py +12 -0
  34. cli/entrypoints.py +166 -0
  35. cli/env.example +209 -0
  36. cli/launchers/__init__.py +1 -0
  37. cli/launchers/claude.py +84 -0
  38. cli/launchers/codex.py +204 -0
  39. cli/launchers/codex_model_catalog.py +186 -0
  40. cli/launchers/common.py +93 -0
  41. cli/managed/__init__.py +6 -0
  42. cli/managed/claude.py +215 -0
  43. cli/managed/manager.py +157 -0
  44. cli/managed/session.py +260 -0
  45. cli/process_registry.py +78 -0
  46. config/__init__.py +5 -0
  47. config/constants.py +13 -0
  48. config/logging_config.py +159 -0
  49. config/nim.py +118 -0
  50. config/paths.py +91 -0
  51. config/provider_catalog.py +259 -0
  52. config/provider_ids.py +7 -0
  53. config/settings.py +538 -0
  54. core/__init__.py +1 -0
  55. core/anthropic/__init__.py +46 -0
  56. core/anthropic/content.py +31 -0
  57. core/anthropic/conversion.py +587 -0
  58. core/anthropic/emitted_sse_tracker.py +346 -0
  59. core/anthropic/errors.py +70 -0
  60. core/anthropic/native_messages_request.py +280 -0
  61. core/anthropic/native_sse_block_policy.py +313 -0
  62. core/anthropic/provider_stream_error.py +34 -0
  63. core/anthropic/server_tool_sse.py +14 -0
  64. core/anthropic/sse.py +440 -0
  65. core/anthropic/stream_contracts.py +205 -0
  66. core/anthropic/stream_recovery.py +346 -0
  67. core/anthropic/stream_recovery_session.py +133 -0
  68. core/anthropic/thinking.py +140 -0
  69. core/anthropic/tokens.py +117 -0
  70. core/anthropic/tools.py +212 -0
  71. core/anthropic/utils.py +9 -0
  72. core/openai_responses/__init__.py +5 -0
  73. core/openai_responses/adapter.py +31 -0
  74. core/openai_responses/anthropic_sse.py +59 -0
  75. core/openai_responses/errors.py +22 -0
  76. core/openai_responses/events.py +19 -0
  77. core/openai_responses/ids.py +21 -0
  78. core/openai_responses/input.py +258 -0
  79. core/openai_responses/items.py +37 -0
  80. core/openai_responses/reasoning.py +52 -0
  81. core/openai_responses/stream.py +25 -0
  82. core/openai_responses/stream_state.py +654 -0
  83. core/openai_responses/tools.py +374 -0
  84. core/openai_responses/usage.py +37 -0
  85. core/rate_limit.py +60 -0
  86. core/trace.py +216 -0
  87. devcopilot-0.2.0.dist-info/METADATA +687 -0
  88. devcopilot-0.2.0.dist-info/RECORD +189 -0
  89. devcopilot-0.2.0.dist-info/WHEEL +4 -0
  90. devcopilot-0.2.0.dist-info/entry_points.txt +6 -0
  91. devcopilot-0.2.0.dist-info/licenses/LICENSE +21 -0
  92. messaging/__init__.py +26 -0
  93. messaging/cli_event_constants.py +67 -0
  94. messaging/command_context.py +66 -0
  95. messaging/command_dispatcher.py +37 -0
  96. messaging/commands.py +275 -0
  97. messaging/event_parser.py +181 -0
  98. messaging/limiter.py +300 -0
  99. messaging/models.py +36 -0
  100. messaging/node_event_pipeline.py +127 -0
  101. messaging/node_runner.py +342 -0
  102. messaging/platforms/__init__.py +15 -0
  103. messaging/platforms/base.py +228 -0
  104. messaging/platforms/discord.py +567 -0
  105. messaging/platforms/factory.py +103 -0
  106. messaging/platforms/outbox.py +144 -0
  107. messaging/platforms/telegram.py +688 -0
  108. messaging/platforms/voice_flow.py +295 -0
  109. messaging/rendering/__init__.py +3 -0
  110. messaging/rendering/discord_markdown.py +318 -0
  111. messaging/rendering/markdown_tables.py +49 -0
  112. messaging/rendering/profiles.py +55 -0
  113. messaging/rendering/telegram_markdown.py +327 -0
  114. messaging/safe_diagnostics.py +17 -0
  115. messaging/session.py +334 -0
  116. messaging/transcript.py +581 -0
  117. messaging/transcription.py +164 -0
  118. messaging/trees/__init__.py +15 -0
  119. messaging/trees/data.py +482 -0
  120. messaging/trees/manager.py +433 -0
  121. messaging/trees/processor.py +179 -0
  122. messaging/trees/repository.py +177 -0
  123. messaging/turn_intake.py +235 -0
  124. messaging/ui_updates.py +101 -0
  125. messaging/voice.py +76 -0
  126. messaging/workflow.py +200 -0
  127. providers/__init__.py +31 -0
  128. providers/base.py +152 -0
  129. providers/cerebras/__init__.py +7 -0
  130. providers/cerebras/client.py +31 -0
  131. providers/cerebras/request.py +55 -0
  132. providers/codestral/__init__.py +7 -0
  133. providers/codestral/client.py +34 -0
  134. providers/deepseek/__init__.py +11 -0
  135. providers/deepseek/client.py +51 -0
  136. providers/deepseek/request.py +475 -0
  137. providers/defaults.py +41 -0
  138. providers/error_mapping.py +309 -0
  139. providers/exceptions.py +113 -0
  140. providers/fireworks/__init__.py +5 -0
  141. providers/fireworks/client.py +45 -0
  142. providers/fireworks/request.py +48 -0
  143. providers/gemini/__init__.py +7 -0
  144. providers/gemini/client.py +49 -0
  145. providers/gemini/request.py +199 -0
  146. providers/groq/__init__.py +7 -0
  147. providers/groq/client.py +31 -0
  148. providers/groq/request.py +83 -0
  149. providers/kimi/__init__.py +10 -0
  150. providers/kimi/client.py +53 -0
  151. providers/kimi/request.py +42 -0
  152. providers/llamacpp/__init__.py +3 -0
  153. providers/llamacpp/client.py +16 -0
  154. providers/lmstudio/__init__.py +5 -0
  155. providers/lmstudio/client.py +16 -0
  156. providers/mistral/__init__.py +7 -0
  157. providers/mistral/client.py +31 -0
  158. providers/mistral/request.py +37 -0
  159. providers/model_listing.py +133 -0
  160. providers/nvidia_nim/__init__.py +7 -0
  161. providers/nvidia_nim/client.py +91 -0
  162. providers/nvidia_nim/request.py +430 -0
  163. providers/nvidia_nim/voice.py +95 -0
  164. providers/ollama/__init__.py +7 -0
  165. providers/ollama/client.py +39 -0
  166. providers/open_router/__init__.py +7 -0
  167. providers/open_router/client.py +124 -0
  168. providers/open_router/request.py +42 -0
  169. providers/opencode/__init__.py +11 -0
  170. providers/opencode/client.py +31 -0
  171. providers/opencode/request.py +35 -0
  172. providers/rate_limit.py +300 -0
  173. providers/registry.py +527 -0
  174. providers/transports/__init__.py +1 -0
  175. providers/transports/anthropic_messages/__init__.py +5 -0
  176. providers/transports/anthropic_messages/http.py +118 -0
  177. providers/transports/anthropic_messages/recovery.py +206 -0
  178. providers/transports/anthropic_messages/stream.py +295 -0
  179. providers/transports/anthropic_messages/transport.py +236 -0
  180. providers/transports/openai_chat/__init__.py +5 -0
  181. providers/transports/openai_chat/recovery.py +217 -0
  182. providers/transports/openai_chat/stream.py +384 -0
  183. providers/transports/openai_chat/tool_calls.py +293 -0
  184. providers/transports/openai_chat/transport.py +156 -0
  185. providers/wafer/__init__.py +10 -0
  186. providers/wafer/client.py +50 -0
  187. providers/zai/__init__.py +10 -0
  188. providers/zai/client.py +46 -0
  189. providers/zai/request.py +42 -0
api/admin_routes.py ADDED
@@ -0,0 +1,287 @@
1
+ """Local admin UI routes and APIs."""
2
+
3
+ from __future__ import annotations
4
+
5
+ import inspect
6
+ import ipaddress
7
+ from pathlib import Path
8
+ from typing import Any
9
+ from urllib.parse import urlsplit
10
+
11
+ import httpx
12
+ from fastapi import APIRouter, BackgroundTasks, HTTPException, Request
13
+ from fastapi.responses import FileResponse
14
+ from pydantic import BaseModel, Field
15
+
16
+ from config.settings import Settings
17
+ from config.settings import get_settings as get_cached_settings
18
+ from providers.registry import ProviderRegistry
19
+
20
+ from .admin_config import (
21
+ FIELD_BY_KEY,
22
+ load_config_response,
23
+ provider_config_status,
24
+ validate_updates,
25
+ write_managed_env,
26
+ )
27
+ from .admin_urls import local_admin_url
28
+
29
+ router = APIRouter()
30
+
31
+ STATIC_DIR = Path(__file__).resolve().parent / "admin_static"
32
+ LOCAL_PROVIDER_PATHS = {
33
+ "lmstudio": "/models",
34
+ "llamacpp": "/models",
35
+ "ollama": "/api/tags",
36
+ }
37
+
38
+
39
+ class AdminConfigPayload(BaseModel):
40
+ """Partial config update submitted by the admin UI."""
41
+
42
+ values: dict[str, Any] = Field(default_factory=dict)
43
+
44
+
45
+ def _is_loopback_host(host: str | None) -> bool:
46
+ if host is None:
47
+ return False
48
+ normalized = host.strip().strip("[]").lower()
49
+ if normalized == "localhost":
50
+ return True
51
+ try:
52
+ return ipaddress.ip_address(normalized).is_loopback
53
+ except ValueError:
54
+ return False
55
+
56
+
57
+ def _origin_is_local(origin: str | None) -> bool:
58
+ if not origin:
59
+ return True
60
+ parsed = urlsplit(origin)
61
+ return _is_loopback_host(parsed.hostname)
62
+
63
+
64
+ def require_loopback_admin(request: Request) -> None:
65
+ """Allow admin access only from the local machine."""
66
+
67
+ client_host = request.client.host if request.client else None
68
+ if not _is_loopback_host(client_host):
69
+ raise HTTPException(status_code=403, detail="Admin UI is local-only")
70
+
71
+ origin = request.headers.get("origin")
72
+ if not _origin_is_local(origin):
73
+ raise HTTPException(status_code=403, detail="Admin UI is local-only")
74
+
75
+
76
+ def _asset_response(filename: str) -> FileResponse:
77
+ path = STATIC_DIR / filename
78
+ if not path.is_file():
79
+ raise HTTPException(status_code=404, detail="Admin asset not found")
80
+ return FileResponse(path)
81
+
82
+
83
+ @router.get("/admin", include_in_schema=False)
84
+ async def admin_page(request: Request):
85
+ require_loopback_admin(request)
86
+ return _asset_response("index.html")
87
+
88
+
89
+ @router.get("/admin/assets/{filename}", include_in_schema=False)
90
+ async def admin_asset(filename: str, request: Request):
91
+ require_loopback_admin(request)
92
+ if filename not in {"admin.css", "admin.js"}:
93
+ raise HTTPException(status_code=404, detail="Admin asset not found")
94
+ return _asset_response(filename)
95
+
96
+
97
+ @router.get("/admin/api/config")
98
+ async def get_admin_config(request: Request):
99
+ require_loopback_admin(request)
100
+ return load_config_response()
101
+
102
+
103
+ @router.post("/admin/api/config/validate")
104
+ async def validate_admin_config(payload: AdminConfigPayload, request: Request):
105
+ require_loopback_admin(request)
106
+ return validate_updates(_filtered_values(payload.values))
107
+
108
+
109
+ @router.post("/admin/api/config/apply")
110
+ async def apply_admin_config(
111
+ payload: AdminConfigPayload,
112
+ request: Request,
113
+ background_tasks: BackgroundTasks,
114
+ ):
115
+ require_loopback_admin(request)
116
+ result = write_managed_env(_filtered_values(payload.values))
117
+ if not result["applied"]:
118
+ return result
119
+
120
+ get_cached_settings.cache_clear()
121
+ restart = _restart_metadata(result["pending_fields"], request)
122
+ result["restart"] = restart
123
+ if restart["required"] and restart["automatic"]:
124
+ callback = request.app.state.admin_restart_callback
125
+ background_tasks.add_task(_invoke_admin_restart_callback, callback)
126
+ request.app.state.admin_pending_fields = []
127
+ return result
128
+
129
+ old_registry = getattr(request.app.state, "provider_registry", None)
130
+ if isinstance(old_registry, ProviderRegistry):
131
+ await old_registry.cleanup()
132
+ request.app.state.provider_registry = ProviderRegistry()
133
+ request.app.state.admin_pending_fields = result["pending_fields"]
134
+ return result
135
+
136
+
137
+ @router.get("/admin/api/status")
138
+ async def admin_status(request: Request):
139
+ require_loopback_admin(request)
140
+ settings = get_cached_settings()
141
+ registry = getattr(request.app.state, "provider_registry", None)
142
+ cached_models: dict[str, list[str]] = {}
143
+ if isinstance(registry, ProviderRegistry):
144
+ cached_models = {
145
+ provider_id: sorted(model_ids)
146
+ for provider_id, model_ids in registry.cached_model_ids().items()
147
+ }
148
+ return {
149
+ "status": "running",
150
+ "host": settings.host,
151
+ "port": settings.port,
152
+ "model": settings.model,
153
+ "provider": settings.provider_type,
154
+ "pending_fields": getattr(request.app.state, "admin_pending_fields", []),
155
+ "provider_status": provider_config_status(),
156
+ "cached_models": cached_models,
157
+ }
158
+
159
+
160
+ @router.get("/admin/api/providers/local-status")
161
+ async def local_provider_status(request: Request):
162
+ require_loopback_admin(request)
163
+ config = load_config_response()
164
+ values = {field["key"]: field["value"] for field in config["fields"]}
165
+ checks = []
166
+ for provider_id, path in LOCAL_PROVIDER_PATHS.items():
167
+ base_url = _local_provider_url(provider_id, values)
168
+ checks.append(await _check_local_provider(provider_id, base_url, path))
169
+ return {"providers": checks}
170
+
171
+
172
+ @router.post("/admin/api/providers/{provider_id}/test")
173
+ async def test_provider(provider_id: str, request: Request):
174
+ require_loopback_admin(request)
175
+ settings = get_cached_settings()
176
+ registry = getattr(request.app.state, "provider_registry", None)
177
+ if not isinstance(registry, ProviderRegistry):
178
+ registry = ProviderRegistry()
179
+ request.app.state.provider_registry = registry
180
+ try:
181
+ provider = registry.get(provider_id, settings)
182
+ infos = await provider.list_model_infos()
183
+ except Exception as exc:
184
+ return {
185
+ "provider_id": provider_id,
186
+ "ok": False,
187
+ "error_type": type(exc).__name__,
188
+ }
189
+ registry.cache_model_infos(provider_id, infos)
190
+ return {
191
+ "provider_id": provider_id,
192
+ "ok": True,
193
+ "models": sorted(info.model_id for info in infos),
194
+ }
195
+
196
+
197
+ @router.post("/admin/api/models/refresh")
198
+ async def refresh_models(request: Request):
199
+ require_loopback_admin(request)
200
+ settings = get_cached_settings()
201
+ registry = getattr(request.app.state, "provider_registry", None)
202
+ if not isinstance(registry, ProviderRegistry):
203
+ registry = ProviderRegistry()
204
+ request.app.state.provider_registry = registry
205
+ await registry.refresh_model_list_cache(settings)
206
+ return {
207
+ "cached_models": {
208
+ provider_id: sorted(model_ids)
209
+ for provider_id, model_ids in registry.cached_model_ids().items()
210
+ }
211
+ }
212
+
213
+
214
+ def _filtered_values(values: dict[str, Any]) -> dict[str, Any]:
215
+ return {key: value for key, value in values.items() if key in FIELD_BY_KEY}
216
+
217
+
218
+ async def _invoke_admin_restart_callback(callback: Any) -> None:
219
+ result = callback()
220
+ if inspect.isawaitable(result):
221
+ await result
222
+
223
+
224
+ def _restart_metadata(fields: list[str], request: Request) -> dict[str, Any]:
225
+ callback = getattr(request.app.state, "admin_restart_callback", None)
226
+ automatic = bool(fields and callable(callback))
227
+ return {
228
+ "required": bool(fields),
229
+ "automatic": automatic,
230
+ "admin_url": _next_admin_url() if automatic else None,
231
+ "fields": fields,
232
+ }
233
+
234
+
235
+ def _next_admin_url() -> str:
236
+ fields = {
237
+ field["key"]: field["value"] for field in load_config_response()["fields"]
238
+ }
239
+ settings = Settings.model_construct(
240
+ host=fields.get("HOST") or "0.0.0.0",
241
+ port=int(fields.get("PORT") or 8082),
242
+ )
243
+ return local_admin_url(settings)
244
+
245
+
246
+ def _local_provider_url(provider_id: str, values: dict[str, str]) -> str:
247
+ if provider_id == "lmstudio":
248
+ return values.get("LM_STUDIO_BASE_URL", "")
249
+ if provider_id == "llamacpp":
250
+ return values.get("LLAMACPP_BASE_URL", "")
251
+ if provider_id == "ollama":
252
+ return values.get("OLLAMA_BASE_URL", "")
253
+ return ""
254
+
255
+
256
+ async def _check_local_provider(
257
+ provider_id: str, base_url: str, path: str
258
+ ) -> dict[str, Any]:
259
+ clean_url = base_url.strip().rstrip("/")
260
+ if not clean_url:
261
+ return {
262
+ "provider_id": provider_id,
263
+ "status": "missing_url",
264
+ "label": "Missing URL",
265
+ "base_url": base_url,
266
+ }
267
+
268
+ url = f"{clean_url}{path}"
269
+ try:
270
+ async with httpx.AsyncClient(timeout=1.5) as client:
271
+ response = await client.get(url)
272
+ ok = 200 <= response.status_code < 300
273
+ return {
274
+ "provider_id": provider_id,
275
+ "status": "reachable" if ok else "offline",
276
+ "label": "Reachable" if ok else "Offline",
277
+ "base_url": base_url,
278
+ "status_code": response.status_code,
279
+ }
280
+ except Exception as exc:
281
+ return {
282
+ "provider_id": provider_id,
283
+ "status": "offline",
284
+ "label": "Offline",
285
+ "base_url": base_url,
286
+ "error_type": type(exc).__name__,
287
+ }