fix: v0.4.2 — inspect_image used wrong DSM parameter contract

Live DSM API capture revealed the actual SYNO.Docker.Image/get
contract: a single JSON-encoded parameter named `identity` that
accepts both `name:tag` and `sha256:<hash>` forms. The 0.4.0 code
passed `name` + `tag` + `id` and was rejected by DSM with error 114.

Response shape also corrected — the endpoint returns flat top-level
fields (image, tag, id, digest, size, virtual_size, author,
docker_version, cmd, entrypoint, env, ports, volumes), NOT the
Docker-engine inspect shape with details.Config.* + RootFS.Layers
that the previous implementation assumed. Layer rendering removed;
digest / author / docker_version / volumes are now displayed.

Pre-resolution against list_images is no longer needed — the user
input goes straight into `identity` JSON-encoded.

Closes #4.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
2026-05-18 13:07:53 +02:00
parent 24b97338ba
commit 4030b8d5ee
5 changed files with 167 additions and 190 deletions
+21
View File
@@ -2,6 +2,27 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
## [0.4.2] - 2026-05-18
### Fixed
- `inspect_image` (#4): the 0.4.0 implementation called
`SYNO.Docker.Image/get` with `name` + `tag` + `id` parameters, which
DSM rejected with error 114 ("invalid parameter"). Reverse-
engineering the live API capture revealed the correct contract:
one JSON-encoded parameter named `identity` that accepts either form:
- `name:tag` (e.g. `"gitea/gitea:1.26.1"`)
- `sha256:<hash>` (e.g. `"sha256:cd21e54e..."`)
The pre-resolution lookup against `list_images` is no longer needed —
the user input is JSON-encoded and passed straight through.
- `inspect_image` response parsing: the endpoint does NOT return
`details.Config.*` / `RootFS.Layers` (the Docker-engine inspect shape
the previous code assumed). The actual top-level fields are `image`,
`tag`, `id`, `digest`, `size`, `virtual_size`, `author`,
`docker_version`, `cmd`, `entrypoint`, `env`, `ports`, `volumes`.
Layer rendering is removed (the endpoint does not surface layers);
digest, author, docker version, and volumes are now displayed.
## [0.4.1] - 2026-05-18 ## [0.4.1] - 2026-05-18
### Removed ### Removed
+1 -1
View File
@@ -1,6 +1,6 @@
[project] [project]
name = "mcp-synology-container" name = "mcp-synology-container"
version = "0.4.1" version = "0.4.2"
description = "MCP server for Synology Container Manager" description = "MCP server for Synology Container Manager"
requires-python = ">=3.12" requires-python = ">=3.12"
dependencies = [ dependencies = [
+47 -125
View File
@@ -273,109 +273,51 @@ def register_images(mcp: FastMCP, config: AppConfig, client: DsmClient) -> None:
@mcp.tool() @mcp.tool()
async def inspect_image(image_id: str): async def inspect_image(image_id: str):
"""Inspect a local image by name:tag or hash — shows config, layers, env, ports.""" """Inspect a local image by name:tag or sha256 hash — shows config, env, ports."""
# Parse name and tag using the last ":" as separator so registry-prefixed # SYNO.Docker.Image/get version=1 expects the parameter "identity"
# images (e.g. "ghcr.io/foo/bar:v1") are handled correctly. # (JSON-encoded). It accepts both `name:tag` and `sha256:<hash>` —
name, sep, tag = image_id.rpartition(":") # no pre-resolution needed. Confirmed via DSM API capture.
if not sep:
name = image_id
tag = "latest"
# Resolve the image against the local list so the user can pass either
# name:tag or a hash (full or 12-char prefix).
try: try:
img_data = await client.request(
"SYNO.Docker.Image",
"list",
params={"limit": "-1", "offset": "0", "show_dsm": "false"},
)
except Exception as e:
return f"Error inspecting image '{image_id}': {e}"
images: list[dict[str, Any]] = img_data.get("images", [])
is_hash = image_id.startswith("sha256:") or (len(image_id) >= 12 and ":" not in image_id)
target: dict[str, Any] | None = None
for img in images:
if is_hash:
img_hash = img.get("id", "")
if img_hash == image_id or img_hash.startswith(image_id):
target = img
break
else:
repo = img.get("repository", "")
img_tags = img.get("tags") or []
if repo == name and tag in img_tags:
target = img
break
if target is None:
return f"Image '{image_id}' not found locally."
repo = target.get("repository", name)
img_tags = target.get("tags") or [tag]
display_name = f"{repo}:{img_tags[0]}"
img_hash = target.get("id", "")
# Call SYNO.Docker.Image/get — consistent with SYNO.Docker.Container/get
# used elsewhere in this codebase. Pass both name+tag and id defensively
# so DSM can pick whichever shape it accepts.
try:
inspect_params: dict[str, Any] = {
"name": repo,
"tag": img_tags[0] if img_tags else tag,
}
if img_hash:
inspect_params["id"] = img_hash
data = await client.request( data = await client.request(
"SYNO.Docker.Image", "SYNO.Docker.Image",
"get", "get",
params=inspect_params, version=1,
params={"identity": json.dumps(image_id)},
) )
except Exception as e: except Exception as e:
return f"Error inspecting image '{image_id}': {e}" return f"Error inspecting image '{image_id}': {e}"
# DSM may wrap the inspect blob under "details" (like SYNO.Docker.Container/get) if not isinstance(data, dict) or not data:
# or return Docker-engine-style fields at top level. Try both. return f"Image '{image_id}' not found."
details: dict[str, Any] = data.get("details") if isinstance(data, dict) else None
if not isinstance(details, dict):
details = data if isinstance(data, dict) else {}
# Identity — prefer fields from inspect, fall back to list_images entry repo = data.get("image") or "?"
inspect_id = details.get("Id") or img_hash or "unknown" tag = data.get("tag") or "?"
size_val = details.get("Size", target.get("size", 0)) or 0 display_name = f"{repo}:{tag}"
size_str = _human_size(size_val) img_hash = data.get("id") or ""
digest = data.get("digest") or ""
size_val = data.get("size") or 0
virtual_size_val = data.get("virtual_size") or 0
author = data.get("author") or ""
docker_version = data.get("docker_version") or ""
cmd = data.get("cmd") or []
entrypoint = data.get("entrypoint") or []
env_list: list[str] = data.get("env") or []
ports: list[str] = data.get("ports") or []
volumes: list[str] = data.get("volumes") or []
# Created — DSM list_images returns a Unix int; inspect typically returns ISO string lines = [f"Image: {display_name}"]
created_field: Any = details.get("Created") or target.get("created", 0) if img_hash:
if isinstance(created_field, int): lines.append(f" Hash: {img_hash}")
created_str = _format_created(created_field) if digest:
elif isinstance(created_field, str) and created_field: lines.append(f" Digest: {digest}")
# Trim ISO timestamp to date portion if possible if size_val:
created_str = created_field.split("T")[0] lines.append(f" Size: {_human_size(size_val)}")
else: if virtual_size_val and virtual_size_val != size_val:
created_str = "unknown" lines.append(f" Virtual: {_human_size(virtual_size_val)}")
if author:
config: dict[str, Any] = details.get("Config") or {} lines.append(f" Author: {author}")
env_list: list[str] = config.get("Env") or [] if docker_version:
exposed_ports: dict[str, Any] = config.get("ExposedPorts") or {} lines.append(f" Docker: {docker_version}")
entrypoint = config.get("Entrypoint")
cmd = config.get("Cmd")
working_dir = config.get("WorkingDir") or ""
labels: dict[str, Any] = config.get("Labels") or {}
rootfs: dict[str, Any] = details.get("RootFS") or {}
layers: list[Any] = rootfs.get("Layers") or []
lines = [
f"Image: {display_name}",
f" Hash: {inspect_id}",
f" Size: {size_str}",
f" Created: {created_str}",
]
if working_dir:
lines.append(f" Working dir: {working_dir}")
if entrypoint: if entrypoint:
ep_str = ( ep_str = (
@@ -383,52 +325,32 @@ def register_images(mcp: FastMCP, config: AppConfig, client: DsmClient) -> None:
if isinstance(entrypoint, list) if isinstance(entrypoint, list)
else str(entrypoint) else str(entrypoint)
) )
lines.append("")
lines.append(f" Entrypoint: {ep_str}") lines.append(f" Entrypoint: {ep_str}")
if cmd: if cmd:
cmd_str = " ".join(str(x) for x in cmd) if isinstance(cmd, list) else str(cmd) cmd_str = " ".join(str(x) for x in cmd) if isinstance(cmd, list) else str(cmd)
if not entrypoint:
lines.append("")
lines.append(f" Cmd: {cmd_str}") lines.append(f" Cmd: {cmd_str}")
if exposed_ports: if ports:
lines.append("") lines.append("")
lines.append(f"Exposed ports ({len(exposed_ports)}):") lines.append(f"Exposed ports ({len(ports)}):")
for port in exposed_ports: for port in ports:
lines.append(f" {port}") lines.append(f" {port}")
if volumes:
lines.append("")
lines.append(f"Volumes ({len(volumes)}):")
for vol in volumes:
lines.append(f" {vol}")
if env_list: if env_list:
lines.append("") lines.append("")
lines.append(f"Environment ({len(env_list)}):") lines.append(f"Environment ({len(env_list)}):")
for var in env_list: for var in env_list:
lines.append(f" {var}") lines.append(f" {var}")
if layers:
lines.append("")
lines.append(f"Layers ({len(layers)}):")
for layer in layers:
# Layer may be a string hash or a dict with size info
if isinstance(layer, dict):
layer_hash = layer.get("digest") or layer.get("Id") or ""
layer_size = layer.get("size") or layer.get("Size")
short = layer_hash.split(":")[-1][:12] if layer_hash else "?"
if isinstance(layer_size, int) and layer_size > 0:
lines.append(f" {short} {_human_size(layer_size)}")
else:
lines.append(f" {short}")
else:
layer_str = str(layer)
short = layer_str.split(":")[-1][:12] if layer_str else "?"
lines.append(f" {short}")
if labels:
label_items = list(labels.items())
shown = label_items[:5]
lines.append("")
lines.append(f"Labels ({len(label_items)}):")
for key, value in shown:
lines.append(f" {key}={value}")
if len(label_items) > len(shown):
lines.append(f" ... and {len(label_items) - len(shown)} more")
return "\n".join(lines) return "\n".join(lines)
@mcp.tool() @mcp.tool()
+97 -63
View File
@@ -416,35 +416,31 @@ async def test_delete_image_api_error():
# ────────────────────────────────────────────────────────────────────────────── # ──────────────────────────────────────────────────────────────────────────────
# Authoritative DSM SYNO.Docker.Image/get response shape (v1). Top-level
# fields only — there is NO "layers" field on this endpoint, confirmed
# via API capture against the live NAS.
SAMPLE_INSPECT = { SAMPLE_INSPECT = {
"details": { "image": "nginx",
"Id": "sha256:aaaa", "tag": "1.24",
"RepoTags": ["nginx:1.24"], "id": "sha256:aaaa1234567890abcdef",
"Size": 50 * 1024 * 1024, "digest": "sha256:digestabcdef",
"Created": "2024-01-01T00:00:00Z", "size": 50 * 1024 * 1024,
"Config": { "virtual_size": 50 * 1024 * 1024,
"Env": ["NGINX_VERSION=1.24", "PATH=/usr/local/sbin"], "author": "NGINX Team",
"ExposedPorts": {"80/tcp": {}, "443/tcp": {}}, "docker_version": "20.10.0",
"Entrypoint": ["/docker-entrypoint.sh"], "cmd": ["nginx", "-g", "daemon off;"],
"Cmd": ["nginx", "-g", "daemon off;"], "entrypoint": ["/docker-entrypoint.sh"],
"WorkingDir": "/", "env": ["NGINX_VERSION=1.24", "PATH=/usr/local/sbin"],
"Labels": {"maintainer": "NGINX Docker Maintainers"}, "ports": ["80/tcp", "443/tcp"],
}, "volumes": ["/var/cache/nginx"],
"RootFS": {
"Type": "layers",
"Layers": ["sha256:layer1", "sha256:layer2"],
},
}
} }
def _make_inspect_client(inspect_payload=None, images_payload=None): def _make_inspect_client(inspect_payload=None):
"""Build a mock DsmClient that returns SAMPLE_IMAGES for list and inspect_payload for get.""" """Build a mock DsmClient that returns inspect_payload for SYNO.Docker.Image/get."""
client = AsyncMock() client = AsyncMock()
async def mock_request(api, method, **kwargs): async def mock_request(api, method, **kwargs):
if api == "SYNO.Docker.Image" and method == "list":
return images_payload if images_payload is not None else SAMPLE_IMAGES
if api == "SYNO.Docker.Image" and method == "get": if api == "SYNO.Docker.Image" and method == "get":
return inspect_payload if inspect_payload is not None else SAMPLE_INSPECT return inspect_payload if inspect_payload is not None else SAMPLE_INSPECT
return {} return {}
@@ -471,15 +467,19 @@ async def test_inspect_image_by_name_tag():
async def test_inspect_image_by_hash(): async def test_inspect_image_by_hash():
from mcp_synology_container.modules.images import register_images from mcp_synology_container.modules.images import register_images
# Inspect data shaped for redis (sha256:cccc) # Inspect data shaped for redis returned by hash lookup
redis_inspect = { redis_inspect = {
"details": { "image": "redis",
"Id": "sha256:cccc", "tag": "7",
"RepoTags": ["redis:7"], "id": "sha256:ccccredis",
"Size": 30 * 1024 * 1024, "digest": "sha256:rdigest",
"Config": {"Env": [], "ExposedPorts": {}, "Cmd": ["redis-server"]}, "size": 30 * 1024 * 1024,
"RootFS": {"Layers": ["sha256:rlayer1"]}, "virtual_size": 30 * 1024 * 1024,
} "cmd": ["redis-server"],
"entrypoint": [],
"env": [],
"ports": ["6379/tcp"],
"volumes": [],
} }
client = _make_inspect_client(inspect_payload=redis_inspect) client = _make_inspect_client(inspect_payload=redis_inspect)
mcp, tools = make_mock_mcp() mcp, tools = make_mock_mcp()
@@ -490,13 +490,41 @@ async def test_inspect_image_by_hash():
@pytest.mark.asyncio @pytest.mark.asyncio
async def test_inspect_image_not_found(): async def test_inspect_image_uses_identity_param():
"""DSM SYNO.Docker.Image/get expects parameter 'identity' (JSON-encoded), version=1.
Using name/tag/id (the old shape) returned DSM error 114 — this test guards
against regressing to that contract.
"""
import json
from mcp_synology_container.modules.images import register_images from mcp_synology_container.modules.images import register_images
client = _make_inspect_client() client = _make_inspect_client()
mcp, tools = make_mock_mcp() mcp, tools = make_mock_mcp()
register_images(mcp, make_config(), client) register_images(mcp, make_config(), client)
await tools["inspect_image"](image_id="nginx:1.24")
get_calls = [
c for c in client.request.call_args_list if c.args[:2] == ("SYNO.Docker.Image", "get")
]
assert len(get_calls) == 1
call = get_calls[0]
assert call.kwargs.get("version") == 1
params = call.kwargs.get("params") or {}
assert params == {"identity": json.dumps("nginx:1.24")}
@pytest.mark.asyncio
async def test_inspect_image_empty_response_treated_as_not_found():
"""An empty response dict surfaces as a clean 'not found' message."""
from mcp_synology_container.modules.images import register_images
client = _make_inspect_client(inspect_payload={})
mcp, tools = make_mock_mcp()
register_images(mcp, make_config(), client)
result = await tools["inspect_image"](image_id="bogus:latest") result = await tools["inspect_image"](image_id="bogus:latest")
assert "not found" in result assert "not found" in result
@@ -528,7 +556,7 @@ async def test_inspect_image_shows_exposed_ports():
@pytest.mark.asyncio @pytest.mark.asyncio
async def test_inspect_image_shows_layers(): async def test_inspect_image_shows_volumes():
from mcp_synology_container.modules.images import register_images from mcp_synology_container.modules.images import register_images
client = _make_inspect_client() client = _make_inspect_client()
@@ -536,10 +564,7 @@ async def test_inspect_image_shows_layers():
register_images(mcp, make_config(), client) register_images(mcp, make_config(), client)
result = await tools["inspect_image"](image_id="nginx:1.24") result = await tools["inspect_image"](image_id="nginx:1.24")
assert "Layers" in result assert "/var/cache/nginx" in result
# Layer hashes truncated to 12 chars after sha256:
assert "layer1" in result
assert "layer2" in result
@pytest.mark.asyncio @pytest.mark.asyncio
@@ -557,31 +582,42 @@ async def test_inspect_image_shows_entrypoint_cmd():
@pytest.mark.asyncio @pytest.mark.asyncio
async def test_inspect_image_registry_prefixed(): async def test_inspect_image_shows_digest_author_docker_version():
"""Identity-block fields specific to the DSM Image/get response are surfaced."""
from mcp_synology_container.modules.images import register_images
client = _make_inspect_client()
mcp, tools = make_mock_mcp()
register_images(mcp, make_config(), client)
result = await tools["inspect_image"](image_id="nginx:1.24")
assert "sha256:digestabcdef" in result
assert "NGINX Team" in result
assert "20.10.0" in result
@pytest.mark.asyncio
async def test_inspect_image_registry_prefixed():
"""A registry-prefixed identifier like 'ghcr.io/foo/bar:v1' is passed through verbatim
in the JSON-encoded identity parameter."""
import json
from mcp_synology_container.modules.images import register_images from mcp_synology_container.modules.images import register_images
registry_images = {
"images": [
{
"id": "sha256:dddd",
"repository": "ghcr.io/foo/bar",
"tags": ["v1"],
"size": 100 * 1024 * 1024,
"created": 1700000000,
"upgradable": False,
}
]
}
registry_inspect = { registry_inspect = {
"details": { "image": "ghcr.io/foo/bar",
"Id": "sha256:dddd", "tag": "v1",
"RepoTags": ["ghcr.io/foo/bar:v1"], "id": "sha256:dddd",
"Size": 100 * 1024 * 1024, "digest": "sha256:gdigest",
"Config": {"Env": [], "ExposedPorts": {}, "Cmd": ["/app"]}, "size": 100 * 1024 * 1024,
"RootFS": {"Layers": ["sha256:rlayer1"]}, "virtual_size": 100 * 1024 * 1024,
} "cmd": ["/app"],
"entrypoint": [],
"env": [],
"ports": [],
"volumes": [],
} }
client = _make_inspect_client(inspect_payload=registry_inspect, images_payload=registry_images) client = _make_inspect_client(inspect_payload=registry_inspect)
mcp, tools = make_mock_mcp() mcp, tools = make_mock_mcp()
register_images(mcp, make_config(), client) register_images(mcp, make_config(), client)
@@ -589,14 +625,14 @@ async def test_inspect_image_registry_prefixed():
assert "ghcr.io/foo/bar" in result assert "ghcr.io/foo/bar" in result
assert "v1" in result assert "v1" in result
# Verify the get call used the full registry-prefixed repository name # The full identifier (with ':') is JSON-encoded into a single string —
# registry-prefixed names are not split into name + tag.
get_calls = [ get_calls = [
c for c in client.request.call_args_list if c.args[:2] == ("SYNO.Docker.Image", "get") c for c in client.request.call_args_list if c.args[:2] == ("SYNO.Docker.Image", "get")
] ]
assert get_calls, "inspect_image must call SYNO.Docker.Image/get" assert get_calls
params = get_calls[0].kwargs.get("params") or {} params = get_calls[0].kwargs.get("params") or {}
assert params.get("name") == "ghcr.io/foo/bar" assert params.get("identity") == json.dumps("ghcr.io/foo/bar:v1")
assert params.get("tag") == "v1"
@pytest.mark.asyncio @pytest.mark.asyncio
@@ -607,8 +643,6 @@ async def test_inspect_image_api_error():
client = AsyncMock() client = AsyncMock()
async def mock_request(api, method, **kwargs): async def mock_request(api, method, **kwargs):
if api == "SYNO.Docker.Image" and method == "list":
return SAMPLE_IMAGES
if api == "SYNO.Docker.Image" and method == "get": if api == "SYNO.Docker.Image" and method == "get":
raise SynologyError("inspect failed", code=120) raise SynologyError("inspect failed", code=120)
return {} return {}
Generated
+1 -1
View File
@@ -362,7 +362,7 @@ wheels = [
[[package]] [[package]]
name = "mcp-synology-container" name = "mcp-synology-container"
version = "0.4.1" version = "0.4.2"
source = { editable = "." } source = { editable = "." }
dependencies = [ dependencies = [
{ name = "click" }, { name = "click" },