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
49 lines
1.5 KiB
Python
49 lines
1.5 KiB
Python
"""Hamilton review MINOR #6: `_to_int` silently coerced bad values to None.
|
|
|
|
Sort fields built on `_to_int` returns then defaulted None to 0, which
|
|
jumbled the failover order in the displayed result. Fix: when the conversion
|
|
fails, log to stderr (so an operator can see) and return None — but the
|
|
caller code path that does the sort now uses a stable tie-breaker that
|
|
doesn't silently rewrite real-zero into "no value."
|
|
"""
|
|
|
|
import sys
|
|
from io import StringIO
|
|
|
|
from mcaxl.route_plan import _to_int
|
|
|
|
|
|
def test_to_int_passthrough_normal():
|
|
assert _to_int("5") == 5
|
|
assert _to_int(7) == 7
|
|
|
|
|
|
def test_to_int_none_returns_none_silently():
|
|
"""Real Nones are valid (column not set) — don't log noise for them."""
|
|
captured = StringIO()
|
|
real_stderr = sys.stderr
|
|
sys.stderr = captured
|
|
try:
|
|
assert _to_int(None) is None
|
|
finally:
|
|
sys.stderr = real_stderr
|
|
assert "warning" not in captured.getvalue().lower()
|
|
|
|
|
|
def test_to_int_bad_value_logs_warning():
|
|
"""A non-numeric string from the cluster (data corruption / unexpected
|
|
type) should be loud enough for an operator to notice in stderr."""
|
|
captured = StringIO()
|
|
real_stderr = sys.stderr
|
|
sys.stderr = captured
|
|
try:
|
|
result = _to_int("not-a-number")
|
|
finally:
|
|
sys.stderr = real_stderr
|
|
assert result is None
|
|
output = captured.getvalue()
|
|
assert "not-a-number" in output, (
|
|
f"unexpected non-numeric value should be logged with the offending value; "
|
|
f"got stderr: {output!r}"
|
|
)
|