Metadata-Version: 2.4
Name: agnflow
Version: 0.1.1
Summary: 一个简洁的 Python 智能体工作流引擎，支持同步与异步节点、分支、循环、流程图渲染。
Project-URL: Homepage, https://github.com/jianduo1/agnflow
Project-URL: Documentation, https://github.com/jianduo1/agnflow/tree/main/docs
Project-URL: Repository, https://github.com/jianduo1/agnflow.git
Project-URL: Issues, https://github.com/jianduo1/agnflow/issues
Project-URL: Changelog, https://github.com/jianduo1/agnflow/releases
Author-email: jianduo1 <jianduo1@gmail.com>
License-File: LICENSE
Keywords: agent,async,automation,llm,python,workflow
Classifier: Development Status :: 4 - Beta
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: MIT License
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Classifier: Topic :: System :: Distributed Computing
Requires-Python: >=3.8
Requires-Dist: duckduckgo-search>=4.0.0
Requires-Dist: numpy>=1.21.0
Requires-Dist: openai>=1.0.0
Requires-Dist: python-dotenv>=0.19.0
Requires-Dist: pyyaml>=6.0
Requires-Dist: requests>=2.25.0
Requires-Dist: scikit-learn>=1.7.0
Description-Content-Type: text/markdown

<div align="center">
  <h1>agnflow</h1>
  <strong>A concise Python workflow agentic engine</strong>
  <br>
  <h3>
    <a href="LICENSE"><img src="https://img.shields.io/badge/License-MIT-yellow.svg" alt="License"></a>
    <a href="https://jianduo1.github.io/agnflow/"><img src="https://img.shields.io/badge/docs-latest-blue.svg" alt="Docs"></a>
    <a href="https://pypi.org/project/agnflow/"><img src="https://img.shields.io/badge/pypi-v0.1.1-blue.svg" alt="PyPI"></a>
    <a href="https://www.python.org/downloads/"><img src="https://img.shields.io/badge/python-3.8+-blue.svg" alt="Python Version"></a>
  </h3>
</div>

[中文](README_zh.md) | English

**agnflow** pursues simplicity, ease of use, and extensibility, suitable for rapid prototyping, customized LLM workflows, and Agent task flows.

## 🎯 Core Features Showcase

| Agent Type | Code Example | Flowchart |
|:----------:|:--------|:------:|
| **Complex Node Connection** | `n1 >> [n2 >> n3, n3 >> n4] >> n5` | <img src="assets/node_mermaid.png" height="150" alt="Node Connection Flowchart"> |
| **Complex Workflow Connection** | `f1[n1 >> n2 >> f2[n3]] >> f3[n4]` | <img src="assets/flow_mermaid.png" height="150" alt="Workflow Connection Flowchart"> |
| **Supervisor Agent**<br>*First node bidirectionally connected to other nodes* | `s1[n1, n2, n3] >> n4` | <img src="assets/supervisor_mermaid.png" height="150" alt="Supervisor Agent Flowchart"> |
| **Basic Swarm Connection**<br>*Arbitrary nodes bidirectionally connected* | `s1[n1, n2, n3, n4]` | <img src="assets/swarm_mermaid1.png" height="150" alt="Basic Swarm Connection Flowchart"> |
| **Node-Swarm Connection** | `n1 >> s1[n2, n3] >> n4` | <img src="assets/swarm_mermaid2.png" height="150" alt="Node-Swarm Connection Flowchart"> |
| **Multiple Swarm Connection** | `s1[n1, n2] >> s2[n3, n4]` | <img src="assets/swarm_mermaid3.png" height="150" alt="Multiple Swarm Connection Flowchart"> |

## 1. TODO (Future Extension Directions)

- [ ] llm (supporting stream, multimodal, async, structured output)
- [ ] memory
- [ ] rag
- [ ] mcp tool
- [ ] ReAct (reasoning + action)
- [ ] TAO (thought + action + observation)
- [ ] ToT (Tree of Thought)
- [ ] CoT (Chain of Thought)
- [ ] hitl (human in the loop)
- [X] 👏🏻 supervisor swarm

> The above are future extensible intelligent agent/reasoning/tool integration directions. Contributions and suggestions are welcome.

## 2. Features
- Node-based workflows with support for branching, loops, and sub-flows
- Support for synchronous and asynchronous execution
- Support for flowchart rendering (dot/mermaid)
- Clean code, easy to extend

## 3. Installation

### 3.1 Install from PyPI (Recommended)

```bash
# Install using pip
pip install agnflow

# Install using rye
rye add agnflow

# Install using poetry
poetry add agnflow

# Install specific version
pip install agnflow==0.1.0

# Install latest development version
pip install --upgrade agnflow
```

### 3.2 Install from Source

