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.
- api/__init__.py +17 -0
- api/admin_config.py +1303 -0
- api/admin_routes.py +287 -0
- api/admin_static/admin.css +459 -0
- api/admin_static/admin.js +497 -0
- api/admin_static/index.html +77 -0
- api/admin_urls.py +34 -0
- api/app.py +194 -0
- api/command_utils.py +164 -0
- api/dependencies.py +144 -0
- api/detection.py +152 -0
- api/gateway_model_ids.py +54 -0
- api/model_catalog.py +133 -0
- api/model_router.py +125 -0
- api/models/__init__.py +45 -0
- api/models/anthropic.py +234 -0
- api/models/openai_responses.py +28 -0
- api/models/responses.py +60 -0
- api/optimization_handlers.py +154 -0
- api/request_pipeline.py +424 -0
- api/routes.py +156 -0
- api/runtime.py +334 -0
- api/validation_log.py +48 -0
- api/web_server_tools.py +22 -0
- api/web_tools/__init__.py +17 -0
- api/web_tools/constants.py +15 -0
- api/web_tools/egress.py +99 -0
- api/web_tools/outbound.py +278 -0
- api/web_tools/parsers.py +104 -0
- api/web_tools/request.py +87 -0
- api/web_tools/streaming.py +206 -0
- cli/__init__.py +5 -0
- cli/claude_env.py +12 -0
- cli/entrypoints.py +166 -0
- cli/env.example +209 -0
- cli/launchers/__init__.py +1 -0
- cli/launchers/claude.py +84 -0
- cli/launchers/codex.py +204 -0
- cli/launchers/codex_model_catalog.py +186 -0
- cli/launchers/common.py +93 -0
- cli/managed/__init__.py +6 -0
- cli/managed/claude.py +215 -0
- cli/managed/manager.py +157 -0
- cli/managed/session.py +260 -0
- cli/process_registry.py +78 -0
- config/__init__.py +5 -0
- config/constants.py +13 -0
- config/logging_config.py +159 -0
- config/nim.py +118 -0
- config/paths.py +91 -0
- config/provider_catalog.py +259 -0
- config/provider_ids.py +7 -0
- config/settings.py +538 -0
- core/__init__.py +1 -0
- core/anthropic/__init__.py +46 -0
- core/anthropic/content.py +31 -0
- core/anthropic/conversion.py +587 -0
- core/anthropic/emitted_sse_tracker.py +346 -0
- core/anthropic/errors.py +70 -0
- core/anthropic/native_messages_request.py +280 -0
- core/anthropic/native_sse_block_policy.py +313 -0
- core/anthropic/provider_stream_error.py +34 -0
- core/anthropic/server_tool_sse.py +14 -0
- core/anthropic/sse.py +440 -0
- core/anthropic/stream_contracts.py +205 -0
- core/anthropic/stream_recovery.py +346 -0
- core/anthropic/stream_recovery_session.py +133 -0
- core/anthropic/thinking.py +140 -0
- core/anthropic/tokens.py +117 -0
- core/anthropic/tools.py +212 -0
- core/anthropic/utils.py +9 -0
- core/openai_responses/__init__.py +5 -0
- core/openai_responses/adapter.py +31 -0
- core/openai_responses/anthropic_sse.py +59 -0
- core/openai_responses/errors.py +22 -0
- core/openai_responses/events.py +19 -0
- core/openai_responses/ids.py +21 -0
- core/openai_responses/input.py +258 -0
- core/openai_responses/items.py +37 -0
- core/openai_responses/reasoning.py +52 -0
- core/openai_responses/stream.py +25 -0
- core/openai_responses/stream_state.py +654 -0
- core/openai_responses/tools.py +374 -0
- core/openai_responses/usage.py +37 -0
- core/rate_limit.py +60 -0
- core/trace.py +216 -0
- devcopilot-0.2.0.dist-info/METADATA +687 -0
- devcopilot-0.2.0.dist-info/RECORD +189 -0
- devcopilot-0.2.0.dist-info/WHEEL +4 -0
- devcopilot-0.2.0.dist-info/entry_points.txt +6 -0
- devcopilot-0.2.0.dist-info/licenses/LICENSE +21 -0
- messaging/__init__.py +26 -0
- messaging/cli_event_constants.py +67 -0
- messaging/command_context.py +66 -0
- messaging/command_dispatcher.py +37 -0
- messaging/commands.py +275 -0
- messaging/event_parser.py +181 -0
- messaging/limiter.py +300 -0
- messaging/models.py +36 -0
- messaging/node_event_pipeline.py +127 -0
- messaging/node_runner.py +342 -0
- messaging/platforms/__init__.py +15 -0
- messaging/platforms/base.py +228 -0
- messaging/platforms/discord.py +567 -0
- messaging/platforms/factory.py +103 -0
- messaging/platforms/outbox.py +144 -0
- messaging/platforms/telegram.py +688 -0
- messaging/platforms/voice_flow.py +295 -0
- messaging/rendering/__init__.py +3 -0
- messaging/rendering/discord_markdown.py +318 -0
- messaging/rendering/markdown_tables.py +49 -0
- messaging/rendering/profiles.py +55 -0
- messaging/rendering/telegram_markdown.py +327 -0
- messaging/safe_diagnostics.py +17 -0
- messaging/session.py +334 -0
- messaging/transcript.py +581 -0
- messaging/transcription.py +164 -0
- messaging/trees/__init__.py +15 -0
- messaging/trees/data.py +482 -0
- messaging/trees/manager.py +433 -0
- messaging/trees/processor.py +179 -0
- messaging/trees/repository.py +177 -0
- messaging/turn_intake.py +235 -0
- messaging/ui_updates.py +101 -0
- messaging/voice.py +76 -0
- messaging/workflow.py +200 -0
- providers/__init__.py +31 -0
- providers/base.py +152 -0
- providers/cerebras/__init__.py +7 -0
- providers/cerebras/client.py +31 -0
- providers/cerebras/request.py +55 -0
- providers/codestral/__init__.py +7 -0
- providers/codestral/client.py +34 -0
- providers/deepseek/__init__.py +11 -0
- providers/deepseek/client.py +51 -0
- providers/deepseek/request.py +475 -0
- providers/defaults.py +41 -0
- providers/error_mapping.py +309 -0
- providers/exceptions.py +113 -0
- providers/fireworks/__init__.py +5 -0
- providers/fireworks/client.py +45 -0
- providers/fireworks/request.py +48 -0
- providers/gemini/__init__.py +7 -0
- providers/gemini/client.py +49 -0
- providers/gemini/request.py +199 -0
- providers/groq/__init__.py +7 -0
- providers/groq/client.py +31 -0
- providers/groq/request.py +83 -0
- providers/kimi/__init__.py +10 -0
- providers/kimi/client.py +53 -0
- providers/kimi/request.py +42 -0
- providers/llamacpp/__init__.py +3 -0
- providers/llamacpp/client.py +16 -0
- providers/lmstudio/__init__.py +5 -0
- providers/lmstudio/client.py +16 -0
- providers/mistral/__init__.py +7 -0
- providers/mistral/client.py +31 -0
- providers/mistral/request.py +37 -0
- providers/model_listing.py +133 -0
- providers/nvidia_nim/__init__.py +7 -0
- providers/nvidia_nim/client.py +91 -0
- providers/nvidia_nim/request.py +430 -0
- providers/nvidia_nim/voice.py +95 -0
- providers/ollama/__init__.py +7 -0
- providers/ollama/client.py +39 -0
- providers/open_router/__init__.py +7 -0
- providers/open_router/client.py +124 -0
- providers/open_router/request.py +42 -0
- providers/opencode/__init__.py +11 -0
- providers/opencode/client.py +31 -0
- providers/opencode/request.py +35 -0
- providers/rate_limit.py +300 -0
- providers/registry.py +527 -0
- providers/transports/__init__.py +1 -0
- providers/transports/anthropic_messages/__init__.py +5 -0
- providers/transports/anthropic_messages/http.py +118 -0
- providers/transports/anthropic_messages/recovery.py +206 -0
- providers/transports/anthropic_messages/stream.py +295 -0
- providers/transports/anthropic_messages/transport.py +236 -0
- providers/transports/openai_chat/__init__.py +5 -0
- providers/transports/openai_chat/recovery.py +217 -0
- providers/transports/openai_chat/stream.py +384 -0
- providers/transports/openai_chat/tool_calls.py +293 -0
- providers/transports/openai_chat/transport.py +156 -0
- providers/wafer/__init__.py +10 -0
- providers/wafer/client.py +50 -0
- providers/zai/__init__.py +10 -0
- providers/zai/client.py +46 -0
- 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
|
+
}
|