Metadata-Version: 2.4
Name: mcp-skill
Version: 0.3.0
Summary: Convert any MCP server into an Agent Skill
Project-URL: Homepage, https://github.com/manojbajaj95/mcp-skill
Project-URL: Repository, https://github.com/manojbajaj95/mcp-skill
Project-URL: Issues, https://github.com/manojbajaj95/mcp-skill/issues
License-Expression: MIT
License-File: LICENSE
Keywords: agent,code-generation,fastmcp,llm,mcp,skill
Classifier: Development Status :: 3 - Alpha
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: MIT License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Topic :: Scientific/Engineering :: Artificial Intelligence
Classifier: Topic :: Software Development :: Code Generators
Requires-Python: >=3.10
Requires-Dist: asyncclick>=8.3.0.3
Requires-Dist: click>=8.0
Requires-Dist: fastmcp>=2.0
Requires-Dist: jinja2>=3.1
Requires-Dist: py-key-value-aio[disk,filetree]
Requires-Dist: ruff>=0.15.4
Requires-Dist: ty>=0.0.20
Description-Content-Type: text/markdown

# mcp-skill

Turn any MCP server into a typed Python SDK.

> **Compile MCP tools into code.**

`mcp-skill` introspects an MCP server and generates a Python class where each tool becomes a typed async method.

## Before and After

### Before

Agents call MCP tools through the model loop — one round-trip per tool call:

```
llm.call_tool("web_search_preview", {"query": "..."})
# → model decides next step → calls another tool → model decides again → ...
```

### After

Agents call tools directly in code:

```python
result = await app.web_search_preview(
    objective="find latest news",
    search_queries=["topic X 2026"]
)
# Agent processes result in code — no round-trip back to model
```

## How It Works

```
┌─────────────┐      ┌───────────────┐      ┌────────────────────┐
│  MCP Server  │─────▶│  mcp-skill    │─────▶│  Generated Skill   │
│  (any URL)   │      │  CLI          │      │                    │
│              │      │               │      │  app.py            │
│  Tools:      │      │  1. Connect   │      │  ├─ Typed class    │
│  - search    │      │  2. Introspec │      │  ├─ Async methods  │
│  - fetch     │      │  3. Map types │      │  ├─ Auth + storage │
│  - ...       │      │  4. Generate  │      │  └─ JSON parsing   │
│              │      │  5. Validate  │      │                    │
└─────────────┘      └───────────────┘      │  SKILL.md          │
                                             │  └─ Agent docs     │
                                             └────────────────────┘
```

1. Connects to the MCP server using [fastmcp](https://github.com/jlowin/fastmcp)
2. Introspects all available tools via `list_tools()`
3. Converts each tool's JSON Schema into Python type annotations
4. Generates a typed `App` class where each MCP tool becomes an `async` method
5. Validates the output with `ast.parse` → `ruff` → `ty`
6. Generates `SKILL.md` with tool documentation and usage examples for agents

## Motivation

MCP servers give agents access to tools, but every tool call round-trips through the model — request tool, execute, full result back into context, decide next step. For large payloads or sequential calls, this burns tokens and adds latency.

[Programmatic Tool Calling](https://platform.claude.com/cookbook/tool-use-programmatic-tool-calling-ptc) fixes this: the agent writes code that calls tools directly, without model round-trips per invocation. Fetch, filter, aggregate — all in one code block.

**mcp-skill** makes this possible by compiling any MCP server into a plain Python class. Each tool becomes a typed async method. The agent just writes Python.

```python
from parallel_search.app import ParallelApp

app = ParallelApp(auth="sk-...")
result = await app.web_search_preview(
    objective="find latest news on topic X",
    search_queries=["topic X 2026"]
)
```

## Setup

```bash
# Install with uv
uv pip install -e .

# Or use directly
uv run mcp-skill create --url https://your-mcp-server.com/mcp --auth api-key
```

Requires [uv](https://github.com/astral-sh/uv) and Python 3.10+.

## Usage

```bash
# Interactive mode — prompts for URL, auth type, etc.
mcp-skill create

# Non-interactive mode
mcp-skill create \
  --url https://search-mcp.parallel.ai/mcp \
  --auth api-key \
  --api-key YOUR_KEY \
  --name parallel-search \
  --non-interactive
```

### Generated Output

The skill lands in `.agents/skills/<name>/` as a Python package:

```
.agents/skills/parallel_search/
├── __init__.py
├── app.py          # Typed Python class wrapping the MCP server
└── SKILL.md        # Agent-facing docs, dependencies, and usage
```

Here's what the generated `app.py` looks like:

```python
class ParallelApp:

    def __init__(self, url: str = "https://...", auth=None) -> None:
        ...

    async def web_search_preview(
        self,
        objective: str,
        search_queries: list[str],
    ) -> dict[str, Any]:
        """Search the web with multiple queries in parallel."""
        ...

    async def fetch_url(
        self,
        url: str,
        max_length: int = None,
    ) -> dict[str, Any]:
        """Fetch and extract content from a URL."""
        ...

    def list_tools(self):
        return [self.web_search_preview, self.fetch_url]
```

Each method connects to the MCP server, calls the underlying tool, and returns parsed JSON. Auth credentials are persisted to `~/.mcp-skill/auth/` after first use — keyed by server URL, so credentials persist across restarts automatically.

## Who Is This For?

Developers building:

- **MCP-based agents** that need direct tool access without model round-trips
- **Automation systems** using MCP tools as programmatic building blocks
- **Code-execution agents** using [Programmatic Tool Calling](https://platform.claude.com/cookbook/tool-use-programmatic-tool-calling-ptc)

## Current Limitations

- **Auth**: Supports API key (Bearer or custom header), OAuth, and none — no mTLS or complex auth flows
- **Runtime dependency**: Generated code depends on [fastmcp](https://github.com/jlowin/fastmcp) for MCP client connections
- **Connection per call**: Each method creates a new MCP client connection (no pooling)
- **Tools only**: MCP resources and prompts not yet supported

## Task List

Tracked improvements based on real-world usage:

- [x] **Fix output directory path** — Changed from `.agents/skill/<name>` to `.agents/skills/<name>`
- [x] **Add dependency info to SKILL.md** — Dependencies listed with `uv` and `pip` install commands
- [x] **Generate `__init__.py`** — Skill directory is a proper Python package
- [x] **Post-generation validation** — `ast.parse` → `ruff check` → `ty check` with `uvx` fallback
- [x] **Package-style imports** — Moved `app.py` to skill root; import via `from <skill>.app import <Class>`
- [x] **Persistent token storage** — FileTree-backed credential storage at `~/.mcp-skill/auth/`, keyed by server URL
- [x] **Unified auth signature** — All auth types use `auth=None` in `__init__`
- [x] **Sanitize skill names** — Hyphens/dots converted to underscores for valid Python identifiers
- [x] **Async CLI** — CLI commands are fully async via `asyncclick`
- [ ] **Support MCP resources and prompts** — Currently only tools are introspected and generated
