8878eda0b2
Add an explicit rule that every user-visible change updates CHANGELOG.md in the same commit, under an `## [Unreleased]` heading between releases. References the C-2 gap (0.2.7 and 0.2.8 shipped without changelog entries) so the motivation is durable. Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
4.0 KiB
4.0 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 (23)
| Category | Tools |
|---|---|
| Projects | list_projects, get_project_status, start_project, stop_project, redeploy_project |
| Containers | list_containers, get_container_status, get_container_logs, exec_in_container, container_stats, delete_container |
| Compose | read_compose, update_compose, update_image_tag, update_env_var |
| Images | check_image_updates, list_images, delete_image |
| Networks | list_networks, create_network, delete_network |
| System | system_df, system_prune |
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/pull— API method exists but behaviour varies by DSM version; not exposed as a standalone tool.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.
Implementation rules
- Confirmation required before destructive operations:
stop_project,redeploy_project,exec_in_container,update_image_tag,update_env_var,update_compose,delete_container - 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