Renames the package from `mcp-cucm-axl` to `mcaxl` to fit the
operator's mc<interface> naming convention (mcusb, mcaxl, …),
and scrubs Bingham-specific defaults so the package works for
anyone, anywhere.
Rename:
- pyproject.toml: name, scripts entry point, description
- src/mcp_cucm_axl/ → src/mcaxl/ (git mv preserves history)
- All Python imports updated via sed
- Cache directory: ~/.cache/mcp-cucm-axl/ → ~/.cache/mcaxl/
- Log prefix [mcp-cucm-axl] → [mcaxl]
- Package version lookup: importlib.metadata.version("mcaxl")
- .mcp.json command updated to invoke `mcaxl` script
- All 155 tests pass under the new name (verified)
Bingham-specific scrubs:
- docs_loader._DEFAULT_INDEX_DIR: hardcoded /home/rpm/bingham/...
path removed; defaults to None. Operators set CISCO_DOCS_INDEX_PATH
env var; without it, prompts gracefully degrade with a fallback
notice instructing the LLM to use the cisco-docs MCP search_docs
tool instead.
- prompts/_common.docs_or_empty_msg: removed the explicit
/home/rpm/bingham/... path from the fallback message text.
- server.py: removed dead-code copy of _docs_or_empty_msg() that
was leftover from before the prompts package extraction.
- README.md: completely rewritten as a public-facing readme. Lead
paragraph names CUCM as the target platform, install instructions
cover uvx / pip / Claude Code MCP add. Recommends cisco-cucm-mcp
as the operations counterpart.
PyPI metadata:
- Initial CalVer version: 2026.04.27
- License: MIT (LICENSE file added)
- Project URLs: Homepage / Source / Issues / Changelog all point
at git.supported.systems/mcp/mcaxl (newly-created Gitea repo
in the mcp/ org for PyPI releases)
- Classifiers: Beta / Telecommunications Industry / Topic:Telephony
- Keywords: mcp, cisco, cucm, axl, risport, voip, sip, audit
- sdist excludes: CLAUDE.md, .env*, axlsqltoolkit.zip, audits/,
tests/, pytest/ruff caches. Verified clean: wheel ships only the
mcaxl/ source tree + LICENSE + METADATA + entry_points.
CHANGELOG.md added with a 2026.04.27 initial-release entry,
documenting tool/prompt counts, structural read-only guarantees,
Hamilton review closure, live-cluster verification, and known
limitations.
Build verification:
- `uv build` produces clean wheel + sdist
- Wheel: 22 source files, 195KB total, no Bingham-specific files
- Sdist excludes verified: no CLAUDE.md, no axlsqltoolkit.zip
- Entry point: `mcaxl = mcaxl.server:main`
- Package installs as mcaxl==2026.4.27
180 lines
6.6 KiB
Python
180 lines
6.6 KiB
Python
"""Tests for the SQLite TTL cache."""
|
|
|
|
import time
|
|
from pathlib import Path
|
|
|
|
import pytest
|
|
|
|
from mcaxl.cache import AxlCache
|
|
|
|
|
|
@pytest.fixture
|
|
def cache(tmp_path: Path) -> AxlCache:
|
|
return AxlCache(tmp_path / "test.sqlite", default_ttl=60)
|
|
|
|
|
|
def test_set_and_get(cache: AxlCache):
|
|
cache.set("getCCMVersion", {}, {"version": "15.0.1"})
|
|
assert cache.get("getCCMVersion", {}) == {"version": "15.0.1"}
|
|
|
|
|
|
def test_miss_returns_none(cache: AxlCache):
|
|
assert cache.get("nonexistent", {"x": 1}) is None
|
|
|
|
|
|
def test_kwargs_order_independent(cache: AxlCache):
|
|
cache.set("listPhone", {"name": "SEP1", "limit": 10}, {"rows": ["p1"]})
|
|
# Different order should still hit
|
|
assert cache.get("listPhone", {"limit": 10, "name": "SEP1"}) == {"rows": ["p1"]}
|
|
|
|
|
|
def test_different_args_different_keys(cache: AxlCache):
|
|
cache.set("listPhone", {"name": "SEP1"}, {"rows": ["a"]})
|
|
cache.set("listPhone", {"name": "SEP2"}, {"rows": ["b"]})
|
|
assert cache.get("listPhone", {"name": "SEP1"}) == {"rows": ["a"]}
|
|
assert cache.get("listPhone", {"name": "SEP2"}) == {"rows": ["b"]}
|
|
|
|
|
|
def test_expired_entries_not_returned(tmp_path: Path):
|
|
c = AxlCache(tmp_path / "ttl.sqlite", default_ttl=60)
|
|
c.set("foo", {}, {"x": 1}, ttl=1)
|
|
time.sleep(1.1)
|
|
assert c.get("foo", {}) is None
|
|
|
|
|
|
def test_ttl_zero_disables_caching(tmp_path: Path):
|
|
c = AxlCache(tmp_path / "off.sqlite", default_ttl=0)
|
|
c.set("foo", {}, {"x": 1})
|
|
# default_ttl=0 means writes are no-ops
|
|
assert c.get("foo", {}) is None
|
|
|
|
|
|
def test_stats_reports_breakdown(cache: AxlCache):
|
|
cache.set("listPhone", {}, {"x": 1})
|
|
cache.set("listPhone", {"a": 1}, {"x": 2})
|
|
cache.set("getCCMVersion", {}, {"v": "15"})
|
|
|
|
stats = cache.stats()
|
|
assert stats["live_entries"] == 3
|
|
assert stats["by_method"]["listPhone"] == 2
|
|
assert stats["by_method"]["getCCMVersion"] == 1
|
|
|
|
|
|
def test_clear_all(cache: AxlCache):
|
|
cache.set("a", {}, "x")
|
|
cache.set("b", {}, "y")
|
|
deleted = cache.clear()
|
|
assert deleted == 2
|
|
assert cache.stats()["live_entries"] == 0
|
|
|
|
|
|
def test_clear_by_pattern(cache: AxlCache):
|
|
cache.set("listPhone", {}, "p")
|
|
cache.set("listLine", {}, "l")
|
|
cache.set("getCCMVersion", {}, "v")
|
|
deleted = cache.clear("list*")
|
|
assert deleted == 2
|
|
assert cache.get("getCCMVersion", {}) == "v"
|
|
|
|
|
|
def test_purge_expired(tmp_path: Path):
|
|
c = AxlCache(tmp_path / "p.sqlite", default_ttl=60)
|
|
c.set("a", {}, "x", ttl=1)
|
|
c.set("b", {}, "y", ttl=60)
|
|
time.sleep(1.1)
|
|
purged = c.purge_expired()
|
|
assert purged == 1
|
|
assert c.stats()["live_entries"] == 1
|
|
|
|
|
|
class TestClusterIsolation:
|
|
"""Hamilton review CRITICAL #2: cache key omitted cluster identity.
|
|
|
|
Prior to the fix, `AXL_URL` swap (test → prod, or one cluster to another)
|
|
served stale results from cluster A as if from cluster B. The cache
|
|
couldn't tell the data came from a different mission. Now each cache
|
|
handle is bound to a cluster_id, and entries from a different cluster
|
|
must miss.
|
|
"""
|
|
|
|
def test_different_cluster_ids_isolate_get(self, tmp_path: Path):
|
|
# Both caches point at the same DB file, but bound to different
|
|
# cluster IDs. A's writes must not be visible to B.
|
|
db = tmp_path / "shared.sqlite"
|
|
a = AxlCache(db, default_ttl=60, cluster_id="cluster-A")
|
|
b = AxlCache(db, default_ttl=60, cluster_id="cluster-B")
|
|
|
|
a.set("getCCMVersion", {}, {"version": "12.5"})
|
|
assert a.get("getCCMVersion", {}) == {"version": "12.5"}
|
|
assert b.get("getCCMVersion", {}) is None, (
|
|
"cluster-B must not see cluster-A's cached value"
|
|
)
|
|
|
|
def test_same_cluster_id_shares_cache(self, tmp_path: Path):
|
|
# Two handles with the SAME cluster_id should share results.
|
|
db = tmp_path / "shared.sqlite"
|
|
a = AxlCache(db, default_ttl=60, cluster_id="cluster-X")
|
|
a.set("listPhone", {"name": "SEP1"}, {"rows": ["one"]})
|
|
b = AxlCache(db, default_ttl=60, cluster_id="cluster-X")
|
|
assert b.get("listPhone", {"name": "SEP1"}) == {"rows": ["one"]}
|
|
|
|
def test_cluster_id_in_stats(self, tmp_path: Path):
|
|
c = AxlCache(tmp_path / "s.sqlite", default_ttl=60, cluster_id="cluster-Y")
|
|
c.set("getCCMVersion", {}, {"v": "15"})
|
|
stats = c.stats()
|
|
assert stats.get("cluster_id") == "cluster-Y", (
|
|
"stats must surface cluster_id so operators can verify which cluster they're caching"
|
|
)
|
|
|
|
def test_no_cluster_id_still_works_legacy(self, tmp_path: Path):
|
|
# Backward compat: no cluster_id keeps the old (but now risky) shape.
|
|
# The cache still functions; we just don't get isolation.
|
|
c = AxlCache(tmp_path / "legacy.sqlite", default_ttl=60)
|
|
c.set("x", {}, "y")
|
|
assert c.get("x", {}) == "y"
|
|
|
|
def test_clear_only_affects_current_cluster(self, tmp_path: Path):
|
|
db = tmp_path / "shared.sqlite"
|
|
a = AxlCache(db, default_ttl=60, cluster_id="cluster-A")
|
|
b = AxlCache(db, default_ttl=60, cluster_id="cluster-B")
|
|
a.set("x", {}, "from-A")
|
|
b.set("x", {}, "from-B")
|
|
deleted = a.clear()
|
|
assert deleted == 1, "clear() must only affect this cluster's entries"
|
|
assert b.get("x", {}) == "from-B", "cluster-B's entry must survive A's clear"
|
|
|
|
def test_migrate_legacy_database(self, tmp_path: Path):
|
|
"""A cache database created before the cluster_id fix must
|
|
upgrade transparently — no `no such column` error on next INSERT.
|
|
"""
|
|
import sqlite3
|
|
db = tmp_path / "legacy.sqlite"
|
|
# Manually create the OLD schema (no cluster_id column)
|
|
conn = sqlite3.connect(db)
|
|
conn.executescript(
|
|
"""
|
|
CREATE TABLE axl_cache (
|
|
cache_key TEXT PRIMARY KEY,
|
|
method TEXT NOT NULL,
|
|
args_json TEXT NOT NULL,
|
|
result_json TEXT NOT NULL,
|
|
created_at REAL NOT NULL,
|
|
expires_at REAL NOT NULL
|
|
);
|
|
INSERT INTO axl_cache VALUES
|
|
('legacy-key', 'oldMethod', '{}', '"old-value"', 0, 9999999999);
|
|
"""
|
|
)
|
|
conn.commit()
|
|
conn.close()
|
|
|
|
# Open with the new code — must not raise, must add the column
|
|
c = AxlCache(db, default_ttl=60, cluster_id="new-cluster")
|
|
# The new client should NOT see the legacy entry (it has no cluster_id)
|
|
# — this is the cautious behavior; legacy entries are isolated to the
|
|
# "unknown cluster" bucket.
|
|
assert c.get("oldMethod", {}) is None
|
|
# And it must be able to write/read its own entries
|
|
c.set("newMethod", {"a": 1}, "new-value")
|
|
assert c.get("newMethod", {"a": 1}) == "new-value"
|