Metadata-Version: 2.1
Name: agentanycast
Version: 0.7.0
Summary: Python SDK for AgentAnycast — A2A protocol over P2P
Project-URL: Homepage, https://github.com/AgentAnycast/agentanycast
Project-URL: Repository, https://github.com/AgentAnycast/agentanycast-python
Project-URL: Issues, https://github.com/AgentAnycast/agentanycast/issues
Project-URL: Changelog, https://github.com/AgentAnycast/agentanycast-python/blob/main/CHANGELOG.md
Author: AgentAnycast Contributors
License: Apache-2.0
License-File: LICENSE
Keywords: a2a,a2a-protocol,agent,agent-communication,claude-agent-sdk,crewai,decentralized,did,end-to-end-encryption,google-adk,grpc,langgraph,libp2p,mcp,mcp-server,nat-traversal,noise-protocol,openai-agents,p2p,strands-agents,zero-config
Classifier: Development Status :: 3 - Alpha
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: Apache Software 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 :: Software Development :: Libraries
Requires-Python: >=3.10
Requires-Dist: base58>=2.1
Requires-Dist: click>=8.0
Requires-Dist: grpcio>=1.60
Requires-Dist: httpx>=0.27
Requires-Dist: protobuf>=4.25
Provides-Extra: adapters
Requires-Dist: claude-agent-sdk>=0.1; extra == 'adapters'
Requires-Dist: crewai>=0.80; extra == 'adapters'
Requires-Dist: google-adk>=1.0; extra == 'adapters'
Requires-Dist: langgraph>=0.2; extra == 'adapters'
Requires-Dist: openai-agents>=0.10; extra == 'adapters'
Requires-Dist: strands-agents>=0.1; extra == 'adapters'
Provides-Extra: adk
Requires-Dist: google-adk>=1.0; extra == 'adk'
Provides-Extra: claude
Requires-Dist: claude-agent-sdk>=0.1; extra == 'claude'
Provides-Extra: crewai
Requires-Dist: crewai>=0.80; extra == 'crewai'
Provides-Extra: dev
Requires-Dist: mypy>=1.10; extra == 'dev'
Requires-Dist: pytest-asyncio>=0.23; extra == 'dev'
Requires-Dist: pytest>=8.0; extra == 'dev'
Requires-Dist: ruff>=0.4; extra == 'dev'
Provides-Extra: langgraph
Requires-Dist: langgraph>=0.2; extra == 'langgraph'
Provides-Extra: mcp
Requires-Dist: mcp>=1.2; extra == 'mcp'
Provides-Extra: openai-agents
Requires-Dist: openai-agents>=0.10; extra == 'openai-agents'
Provides-Extra: strands
Requires-Dist: strands-agents>=0.1; extra == 'strands'
Description-Content-Type: text/markdown

# AgentAnycast Python SDK

Python SDK for AgentAnycast -- decentralized A2A agent-to-agent communication over P2P.

