ism-mcp
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@ism-mcpwhat controls apply to multi-factor authentication?"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
ism-mcp
Agent-friendly query layer over the ASD Information Security Manual, served via MCP.
The ISM PDF is ~700 pages and does not fit in a model context window. This MCP server parses the official Cloud Controls Matrix XLSX into a local SQLite database, attaches surrounding-paragraph excerpts from the ISM PDF, and exposes a small set of typed lookup tools so that an agent (Claude Code, Codex, Cursor, etc.) can interrogate the ISM without re-reading the source documents.
Status
Single-tenant and local: SQLite storage, stdio-transport MCP server, no auth, no network listener. Suitable for local and per-project use, not multi-tenant or networked deployment.
Install
Requires uv and Python 3.14+.
git clone https://github.com/samueldudley/ism-mcp.git
cd ism-mcp
uv syncIngest the ISM
Download the latest:
Cloud controls matrix template (XLSX): https://www.cyber.gov.au/resources-business-and-government/essential-cyber-security/ism
Information security manual (PDF): same page
Then ingest:
uv run ism-mcp ingest \
--xlsx "Cloud controls matrix template (March 2026).xlsx" \
--pdf "Information security manual (March 2026).pdf" \
--revision 2026-03The database lands at ~/.local/share/ism-mcp/ism.db by default. Override with --db PATH. Run ism-mcp ingest --help for the full flag list.
The first ingest downloads the embedding model once (see First-run network requirement). Pass --no-embeddings to skip it and fall back to lexical-only ranking.
Re-run with a new XLSX / PDF on each quarterly revision. The ingester drops and recreates the schema, so there is no migration to worry about.
Use as a Claude Code MCP server
Add to your Claude Code MCP configuration:
{
"mcpServers": {
"ism": {
"command": "uv",
"args": ["--project", "/path/to/ism-mcp", "run", "ism-mcp", "serve"]
}
}
}Restart Claude Code. The tools listed under MCP tools become available, led by ism_applicable for ranked discovery.
Adopt in a project
ism-mcp install writes everything a teammate needs into a consumer repo. The database is committed into the repo, so a clone has data without a local ingest.
uv run ism-mcp install --project /path/to/consumer-repoThis writes, all idempotent on re-run:
.mcp.jsonwith theismserver entry. Other servers in the file are kept.A managed block in
CLAUDE.mdbetween<!-- ism-mcp:begin -->and<!-- ism-mcp:end -->, telling agents when to consult the server..ism-coverage.tomlscaffolded from the template, only if absent. An existing manifest is never overwritten..ism/ism.db, the database, refreshed on each run.
The default uvx mode launches the server with uvx --from git+<repo>@<rev> ism-mcp serve. --repo defaults to this checkout's origin remote and --rev to its short HEAD, so the entry pins a reproducible build. A teammate needs only uv. The first semantic query downloads the embedding model once, then runs offline.
For air-gapped or locked-down environments, --mode docker --image <ref> emits a docker run entry that mounts the committed database. Building and hosting the image is left to you.
The database path uses Claude Code's ${CLAUDE_PROJECT_DIR:-.} expansion, so it resolves to each teammate's project root. Claude Code prompts once per repo to trust a project-scoped server. claude mcp reset-project-choices clears the approval.
Use --dry-run to see the planned writes without changing anything.
Fetching from the published remote
uvx and docker both fetch a pinned source. Point ism-mcp install at the published repository with --repo https://github.com/samueldudley/ism-mcp.git (if you cloned from there, that is already your origin and the default works). To register the server at user scope against the published source:
claude mcp add ism -s user -- uvx --from git+https://github.com/samueldudley/ism-mcp.git@v1.1 ism-mcp serveThe user-scope server keeps the default database at ~/.local/share/ism-mcp/ism.db, which you ingest locally. It carries no ISM_MCP_DB override, since that path is only for project installs where the database travels with the repo.
Use programmatically
from ism_mcp import store, server
conn = store.open_db(server.DEFAULT_DB)
c = store.get_control(conn, "ISM-1781")
print(c.description)
for r in store.search(conn, "session timeout", limit=5):
print(r.identifier, r.topic)Development
uv sync install all deps including dev
uv run pytest run the test suite
./scripts/ci.sh full CI suite (fmt + lint + type + test)
./scripts/ci.sh test single stageThe CI script is the source of truth for what counts as a passing build. Run it before pushing.
MCP tools
Tool | Purpose |
| Hybrid retrieval: rank controls relevant to a free-text description of planned or current work. Recommended default for discovery. |
| Full record for one control by ID. |
| Deterministic FTS5 search. Use when you know the exact term. |
| Controls applicable at NC / OS / P / S / TS. |
| Distinct topic strings. |
| Controls under a topic. |
| Distinct section strings. The vocabulary for the |
| Canonical classification enum plus friendly aliases. |
| Essential Eight maturity levels. |
| Database statistics. |
| Read the project's |
| Create or update one entry with evidence (files, commits, urls, attachments). Validates against the ISM DB and the project filesystem. |
| Return outstanding in-scope controls. With |
Each Control record carries: identifier, guideline, section, topic, revision, updated, description, classification applicability (NC/OS/P/S/TS), maturity applicability (ML1/ML2/ML3), pdf_excerpt, pdf_page.
Discovery for agents
The headline use case is ism_applicable. The agent describes the work in plain language, optionally narrows by classification, maturity, section tags, or repo paths, and gets back a ranked list of relevant controls.
ism_applicable(
work="adding JWT refresh and idle session timeout to our auth flow",
classification="OFFICIAL",
paths=["src/auth/jwt.py", "src/auth/session.py"],
limit=10,
)Returns a ranked list with identifier, topic, section, description, applies, maturity, a normalised RRF score in [0.0, 1.0], and a why list naming the signals that surfaced each result (semantic, lexical, path:<token>). verbose=true adds the PDF excerpt.
Maturity is Essential Eight only.
ML1/ML2/ML3exist for the ~126 of 1081 controls mapped to the Essential Eight Maturity Model, not the wider ISM. Passingmaturity=(here, or in a manifest[scope]) drops every control with no maturity rating, so aPROTECTEDscope collapses from ~966 controls to the ~87 that are also Essential Eight ML2. Leavematurityunset unless you are specifically tracking Essential Eight maturity.
Under the hood: a bge-small-en-v1.5 embedding of the work text is cosine-matched against per-control embeddings, fused with FTS5 BM25 via Reciprocal Rank Fusion, then post-filtered.
First-run network requirement
The first ingest after install downloads the embedding model (~130 MB) to ~/.cache/fastembed/. Subsequent runs are offline. To pre-warm:
uv run python -c "
from fastembed import TextEmbedding
TextEmbedding('BAAI/bge-small-en-v1.5')
"To skip embeddings entirely (offline first run, or for fast iteration during development):
uv run ism-mcp ingest --xlsx PATH --pdf PATH --no-embeddingsWithout embeddings, ism_applicable falls back to lexical-only ranking. Results are still useful but recall on natural-language queries is lower.
Environment variables
Var | Values | Effect |
| path | Override the database location. Default |
|
| Force a specific embedder at server start. |
Project coverage manifest
For projects pursuing IRAP review (or any internal review against the ISM), .ism-coverage.toml at the project root records how each in-scope control is addressed.
schema_version = 1
[scope]
classification = "P"
sections = ["Authentication hardening", "Cryptographic fundamentals"]
[project]
name = "demo-admin"
[controls."ISM-0428"]
status = "covered"
how_met = """
Sessions terminate after 14 min of idle activity, enforced
in the auth middleware. Re-auth requires all original factors.
"""
last_reviewed = 2026-05-28
files = ["src/auth/session.py:42-87"]
commits = ["abc1234"]
[[controls."ISM-0428".attachments]]
path = ".ism-coverage/evidence/ISM-0428/lock-prompt.png"
description = "Admin console at 14:01 showing session-expired modal"[scope] defines the in-scope control set that ism_coverage_gaps measures against. Set classification (and optionally narrow by sections). Do not set maturity unless you are tracking Essential Eight maturity specifically: it filters to the Essential Eight subset and drops every other control from scope (see the maturity note under Discovery for agents).
Recommended layout for binary evidence:
your-repo/
.ism-coverage.toml
.ism-coverage/
evidence/
ISM-0428/
lock-prompt.png
tls-handshake.pcapngA template lives at src/ism_mcp/data/coverage_template.toml if you want to copy and start from a known-good shape. The fields and their allowed values are shown in the example above.
The manifest is machine-managed: ism_coverage_upsert rewrites the whole file, so comments are not preserved. Keep narrative in how_met and evidence in the structured fields rather than in TOML comments.
Architecture
ism-mcp/
src/ism_mcp/
store.py SQLite schema + queries, FTS5 over description/topic/section/guideline
ingest.py XLSX parser (openpyxl) + PDF paragraph extractor (pdfplumber)
retrieve.py cosine search + Reciprocal Rank Fusion
embed.py embedder protocol + fastembed and hash backends
classification.py classification + maturity input normalisation
paths.py repo-path token expansion for query enrichment
coverage.py coverage manifest read, validate, serialise, gaps
install.py consumer-repo install writer
server.py FastMCP server: lookup, discovery, coverage tools, JSON responses
__main__.py CLI: ingest, serve, and install subcommands
data/ path keyword map + coverage template
pyproject.toml uv-managed, hatchling buildSingle SQLite file. One table for controls plus an FTS5 virtual table kept in sync via an AFTER INSERT trigger. A meta table records the ingested revision and source paths for ism_stats.
Known limitations
No incremental updates. Each ingest drops and rebuilds the database.
Single-revision database. No history across ISM revisions. To diff two revisions, ingest into two database paths and diff externally.
No auth on the MCP server. Suitable for local use only.
Licence
MIT. See LICENSE.
This licence covers the code in this repository. The ISM itself is Commonwealth of Australia content published by the ACSC under its own terms. You download and ingest the ISM separately; it is not redistributed here.
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/SamuelDudley/ism-mcp'
If you have feedback or need assistance with the MCP directory API, please join our Discord server