Recommended to use [rye](https://rye-up.com/) for dependency and virtual environment management:

```bash
# Clone repository
git clone https://github.com/jianduo1/agnflow.git
cd agnflow

# Install dependencies
rye sync

# Development mode installation
rye sync --dev
```

### 3.3 Flowchart Rendering Tools (Optional)

**Note: Generating images requires additional tools**

**Dot format image generation (Recommended):**
```bash
# macOS
brew install graphviz

# Ubuntu/Debian
sudo apt-get install graphviz

# CentOS/RHEL
sudo yum install graphviz

# Windows
# Download and install: https://graphviz.org/download/
```

**Mermaid format image generation:**
```bash
# Install mermaid-cli
npm install -g @mermaid-js/mermaid-cli

# Install puppeteer browser (for rendering)
npx puppeteer browsers install chrome-headless-shell
```

### 3.4 Development Environment

Use rye to manage development environment:

```bash
# Install dependencies
rye sync

# Run tests
rye run test

# Code formatting
rye run format

# Code linting
rye run lint

# Run examples
rye run example
```

### 3.5 Publish to PyPI

```bash
# Clean previous builds
rye run clean

# Build package
rye run build

# Upload to test PyPI (recommended to test first)
rye run upload-test

# Upload to official PyPI
rye run upload
```

**Note:** First upload to PyPI requires:
1. Register account on [PyPI](https://pypi.org)
2. Register account on [TestPyPI](https://test.pypi.org)
3. Configure `~/.pypirc` file or use environment variables

## 4. Quick Start

### 4.1 Define Nodes
```python
from agnflow import Node, Flow

def hello_exec(state):
    print("hello", state)
    return {"msg": "world"}

def world_exec(state):
    print("world", state)

n1 = Node("hello", exec=hello_exec)
n2 = Node("world", exec=world_exec)
n1 >> n2
```

### 4.2 Build and Run Workflow
```python
flow = Flow(n1, name="demo")
flow.run({"msg": "hi"})
```

### 4.3 Asynchronous Execution
```python
import asyncio
async def ahello(state):
    print("async hello", state)
    return {"msg": "async world"}
n1 = Node("hello", aexec=ahello)
flow = Flow(n1)
asyncio.run(flow.arun({"msg": "hi"}))
```

### 4.4 Render Flowchart
```python
print(flow.render_dot())      # Output dot format
print(flow.render_mermaid())  # Output mermaid format

# Save as image file
flow.render_dot(saved_file="./flow.png")      # Save dot format image
flow.render_mermaid(saved_file="./flow.png")  # Save mermaid format image
```

## 5. Node Function Details

### 5.1 Function Input Parameter Methods

agnflow supports multiple function input parameter methods and automatically retrieves parameters from the state based on function signatures:

#### Method 1: Receive the entire state
```python
def my_node(state):
    """Receive the entire state dictionary"""
    print(f"Received state: {state}")
    return {"result": "processed"}

n1 = Node("my_node", exec=my_node)
```

#### Method 2: Automatic injection by parameter name
```python
def my_node(user_id, message, data):
    """Automatically get values from state by parameter names"""
    print(f"User ID: {user_id}")
    print(f"Message: {message}")
    print(f"Data: {data}")
    return {"processed": True}

# Pass state containing these fields when calling
flow.run({
    "user_id": "123",
    "message": "hello",
    "data": {"key": "value"}
})
```

#### Method 3: Mixed approach
```python
def my_node(user_id, state):
    """Mixed approach: partial parameters + entire state"""
    print(f"User ID: {user_id}")
    print(f"Complete state: {state}")
    return {"user_processed": True}
```

### 5.2 Function Return Value Methods

Node functions support multiple return value formats:

#### Method 1: Return only new state
```python
def my_node(state):
    """Only update state, use default action"""
    return {"new_data": "value", "timestamp": time.time()}
```

#### Method 2: Return action and new state
```python
def my_node(state):
    """Return action and updated state"""
    if state.get("condition"):
        return "success", {"result": "success"}
    else:
        return "error", {"result": "error"}
```

#### Method 3: Return only action
```python
def my_node(state):
    """Return only action, don't update state"""
    if state.get("condition"):
        return "success"
    else:
        return "error"
```

#### Method 4: Return None (end workflow)
```python
def my_node(state):
    """Return None to end workflow"""
    if state.get("should_stop"):
        return None
    return "continue", {"step": "completed"}
```

### 5.3 Asynchronous Node Functions

Asynchronous node functions use the `aexec` parameter and support all synchronous function features:

```python
import asyncio

async def async_node(state):
    """Asynchronous node function"""
    await asyncio.sleep(0.1)  # Simulate async operation
    return {"async_result": "done"}

async def async_node_with_action(user_id, state):
    """Asynchronous node function - mixed parameters + action"""
    await asyncio.sleep(0.1)
    return "next", {"user_id": user_id, "processed": True}

# Create asynchronous nodes
n1 = Node("async_node", aexec=async_node)
n2 = Node("async_node_with_action", aexec=async_node_with_action)

# Asynchronous execution
asyncio.run(flow.arun({"user_id": "123"}))
```

### 5.4 Node Class Inheritance Method

Besides function methods, you can also create nodes by inheriting from the `Node` class:

```python
class MyNode(Node):
    def exec(self, state):
        """Synchronous execution method"""
        print(f"Executing node: {self.name}")
        return {"class_result": "success"}
    
    async def aexec(self, state):
        """Asynchronous execution method"""
        print(f"Asynchronously executing node: {self.name}")
        return {"async_class_result": "success"}

# Use class node
n1 = MyNode("my_class_node")
```

### 5.5 Error Handling and Retry

Nodes support error handling and retry mechanisms:

```python
def risky_node(state):
    """Node that might fail"""
    if random.random() < 0.5:
        raise Exception("Random error")
    return {"success": True}

# Create node with retry support
n1 = Node("risky_node", exec=risky_node, max_retries=3, wait=1)

# Custom error handling
class SafeNode(Node):
    def exec_fallback(self, state, exc):
        """Custom error handling"""
        return "error", {"error": str(exc), "recovered": True}
    
    async def aexec_fallback(self, state, exc):
        """Custom asynchronous error handling"""
        return "error", {"error": str(exc), "recovered": True}
```

### 5.6 Complete Example

```python
from agnflow import Node, Flow
import time

# Define different types of node functions
def start_node(user_id, message):
    """Receive specific parameters"""
    return "n2", {"user_id": user_id, "message": message}

def process_node(state):
    """Receive entire state"""
    processed = f"Processed: {state['message']}"
    return "n3", {"processed": processed, "timestamp": time.time()}

def complete_node(result, state):
    """Mixed parameters"""
    print(f"Result: {result}")
    print(f"State: {state}")
    return {"final_result": "success"}

# Create nodes
n1 = Node("start", exec=start_node)
n2 = Node("process", exec=process_node)
n3 = Node("complete", exec=complete_node)

# Connect nodes
n1 >> n2 >> n3

# Create workflow
flow = Flow(n1, name="example_flow")

# Run workflow
result = flow.run({
    "user_id": "123",
    "message": "Hello agnflow!"
})

print(f"Workflow result: {result}")
```

## 6. Node Connection Syntax

agnflow provides multiple flexible node connection methods:

### 6.1 Linear Connection
```python
# Method 1: Forward connection
a >> b >> c

# Method 2: Reverse connection  
c << b << a
```

### 6.2 Branch Connection
```python
# Branch based on node return value
a >> [b, c]
```

### 6.3 Sub-flow Connection
```python
# Connect sub-flows
d1 >> flow >> d2
```

## 7. Complex Workflow Example

After running the example code `src/agnflow/example.py`, the following flowcharts will be generated:

Workflow definition:
```py
a >> [b >> flow, c >> a]
d1 >> flow >> d2
```

### 7.1 Dot Format Flowchart
![Dot Flow](assets/flow_dot.png)

### 7.2 Mermaid Format Flowchart  
![Mermaid Flow](assets/flow_mermaid.png)

These flowcharts illustrate:
- Connections between nodes
- Branching and looping structures
- Nesting of subprocesses
- Overall execution paths of workflows

## 8. Reference Frameworks

agnflow references and benchmarks against the following mainstream intelligent agent/workflow frameworks:

![LangGraph](https://img.shields.io/badge/LangGraph-green.svg) ![LlamaIndex](https://img.shields.io/badge/LlamaIndex-green.svg) ![AutoGen](https://img.shields.io/badge/AutoGen-green.svg) ![Haystack](https://img.shields.io/badge/Haystack-green.svg) ![CrewAI](https://img.shields.io/badge/CrewAI-green.svg) ![FastGPT](https://img.shields.io/badge/FastGPT-green.svg) ![PocketFlow](https://img.shields.io/badge/PocketFlow-green.svg)

## 9. Project Status

### 📦 Release Status
- **PyPI**: ✅ [v0.1.1](https://pypi.org/project/agnflow/0.1.1/) Released
- **GitHub**: ✅ [Open Source Repository](https://github.com/jianduo1/agnflow)
- **Documentation**: ✅ [API Documentation](docs/API.md) Complete
- **Testing**: ✅ Functional testing passed

### 🔄 Version Information
- **Current Version**: 0.1.1
- **Python Support**: 3.8+
- **License**: MIT
- **Status**: Beta

## 10. License
MIT 