An open API service indexing awesome lists of open source software.

https://github.com/askmanu/acorn

LLM framework for long running agents
https://github.com/askmanu/acorn

ai ai-agents llm-framework

Last synced: 4 months ago
JSON representation

LLM framework for long running agents

Awesome Lists containing this project

README

          

github

**LLM agent framework with structured I/O**

Build AI agents with type-safe inputs and outputs, automatic tool calling, and powerful agentic loops.

[![Tests](https://img.shields.io/badge/tests-201%20passing-brightgreen)](tests/)
[![Coverage](https://img.shields.io/badge/coverage-85%25-brightgreen)](tests/)
[![Python](https://img.shields.io/badge/python-3.10+-blue)](pyproject.toml)
[![License](https://img.shields.io/badge/license-MIT-blue)](LICENSE)

---

## โœจ Features

- ๐ŸŽฏ **Structured I/O** - Pydantic models for inputs and outputs
- ๐Ÿค– **Agentic Loops** - Multi-turn execution with tool calling
- ๐Ÿ› ๏ธ **Auto Tool Schemas** - Generate from type hints and docstrings
- ๐Ÿ”„ **Dynamic Tools** - Add/remove tools during execution
- โœ… **Parse Error Recovery** - Automatic retry on validation failures
- ๐Ÿ“Š **Step Callbacks** - Full control over loop behavior
- ๐Ÿ”Œ **LiteLLM Integration** - Works with any LLM provider
- ๐ŸŒŠ **Streaming Responses** - Real-time output with partial structured updates
- ๐Ÿ’พ **Provider Caching** - Reduce latency and cost with prompt caching
- ๐Ÿ›ก๏ธ **Model Fallbacks** - Automatic provider failover for high availability
- ๐ŸŒณ **Branching Workflows** - Spawn sub-agents that extend parent capabilities for parallel analysis and map-reduce patterns

---

## ๐Ÿš€ Quick Start

### Installation

```bash
pip install acorn
```

Set your API key:

```bash
# For Anthropic Claude
export ANTHROPIC_API_KEY="your-key-here"

# Or for OpenAI
export OPENAI_API_KEY="your-key-here"

# Or any other LiteLLM-supported provider
```

### Single-Turn Example

```python
from pydantic import BaseModel, Field
from acorn import Module

class Input(BaseModel):
text: str = Field(description="The text to summarize")
max_words: int = Field(default=100, description="Maximum words in summary")

class Output(BaseModel):
summary: str = Field(description="The concise summary")
word_count: int = Field(description="Number of words in summary")

class Summarizer(Module):
"""Summarize text concisely."""

initial_input = Input
final_output = Output
model = "anthropic/claude-sonnet-4-5-20250514"

# Use it
summarizer = Summarizer()
result = summarizer(
text="Long article text here...",
max_words=50
)

print(result.summary)
print(f"Words: {result.word_count}")
```

### Multi-Turn Agentic Loop

```python
from pydantic import BaseModel, Field
from acorn import Module, tool

class Input(BaseModel):
topic: str = Field(description="Research topic")
depth: str = Field(default="shallow", description="Research depth")

class Output(BaseModel):
findings: str = Field(description="Summary of findings")
sources: list[str] = Field(description="Sources consulted")

class ResearchAgent(Module):
"""Research assistant with tools."""

initial_input = Input
max_steps = 5 # Enable agentic loop
final_output = Output
model = "anthropic/claude-sonnet-4-5-20250514"

@tool
def search(self, query: str) -> list:
"""Search for information."""
# Your search implementation
return ["result1", "result2"]

@tool
def analyze(self, data: str) -> str:
"""Analyze collected data."""
# Your analysis implementation
return f"Analysis: {data}"

def on_step(self, step):
"""Called after each step."""
print(f"Step {step.counter}")

# Early termination if done
if len(step.tool_results) >= 3:
step.finish(
findings="Sufficient data collected",
sources=["source1", "source2"]
)

return step

# Use it
agent = ResearchAgent()
result = agent(topic="Large Language Models", depth="shallow")
```

---

## ๐Ÿ“š Core Concepts

### Module
Base class for LLM agents. Configure with:
- `model` - LLM to use (required - no default)
- `temperature` - Sampling temperature
- `max_tokens` - Maximum tokens to generate
- `max_steps` - Max agentic loop iterations (None = single-turn)
- `initial_input` - Pydantic model for input schema
- `final_output` - Pydantic model for output schema
- `tools` - List of available tools
- `cache` - Enable provider-level prompt caching
- `model_fallbacks` - List of fallback models for automatic failover

### Tools
Functions the LLM can call:

```python
@tool
def search(query: str, limit: int = 10) -> list:
"""Search for information.

Args:
query: The search query
limit: Maximum results to return
"""
return search_api(query, limit)
```

Schema is automatically generated from type hints and docstring.

### Step Callback
Control agentic loop execution:

```python
def on_step(self, step):
# Access step info
print(f"Step {step.counter}")
print(f"Tools called: {[tc.name for tc in step.tool_calls]}")

# Dynamic tool management
step.add_tool(new_tool)
step.remove_tool("old_tool")

# Early termination
if condition:
step.finish(result="Early exit")

return step
```

---

## ๐ŸŽฏ Examples

Try them live on the [Gradio app](https://askmanu-acorn.hf.space) or browse the source in `examples/`:

| Example | Category | Description |
|---------|----------|-------------|
| [Simple Q&A](https://askmanu-acorn.hf.space/simple_qa) | Basic | Single-turn question answering with structured output |
| [HN Production Readiness](https://askmanu-acorn.hf.space/hn_production_check) | Agentic | Checks if a trending HN project is production-ready |
| [Documentation Coverage](https://askmanu-acorn.hf.space/doc_coverage) | Agentic | Scores documentation coverage of a GitHub repo (0โ€“100) |
| [Bus Factor Calculator](https://askmanu-acorn.hf.space/bus_factor) | Branching | Calculates the bus factor of a GitHub repository |
| [License Compatibility](https://askmanu-acorn.hf.space/license_checker) | Agentic | Checks dependency license compatibility for conflicts |
| [Dependency Bloat Scanner](https://askmanu-acorn.hf.space/dependency_scanner) | Branching | Finds redundant and overlapping libraries in your deps |

---

## ๐Ÿงช Testing

```bash
# Run all tests
pytest

# With coverage
pytest --cov=acorn

# Specific test file
pytest tests/test_agentic_loop.py -v
```

**Current status:** 201 tests passing, 85% coverage

---

## ๐Ÿ“– Documentation

- [Getting Started](docs/getting-started.md) - Installation and first steps
- [Module Reference](docs/module.md) - Complete Module API documentation

---

## ๐Ÿ›ฃ๏ธ Roadmap

### โœ… Completed
- Single-turn execution
- Multi-turn agentic loops
- Tool calling with auto-schema generation
- Parse error recovery
- Dynamic tool management
- Step callbacks
- Streaming responses with partial structured output
- Forced termination strategies
- Provider caching
- Model fallbacks
- Branching workflows

### ๐Ÿ“‹ Planned
- Async support

---

## ๐Ÿค Contributing

Contributions welcome! Please:

1. Check open issues for areas to help
2. Write tests for new features (maintain >80% coverage)
3. Update documentation
4. Add examples for new features

---

## ๐Ÿ’ฌ Questions?

Check out:
- [Getting Started](docs/getting-started.md) for installation and examples
- [Module Reference](docs/module.md) for detailed API docs
- [Examples](examples/) for working code
- [Tests](tests/) for usage patterns

---

## ๐Ÿ™ Acknowledgments
Thanks to @rosenbrockc for donating the `acorn` pip package name.

---

## ๐Ÿ“„ License

[MIT License](LICENSE)

---