books-catalog
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., "@books-catalogsearch for books by Jane Austen"
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.
mcp-quickstart — a books-catalog MCP server
A clean, well-documented Model Context Protocol (MCP) server, built with FastMCP, that exposes a local SQLite catalog of public-domain books as a set of well-designed tools. It is meant as a reference for how to design MCP tools an LLM can use reliably: clear names, schema-generating type hints, docstrings-as-contracts, validated inputs, and actionable error messages.
The data is a small set of public-domain classics (Austen, Verne, Doyle, Dostoevsky, ...) authored as seed data for this repo — nothing proprietary, and deliberately a generic domain.
What it exposes
Five tools over a 30-book catalog:
Tool | Signature | Purpose |
|
| Substring search over title + author. |
|
| Fetch one book by id; clear error if missing. |
|
| Filter by genre + optional year range. |
|
| Highest-rated, optionally within a genre. |
|
| Totals, per-genre counts, year range, averages. |
Related MCP server: MCP Open Library & File Search Server
Quickstart
git clone <this-repo> && cd mcp-quickstart
python -m pip install -r requirements.txt # installs `mcp` and `pytest`
# Smoke test — list the tools and schemas without a live client:
python -m books_mcp list-tools
# Run the test suite (offline, in-memory catalog):
python -m pytest
# Run the server (stdio transport — this is what an MCP client launches):
python -m books_mcpThe smoke test output:
books-catalog MCP server exposes 5 tools (catalog: 30 books):
- search_books(query, limit)
Search the catalog by a keyword in the title or author.
- get_book(book_id)
Fetch a single book by its numeric id.
- find_books_by_genre(genre, year_from, year_to, limit)
List books in a genre, optionally filtered by publication-year range.
- top_rated_books(limit, genre)
Return the highest-rated books, optionally within a single genre.
- catalog_stats()
Summarize the whole catalog.Example tool result (catalog_stats()):
{
"total_books": 30,
"total_authors": 19,
"genres": { "Adventure": 7, "Fiction": 7, "Gothic": 6, "ScienceFiction": 3,
"Historical": 2, "Mystery": 2, "Romance": 2, "Fantasy": 1 },
"year_range": [1811, 1902],
"avg_pages": 468.4,
"avg_rating": 4.17
}MCP tool-design patterns demonstrated
These are the choices that make tools usable by a model, not just callable:
One job per tool, named as a verb-phrase.
search_books,find_books_by_genre,get_book. A model picks the right tool from the name alone.The docstring is the contract. FastMCP turns each function's signature into the tool's JSON input schema and its docstring into the description the model sees. Both are written for a model reader: what it does, every argument, the return shape, and what raises.
Type hints drive the schema.
book_id: int,genre: str,year_from: int | None = Nonebecome a validated input schema with the right types and optionality — no hand-written JSON schema.Validate and bound every input. Limits are clamped to a hard
MAX_LIMITso a tool call can never trigger an unbounded scan; empty queries and bad ids are rejected up front.Errors are actionable. An unknown genre returns
Unknown genre 'Cyberpunk'. Available genres: Adventure, Fiction, ...; a bad id returns the valid range. The model can correct itself from the message.Outputs are plain JSON-serializable dicts, never database row objects — stable, predictable, and easy for the model to consume.
The data layer is separate from the protocol.
db.pyis a typed, independently testable SQLite layer;server.pyonly does validation + wiring. You can unit-test all the logic without standing up a server.
Connect it to an MCP client
Claude Desktop
Add the server to your claude_desktop_config.json
(macOS: ~/Library/Application Support/Claude/;
Windows: %APPDATA%\Claude\):
{
"mcpServers": {
"books-catalog": {
"command": "python",
"args": ["-m", "books_mcp"],
"cwd": "/absolute/path/to/mcp-quickstart"
}
}
}Restart Claude Desktop; the five tools appear under the server. Then ask
natural-language questions like "What are the top-rated adventure books before
1880?" and the model will call find_books_by_genre / top_rated_books.
Any MCP client
The server speaks the standard MCP stdio transport, so any compliant client
(the MCP Inspector, a custom client using the mcp SDK, etc.) can launch
python -m books_mcp and discover the tools via the normal list_tools /
call_tool handshake.
Storage
By default the server builds an in-memory SQLite catalog from the bundled
seed data at startup — zero setup, always fresh, no files left behind. To
materialize a file-backed catalog (e.g. to inspect it with the sqlite3 CLI or
share it across processes):
python -m books_mcp seed --db catalog.db
BOOKS_DB_PATH=catalog.db python -m books_mcp # serve from the fileProject layout
mcp-quickstart/
├── books_mcp/
│ ├── __init__.py
│ ├── __main__.py # serve | list-tools | seed
│ ├── server.py # FastMCP server + the five tools
│ ├── db.py # typed SQLite data-access layer (no MCP)
│ ├── seed.py # build/seed a file-backed catalog
│ └── data/
│ └── books.json # public-domain seed catalog
└── tests/ # pytest suite, all offline
├── test_db.py # data-layer behavior
└── test_tools.py # tool behavior, validation, and registrationLicense
MIT — see LICENSE.
This server cannot be installed
Maintenance
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/cggazca/mcp-quickstart'
If you have feedback or need assistance with the MCP directory API, please join our Discord server