A local context engine for AI coding agents
Project description
Nomi
Nomi is a local context engine infrastructure designed to sit between developer codebases and AI coding agents. It reduces token usage and latency by preventing agents from repeatedly loading full files and repositories into LLM prompts. Instead, Nomi extracts and serves minimal structured code context.
Overview
Modern AI coding tools repeatedly send large portions of repositories to LLMs, resulting in:
- Massive token waste - 10,000+ tokens sent for small changes
- Slow responses - Large context increases latency
- Repeated context loading - Same files loaded every turn
- Unnecessary API costs - Expensive at scale
Nomi solves this by acting as a local context engine.
┌─────────────────────────────────────────────────────────────┐
│ NOMI ARCHITECTURE │
└─────────────────────────────────────────────────────────────┘
Codebase
│
▼
┌──────────────┐ ┌──────────────┐ ┌──────────────┐
│ File Watcher │───▶│ Parser │───▶│ Symbol Index │
└──────────────┘ │ (Tree-sitter)│ └──────────────┘
└──────────────┘ │
▼
┌──────────────────────────────────────┐
│ Dependency Graph │
│ (CALLS, IMPORTS, DEFINES, │
│ IMPLEMENTS relationships) │
└──────────────────────────────────────┘
│
▼
┌──────────────────────────────────────┐
│ Context Builder │
│ ┌──────────────┐ │
│ │ 1. Query │ │
│ │ Resolution│ │
│ ├──────────────┤ │
│ │ 2. Focal │ │
│ │ Retrieval │ │
│ ├──────────────┤ │
│ │ 3. Dependency│ │
│ │ Expansion │ │
│ ├──────────────┤ │
│ │ 4. Skeleton │ │
│ │ Creation │ │
│ ├──────────────┤ │
│ │ 5. Bundle │ │
│ │ Assembly │ │
│ └──────────────┘ │
└──────────────────────────────────────┘
│
▼
┌──────────────────────────────────────┐
│ Context Bundle (10-50x smaller) │
│ • Focal code (full implementation) │
│ • Dependency skeletons (signatures) │
│ • Repository map (PageRank-based) │
└──────────────────────────────────────┘
│
▼
┌──────────────────────────────────────┐
│ Interfaces │
│ • HTTP API (port 8345) │
│ • MCP Server (Model Context Protocol)│
└──────────────────────────────────────┘
│
▼
┌──────────────────────────────────────┐
│ Coding Agents / LLMs │
└──────────────────────────────────────┘
Features
Core Capabilities
- 🔍 AST-based Parsing - Tree-sitter for fast, error-tolerant parsing of Python, TypeScript, Go, and more
- 📊 Symbol Indexing - Fast SQLite-backed symbol search with fuzzy matching and prefix search
- 🕸️ Dependency Graphs - Directed graphs mapping CALLS, IMPORTS, DEFINES, and IMPLEMENTS relationships
- 🗜️ Context Compression - Reduces tokens by 10-50x through intelligent skeletonization
- 🗺️ Repository Maps - PageRank-based project structure overview (~1000 tokens)
- 🔌 MCP Integration - Model Context Protocol for seamless agent integration
- 👁️ File Watching - Real-time incremental updates with debouncing
- 🌐 Local API - HTTP endpoints for external tools and IDEs
Performance Targets
| Metric | Target | Status |
|---|---|---|
| Token Reduction | 10-50x | ✅ Implemented |
| Context Retrieval Latency | <100ms | ✅ Implemented |
| Indexing Speed (10k LOC) | <1 second | ✅ Implemented |
| Incremental Updates | <50ms | ✅ Implemented |
Installation
From PyPI
pip install nomi-context
Then use the nomi command:
nomi init
nomi start
From Source
git clone https://github.com/shyamolkonwar/nomi.git
cd nomi
pip install -e .
Quick Start
1. Initialize Nomi in Your Project
cd your-project
nomi init
This creates a .nomi.json configuration file:
{
"languages": ["python", "typescript"],
"watch": true,
"enable_mcp": true,
"ignore_patterns": [
".git",
"node_modules",
"dist",
"build",
"__pycache__",
".venv"
]
}
2. Start the Daemon
nomi start
The daemon will:
- Scan your repository
- Build the symbol index
- Construct the dependency graph
- Start watching for file changes
- Start the API server on port 8345
- Start the MCP server (if enabled)
Output:
✓ Nomi daemon started
PID: 12345
API: http://localhost:8345
Health: http://localhost:8345/health
MCP: stdio enabled
Indexing:
Files: 1,247
Symbols: 8,932
Duration: 0.8s
3. Search Symbols
# Search for a symbol
nomi search "create_user"
# Output:
# NAME | KIND | FILE | LINE | SCORE
# ---------------|----------|-------------------|------|-------
# create_user | function | auth/service.py | 42 | 1.00
# create_user | method | models/user.py | 15 | 0.85
# create_user_view| function | views/auth.py | 23 | 0.72
4. Build Context
# Build context for a query
nomi context "refactor create_user to add validation"
# Output shows:
# - Focal code (full implementation of create_user)
# - Dependencies (signatures only)
# - Repository map (top-level structure)
# - Token count and timing
Testing & Usage Guide
This guide shows you how to test Nomi and integrate it with AI coding agents.
1. Installation
On a fresh project or virtual environment:
pip install nomi-context
Verify installation:
nomi --help
Expected: CLI help should appear with commands (init, start, stop, status, search, context).
2. Initialize Nomi in a Repository
Go to any project you want to test:
cd my-project
nomi init
This creates:
.nomi.json- Configuration file.nomi/- Storage directory for indexes
Example config created:
{
"languages": ["python", "typescript"],
"watch": true,
"enable_mcp": true,
"ignore_patterns": [
".git",
"node_modules",
"dist",
"build",
"__pycache__",
".venv"
]
}
3. Start the Nomi Daemon
Run:
nomi start
This starts:
- Repository scanner
- Parser (Tree-sitter)
- Symbol index
- Dependency graph builder
- MCP server
- Local API server (port 8345)
Expected output:
✓ Nomi daemon started
PID: 12345
API: http://localhost:8345
Health: http://localhost:8345/health
MCP: stdio enabled
Indexing:
Files: 1,247
Symbols: 8,932
Duration: 0.8s
Leave this running in the background.
4. Verify the Local API
Open your browser or use curl:
curl http://localhost:8345/repo-map
Or visit: http://localhost:8345/repo-map
If the server works, it returns JSON describing your repository structure:
{
"modules": [
{
"name": "auth",
"symbols": ["create_user", "login_user", "validate_token"]
},
{
"name": "payments",
"symbols": ["charge_card", "refund_payment"]
}
]
}
This confirms Nomi is running correctly.
5. Enable Nomi as an MCP Tool
Connect Nomi to Claude Code or other agents that support MCP.
For Claude Code:
Find your Claude config (usually at ~/.config/claude/mcp_servers.json) and add:
{
"servers": {
"nomi": {
"command": "nomi",
"args": ["mcp"]
}
}
}
This tells Claude Code: when tools are needed, start the Nomi MCP server.
For Cursor IDE:
Add to Cursor settings:
{
"mcpServers": {
"nomi": {
"command": "nomi",
"args": ["mcp"]
}
}
}
6. Restart Your AI Agent
Restart Claude Code, Cursor, or your agent so it loads the MCP tools.
You should see:
Loaded MCP tools:
- nomi.get_repo_map
- nomi.search_symbol
- nomi.get_symbol_context
- nomi.expand_dependencies
7. Test With a Real Prompt
Ask your agent to refactor something:
Refactor the create_user function to add email validation.
The agent should internally:
- Call
nomi.search_symbol("create_user") - Call
nomi.get_symbol_context("create_user") - Call
nomi.expand_dependencies("create_user")
Instead of reading entire files, it gets minimal context.
8. Manual API Testing
Test the API directly with curl:
# Search symbols
curl http://localhost:8345/symbol/search \
-X POST \
-H "Content-Type: application/json" \
-d '{"query": "create_user", "limit": 5}'
# Get symbol context
curl http://localhost:8345/context \
-X POST \
-H "Content-Type: application/json" \
-d '{"query": "refactor create_user"}'
# Get repository status
curl http://localhost:8345/repo/status
9. Test With Other Agents
Besides Claude Code, Nomi works with:
- Cursor IDE - Add MCP server in settings
- Continue - Add to config.json
- OpenCode - Add to agent configuration
All support MCP-style tools.
10. Recommended First Test Repo
Use a simple project for testing:
test-project/
├ auth.py (create_user, login_user)
├ payments.py (charge_card, refund)
└ main.py (entry point)
This makes it easy to verify context retrieval.
11. Debugging Tips
If the agent doesn't call Nomi:
# Check daemon status
nomi status
# Check logs
tail -f ~/.nomi/logs/nomi.log
# Test API directly
curl http://localhost:8345/health
# Restart daemon
nomi stop
nomi start
12. Benchmark Token Reduction
Measure the improvement:
Without Nomi:
- Agent reads auth.py (2000 lines) = ~8,000 tokens
With Nomi:
- Agent reads create_user + 3 dependencies = ~600 tokens
Result: 13x token reduction, faster responses.
13. Troubleshooting
Problem: nomi start fails
- Check Python version:
python --version(need 3.11+) - Check port 8345 is free:
lsof -i :8345 - Check permissions:
ls -la .nomi/
Problem: API returns 500 errors
- Repository not indexed yet (wait 30 seconds after start)
- Check daemon logs:
nomi status - Restart daemon:
nomi stop && nomi start
Problem: Agent doesn't use Nomi
- Verify MCP config is correct
- Check agent supports MCP tools
- Restart agent after config change
API Usage
REST API Endpoints
# Get repository map
GET http://localhost:8345/repo-map
# Search symbols
POST http://localhost:8345/symbol/search
Content-Type: application/json
{
"query": "create_user",
"limit": 10
}
# Build context
POST http://localhost:8345/context
Content-Type: application/json
{
"query": "refactor create_user",
"max_tokens": 4000,
"dependency_depth": 1
}
# Get symbol by name
GET http://localhost:8345/symbol/create_user
# Get repository status
GET http://localhost:8345/repo/status
MCP Tools
Nomi exposes tools via Model Context Protocol:
| Tool | Description |
|---|---|
get_repo_map |
Returns high-level project structure |
search_symbol |
Finds symbols matching query |
get_symbol_context |
Returns full implementation of symbol |
expand_dependencies |
Returns related code units up to N hops |
build_context |
Returns complete context bundle |
Example MCP usage:
# Agent calls MCP tool
result = await mcp_client.call_tool(
"get_symbol_context",
{"symbol_name": "create_user", "include_dependencies": true}
)
Architecture
Core Components
1. Parser (nomi/core/parser/)
- TreeSitterEngine - Wrapper around tree-sitter parsers
- ASTExtractor - Extracts functions, classes, methods from AST
- NodeMapper - Maps AST nodes to CodeUnit models
- Query Files - Language-specific .scm query files
2. Symbol Index (nomi/core/index/)
- SymbolIndex - Manages indexing operations
- SymbolLookup - Fast exact/prefix/pattern lookups
- SymbolSearch - Fuzzy matching with relevance scoring
3. Dependency Graph (nomi/core/graph/)
- DependencyGraph - Graph construction and management
- EdgeBuilder - Creates CALLS, IMPORTS, DEFINES, IMPLEMENTS edges
- GraphTraversal - BFS/DFS traversal, shortest path, cycle detection
4. Context Builder (nomi/core/context/)
- ContextBuilder - 5-stage retrieval pipeline
- ContextResolver - Resolves queries to symbols
- ContextBundle - Structured output for LLM prompts
5. Compression (nomi/core/compression/)
- Skeletonizer - Removes bodies, keeps signatures
- TokenBudget - Manages token allocation
- ContextPruner - Tiered pruning strategy
6. Repository Map (nomi/repo_map/)
- RepoMapBuilder - PageRank-based importance scoring
- ModuleGraph - Module-level dependency analysis
Data Models
CodeUnit
class CodeUnit:
id: str # "repo/path/file.py:symbol_name"
unit_kind: UnitKind # FUNCTION, CLASS, METHOD, INTERFACE, MODULE
file_path: str
byte_range: Tuple[int, int]
line_range: Tuple[int, int]
signature: str # Declaration without body
body: str # Full implementation
dependencies: List[str] # Symbol IDs this unit depends on
docstring: Optional[str]
language: str
DependencyEdge
class DependencyEdge:
source_id: str # Calling/importing unit
target_id: str # Called/imported unit
edge_type: EdgeType # CALLS, IMPORTS, DEFINES, IMPLEMENTS
Storage
- SQLite - Symbol index and dependency graph
- File-based - Configuration (.nomi.json)
- In-memory - Runtime caches
Configuration
.nomi.json
{
"languages": ["python", "typescript", "go"],
"watch": true,
"enable_mcp": true,
"ignore_patterns": [
".git",
"node_modules",
"dist",
"build",
"vendor",
".cache",
"__pycache__",
".venv",
"venv"
]
}
| Option | Type | Default | Description |
|---|---|---|---|
languages |
List[str] | Auto-detected | Programming languages to index |
watch |
bool | true | Enable file watching |
enable_mcp |
bool | true | Enable MCP server |
ignore_patterns |
List[str] | [...] | Patterns to exclude from indexing |
Commands
# Initialize Nomi in current directory
nomi init
# Start the daemon
nomi start [--port 8345] [--background]
# Stop the daemon
nomi stop
# Check daemon status
nomi status
# Search symbols
nomi search <query> [--limit 10] [--format table|json]
# Build context
nomi context <query> [--max-tokens 4000] [--depth 1]
Development
Setup
git clone https://github.com/shyamolkonwar/nomi.git
cd nomi
python -m venv .venv
source .venv/bin/activate # On Windows: .venv\Scripts\activate
pip install -e ".[dev]"
Testing
# Run integration tests
python tests/test_integration.py
# Run all tests
pytest tests/
# Run with coverage
pytest --cov=nomi tests/
Linting
# Format code
black nomi/ cli/ tests/ --line-length 120
# Check linting
flake8 nomi/ cli/ tests/ --max-line-length=120
# Type checking
mypy nomi/ --ignore-missing-imports
Performance Benchmarks
Tested on repositories of various sizes:
| Repository Size | Index Time | Memory | Symbols |
|---|---|---|---|
| 1,000 LOC | 0.3s | 15 MB | 150 |
| 10,000 LOC | 0.8s | 45 MB | 1,200 |
| 100,000 LOC | 4.2s | 180 MB | 8,500 |
| 1,000,000 LOC | 35s | 850 MB | 65,000 |
Context retrieval latency:
- Symbol search: <10ms
- Dependency expansion (depth=1): <50ms
- Full context build: <100ms
Contributing
- Fork the repository
- Create a feature branch (
git checkout -b feature/amazing-feature) - Make your changes
- Run tests (
pytest) - Run linters (
black,flake8,mypy) - Commit your changes (
git commit -m 'Add amazing feature') - Push to the branch (
git push origin feature/amazing-feature) - Open a Pull Request
Code Style
- Line length: 120 characters
- Use Black for formatting
- Use type hints everywhere
- Follow PEP 8
- Write docstrings for public APIs
Roadmap
Completed (MVP)
- ✅ AST-based parsing (Tree-sitter)
- ✅ Symbol indexing (SQLite)
- ✅ Dependency graph
- ✅ Context compression (skeletonization)
- ✅ Repository map (PageRank)
- ✅ File watching
- ✅ HTTP API
- ✅ MCP integration
- ✅ CLI
Planned
- 🚧 Semantic search (vector embeddings)
- 🚧 Program slicing
- 🚧 IDE integrations (VSCode, JetBrains)
- 🚧 Multi-repository graphs
- 🚧 Streaming context
- 🚧 Web interface
License
This project is licensed under the MIT License - see the LICENSE file for details.
Acknowledgments
- Tree-sitter - Incremental parsing library
- SCIP - Semantic Code Intelligence Protocol
- MCP - Model Context Protocol
- Aider - Inspiration for repository maps
Support
- 🐛 Issues
- 💬 Discussions
Made with ❤️ for the AI coding community
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file nomi_context-0.1.5.tar.gz.
File metadata
- Download URL: nomi_context-0.1.5.tar.gz
- Upload date:
- Size: 114.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d13e21856d4a5ca78eef23aaf0a79c775374191dfb6f0cc872100b02a0e15d72
|
|
| MD5 |
07cc2409a316e504f011054ae1cc670a
|
|
| BLAKE2b-256 |
356d98941bd3094a6215977ebfc075154646747dce92a50220f3418b4bf17be9
|
File details
Details for the file nomi_context-0.1.5-py3-none-any.whl.
File metadata
- Download URL: nomi_context-0.1.5-py3-none-any.whl
- Upload date:
- Size: 144.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5187d8fce4d9efaf5ec2bfe55843c62429b500ce7b5c2d805608679921e6ff8a
|
|
| MD5 |
d70e122604c5981c93440a814b685b53
|
|
| BLAKE2b-256 |
1f5f86b92f616fac063fc16b2fc855ba4894a36877db82fb607d2497e466a830
|