[![Python](https://img.shields.io/badge/Python-3.10+-3776AB?logo=python&logoColor=white)](https://python.org)
[![License](https://img.shields.io/badge/License-Apache%202.0-blue)](LICENSE)

> **AgentAnycast is fully decentralized.** On a local network, it works with zero configuration. For cross-network communication, just deploy your own relay with a single command.

## Installation

```bash
pip install agentanycast
```

With framework adapters:

```bash
pip install agentanycast[crewai]         # CrewAI integration
pip install agentanycast[langgraph]      # LangGraph integration
pip install agentanycast[google-adk]     # Google ADK integration
pip install agentanycast[openai-agents]  # OpenAI Agents SDK integration
```

## How It Works

```
┌─────────────┐         mDNS / Relay         ┌─────────────┐
│  Agent A    │◄──────────────────────────────►│  Agent B    │
│  (Python)   │     E2E encrypted (Noise)     │  (Python)   │
└──────┬──────┘                               └──────┬──────┘
       │ gRPC                                        │ gRPC
┌──────┴──────┐                               ┌──────┴──────┐
│ agentanycastd│                               │ agentanycastd│
│  (daemon)   │                               │  (daemon)    │
└─────────────┘                               └──────────────┘
```

- **Local network (LAN):** Agents discover each other automatically via mDNS. No relay needed.
- **Cross-network (WAN):** Deploy your own relay server, then point agents to it. One command.

## Quick Start

### CLI

```bash
# Start an echo agent
agentanycast demo

# Discover agents by skill
agentanycast discover translate

# Send a task to a specific agent
agentanycast send 12D3KooW... "Hello!"

# Check node status
agentanycast status
```

### Python API

**Server agent:**

```python
from agentanycast import Node, AgentCard, Skill

card = AgentCard(
    name="EchoAgent",
    description="Echoes back any message",
    skills=[Skill(id="echo", description="Echo messages")],
)

async with Node(card=card) as node:
    @node.on_task
    async def handle(task):
        text = task.messages[-1].parts[0].text
        await task.complete(artifacts=[{"parts": [{"text": f"Echo: {text}"}]}])

    await node.serve_forever()
```

**Client agent:**

```python
from agentanycast import Node, AgentCard

card = AgentCard(name="Client", description="A client agent", skills=[])

async with Node(card=card) as node:
    handle = await node.send_task(
        peer_id="12D3KooW...",
        message={"role": "user", "parts": [{"text": "Hello!"}]},
    )
    result = await handle.wait()
    print(result.artifacts[0].parts[0].text)  # "Echo: Hello!"
```

### Three Ways to Send a Task

```python
# 1. Direct — by Peer ID
await node.send_task(peer_id="12D3KooW...", message=msg)

# 2. Anycast — by skill (relay resolves the target)
await node.send_task(skill="translate", message=msg)

# 3. HTTP Bridge — to standard HTTP A2A agents
await node.send_task(url="https://agent.example.com", message=msg)
```

### Skill Discovery

```python
# Find all agents offering a skill
agents = await node.discover("translate")

# With tag filtering
agents = await node.discover("translate", tags={"lang": "fr"})
```

## Framework Adapters

Expose existing agent frameworks as P2P agents with one function call:

```python
# CrewAI
from agentanycast.adapters.crewai import serve_crew
await serve_crew(crew, card=card, relay="...")

# LangGraph
from agentanycast.adapters.langgraph import serve_graph
await serve_graph(compiled_graph, card=card, relay="...")

# Google ADK
from agentanycast.adapters.adk import serve_adk_agent
await serve_adk_agent(agent, card=card, relay="...")

# OpenAI Agents SDK
from agentanycast.adapters.openai_agents import serve_openai_agent
await serve_openai_agent(agent, card=card, relay="...")
```

## Interoperability

### W3C DID

```python
from agentanycast.did import peer_id_to_did_key, did_key_to_peer_id
from agentanycast.did import did_web_to_url, url_to_did_web

did = peer_id_to_did_key("12D3KooW...")      # "did:key:z6Mk..."
pid = did_key_to_peer_id("did:key:z6Mk...")  # "12D3KooW..."

url = did_web_to_url("did:web:example.com:agents:myagent")
# "https://example.com/agents/myagent/did.json"
```

### MCP (Model Context Protocol)

```python
from agentanycast.mcp import mcp_tools_to_agent_card

card = mcp_tools_to_agent_card(mcp_tools, name="MCPAgent")
```

### A2A v1.0 Protocol Compatibility

```python
from agentanycast.compat.a2a_v1 import task_to_a2a_json, task_from_a2a_json

# Convert internal Task ↔ official A2A v1.0 JSON format
a2a_json = task_to_a2a_json(task)
task = task_from_a2a_json(a2a_json)
```

### OASF (Open Agentic Schema Framework)

```python
from agentanycast.compat.oasf import card_to_oasf_record, card_from_oasf_record

# Convert AgentCard ↔ OASF records for AGNTCY Agent Directory
record = card_to_oasf_record(card, authors=["org"])
card = card_from_oasf_record(record)
```

### AGNTCY Directory

```python
from agentanycast.compat.agntcy import AGNTCYDirectory

directory = AGNTCYDirectory(base_url="https://directory.agntcy.org")
agents = await directory.search("translation")
```

## API Reference

| Class | Description |
|---|---|
| `Node` | Main entry point. Manages daemon lifecycle, sends and receives tasks. |
| `AgentCard` | Describes an agent's identity, capabilities, and metadata. |
| `Skill` | Defines a single skill an agent can perform. |
| `TaskHandle` | Returned by `send_task()`. Call `wait()` to block until the remote agent responds. |
| `IncomingTask` | Passed to task handlers. Provides the incoming message and methods to respond. |

## Node Options

| Parameter | Description | Default |
|---|---|---|
| `card` | Your agent's `AgentCard` | Required |
| `relay` | Relay server multiaddr for cross-network communication | `None` (LAN only) |
| `daemon_path` | Path to a local `agentanycastd` binary | Auto-download |
| `daemon_addr` | Address of an externally managed daemon | Auto-managed |
| `key_path` | Path to Ed25519 identity key file | `<home>/key` |
| `home` | Data directory for daemon state. Use different values to run multiple nodes on the same machine. | `~/.agentanycast` |

## Requirements

- Python 3.10+
- The [agentanycastd](https://github.com/AgentAnycast/agentanycast-node) daemon (auto-managed by the SDK, or bring your own)

## License

[Apache License, Version 2.0](LICENSE)
