18fe063691
The 0.5.0 prompt mis-attributed the API: pull_start lives on SYNO.Docker.Image, not SYNO.Docker.Registry (live DSM capture). search and tags ARE correctly on Registry; only pull_start belongs to Image. Registry/pull_start returns "Method does not exist". Parameters are unchanged (repository + tag both JSON-encoded), and the Image/list polling for completion detection is untouched. Tests updated to assert SYNO.Docker.Image/pull_start. CLAUDE.md DSM-quirks section consolidates the Image vs. Registry split so this trap is documented for future surface additions. References #3 (already closed). Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
5.6 KiB
5.6 KiB
mcp-synology-container
Project
mcp-synology-container is an MCP server for managing Docker projects on a
Synology DiskStation via Container Manager. It exposes tools for projects,
containers, images, compose files, networks, and system housekeeping.
Tech stack
| Language | Python 3.12+, uv |
| Key deps | MCP SDK, httpx, keyring, click, rich |
| Compose paths | /volume1/docker/<project>/ (default Synology layout) |
Deploy workflow (after every code change)
1. Claude Code commits and pushes
2. uv tool install --reinstall git+<repo-url>
3. Restart Claude Desktop (tray icon → Quit → relaunch)
Push retry: the Gitea remote (gitea.gecheckt.de) occasionally
returns Unauthorized on the first push attempt. If git push fails
with an auth error, wait 1 s and retry once before reporting back.
Only a second consecutive failure is treated as a real auth problem.
Implemented tools (34)
| Category | Tools |
|---|---|
| Projects | list_projects, get_project_status, start_project, stop_project, redeploy_project, create_project, delete_project |
| Containers | list_containers, get_container_status, get_container_logs, exec_in_container, container_stats, delete_container, start_container, stop_container, restart_container |
| Compose | read_compose, update_compose, update_image_tag, update_env_var |
| Images | check_image_updates, list_images, delete_image, inspect_image |
| Registry | search_registry, list_image_tags, pull_image |
| Networks | list_networks, create_network, delete_network |
| System | system_df, system_prune, system_overview |
DSM API quirks
- Hash-prefixed container names — DSM sometimes returns names like
a1b2c3d4e5f6_myservicewhen the compose service name differs fromcontainer_name. All container tools strip this prefix transparently via_strip_hash_prefix/_resolve_container_name. - Async project start —
SYNO.Docker.Project/startreturns immediately while containers are still initialising.redeploy_projectpollsSYNO.Docker.Project/listevery 2 s for up to 30 s after issuing start. - Image delete — requires a form-encoded POST with a JSON
imagesarray (confirmed via browser DevTools); usesDsmClient.post_request(). SYNO.Docker.Image/pullvs.pull_start— the legacypullmethod exists but behaviour varies by DSM version; not exposed as a standalone tool.pull_imageusesSYNO.Docker.Image/pull_start(asynchronous pull entry point) with bothrepositoryandtagJSON-encoded. Note thatpull_startlives onSYNO.Docker.Image, NOT onSYNO.Docker.Registry— the Registry API only exposes the synchronous read-only methods (search,tags,get/set/create/delete,using); callingRegistry/pull_startreturns "Method does not exist". No matchingpull_statusmethod is confirmed on either API, so completion is detected by pollingSYNO.Docker.Image/listuntilrepository:tagappears (2–10 s backoff, 240 s budget). Timeout returns a "still running" hint instead of raising — DSM keeps pulling server-side regardless of the HTTP response.SYNO.Docker.Registry/tags— usesrepo(JSON-encoded) as the parameter name; the n4s4 reference'snamedoes not work on this DSM version. Returns the tag list as the envelope'sdatafield directly, not wrapped in a sub-key.SYNO.Docker.Volume— endpoint does not exist; volume management is not available via the DSM WebAPI.SYNO.Docker.Registry/get— does not behave as documented; registry listing omitted.SYNO.Docker.Container/pauseand/unpause— not implemented in DSM Container Manager on this firmware. The action menu only offers start/stop/force-stop/restart/reset; calls topause/unpausereturn "Method does not exist".pause_containerandunpause_containerwere briefly shipped in 0.4.0 and removed in 0.4.1.
Implementation rules
- Confirmation required before destructive operations:
stop_project,redeploy_project,create_project,delete_project,exec_in_container,update_image_tag,update_env_var,update_compose,delete_container,stop_container,restart_container,pull_image - After compose changes: suggest
redeploy_project - DSM errors → human-readable message, no stack traces
- No secrets in stderr output
- Type hints and docstrings everywhere
- Formatter:
ruff format· Linter:ruff check· Tests:pytest - All text (docstrings, comments, README): English
- CHANGELOG.md: every user-visible change (bug fix, new/changed
tool, behavior change, security fix, dependency bump) gets a
CHANGELOG.mdentry in the same commit — under a## [Unreleased]heading between releases, which becomes## [X.Y.Z] - YYYY-MM-DDon version bump. Pure internal cleanup (renames without external callers, comment-only edits, ruff autofix) needs no entry. Don't ship a release with a stale changelog (this was the C-2 gap that caused 0.2.7 and 0.2.8 to ship undocumented). - Version consistency: the package version lives in
pyproject.tomland must stay in sync withuv.lockand the[X.Y.Z]heading inCHANGELOG.md.src/mcp_synology_container/__init__.pyderives__version__fromimportlib.metadataand is never hand-edited. Every version bump touches all three files in the same commit.
DSM API reference
cmeans/mcp-synology(GitHub) — auth, keyring, CLI structureN4S4/synology-apidocker_api.py(GitHub) —SYNO.Docker.*calls