Multi-provider AI agents built on pykka.
Project description
actor-ai
Multi-provider AI agents built on pykka.
actor-ai extends pykka — a Python actor-model framework created by Stein Magnus Jodal — so each agent runs in its own thread, processes messages safely from a FIFO queue, and exposes a clean proxy API. On top of that foundation, actor-ai adds natural-language instructions, tool calling, conversation sessions, long-term memory, token accounting, traffic monitoring, shared cross-agent context (SharedContext), multi-agent Chorus coordination, and state-machine Workflow orchestration — all with a provider-agnostic API.
Installation
pip install actor-ai # or: uv add actor-ai
Requires Python ≥ 3.14 and one (or more) provider API keys.
Quick start
from actor_ai import make_agent, Claude
Assistant = make_agent(
"Assistant",
"You are a helpful assistant.",
Claude(), # requires ANTHROPIC_API_KEY
)
with Assistant.get_proxy() as proxy:
reply = proxy.instruct("What is the capital of France?").get()
print(reply) # → "The capital of France is Paris."
make_agent() returns a ready-to-use AIActor subclass without any boilerplate. Agents can delegate to other agents via sub_agents, which are auto-wired as @tool methods the LLM can call:
from actor_ai import make_agent, Claude
Researcher = make_agent("Researcher", "You research topics thoroughly.", Claude())
Writer = make_agent("Writer", "You write clear summaries.", Claude())
Orchestrator = make_agent(
"Orchestrator",
"Coordinate research and writing. Use your tools.",
Claude(),
sub_agents={"researcher": Researcher, "writer": Writer},
)
with Orchestrator.get_proxy() as proxy:
report = proxy.instruct("Write a report on the actor model.").get()
For agents that need lifecycle hooks or stateful tools, the class-based approach is still available:
from actor_ai import AIActor, Claude
class Assistant(AIActor):
system_prompt = "You are a helpful assistant."
provider = Claude()
with Assistant.get_proxy() as proxy:
reply = proxy.instruct("What is the capital of France?").get()
AIActor can also run without a provider — it behaves as a plain pykka actor (memory, session, and message-passing still work; instruct() raises RuntimeError if called without a provider):
DataNode = make_agent("DataNode", "Pure coordination actor.") # no provider
Providers
| Class | Backend | Environment variable |
|---|---|---|
Claude |
Anthropic | ANTHROPIC_API_KEY |
GPT |
OpenAI | OPENAI_API_KEY |
Gemini |
Google AI (OpenAI-compat) | GOOGLE_API_KEY |
Mistral |
Mistral AI | MISTRAL_API_KEY |
DeepSeek |
DeepSeek | DEEPSEEK_API_KEY |
Copilot |
GitHub Copilot (OpenAI-compat) | GITHUB_TOKEN |
LiteLLM |
Any (100+ models) | depends on model |
Copilot routes requests through GitHub Copilot's OpenAI-compatible endpoint and supports multiple underlying models from a single token. Pass use_sdk=True (or use the Copilot_SDK alias) to call the native Copilot SDK session API instead:
from actor_ai import AIActor, Copilot, Copilot_SDK
class Assistant(AIActor):
system_prompt = "You are a helpful coding assistant."
provider = Copilot() # gpt-4o via OpenAI-compat API
provider = Copilot("claude-sonnet-4.5") # Claude via Copilot
provider = Copilot("gpt-5") # GPT-5 via Copilot
provider = Copilot("claude-sonnet-4.5", use_sdk=True) # Claude via Copilot SDK
provider = Copilot_SDK("claude-sonnet-4.5") # same as above
print(Copilot.MODELS) # frozenset of valid model strings
Valid models: gpt-4o, gpt-4o-mini, gpt-5, o1, o1-mini, o3-mini, claude-sonnet-4.5, claude-sonnet-4-5, gemini-2.0-flash. Passing any other string raises ValueError at construction time.
The token is resolved in order: api_key argument → GITHUB_TOKEN env var → gh auth token CLI → OS keyring (GitHub CLI / VS Code). In SDK mode, Copilot CLI authentication is used automatically when no token is supplied.
SharedContext
SharedContext is the cross-cutting coordination layer that lets any number of AIActor instances share state without coupling their implementations. All operations are thread-safe — agents can read and write concurrently.
from actor_ai import make_agent, SharedContext, Claude
ctx = SharedContext()
ctx.remember("project", "Apollo") # visible to every agent on this context
Analyst = make_agent("Analyst", "You are a data analyst.", Claude(), context=ctx)
Writer = make_agent("Writer", "You write concise summaries.", Claude(), context=ctx)
with Analyst.get_proxy() as a, Writer.get_proxy() as w:
analysis = a.instruct("Analyse the project setup.").get()
summary = w.instruct("Summarise the analysis.").get()
# Full interleaved transcript of every agent turn:
for entry in ctx.get_log():
print(entry["agent"], entry["role"], entry["content"][:80])
Memory tiers
| Tier | API | Scope | Cleared by |
|---|---|---|---|
| Long-term | remember / forget |
All agents, all sessions | forget() only |
| Working | remember_working / forget_working |
All agents, current task | clear_working_memory() |
| Conversation log | read-only via get_log() |
Append-only across all agents | clear_log() |
Key behaviours:
- Facts written by any agent are immediately visible to all others.
- Working memory is injected into every agent's system prompt automatically.
clear_session()does not clear shared working memory — callclear_working_memory()explicitly when a task boundary is reached.- Each agent still maintains its own private conversation history for its LLM calls; the shared context is the cross-cutting layer on top.
Chorus
Chorus groups named actors and lets you coordinate them at three levels. Members can be AIActor instances, other Chorus instances, or any plain pykka actor that exposes an instruct() method.
from actor_ai import Chorus, ChorusType
# Create a typed chorus
team = Chorus.start(type="team", agents={"writer": writer_ref, "editor": editor_ref})
# Single-agent instruct
reply = team.proxy().instruct("writer", "Draft a proposal.").get()
# Broadcast to all members in parallel
replies = team.proxy().broadcast("Introduce yourself.").get()
# Pipeline: chain output of each member to the next
final = team.proxy().pipeline(["writer", "editor"], "Write a report.").get()
# Actors can join and leave at runtime
team.proxy().join("reviewer", reviewer_ref).get()
team.proxy().leave("editor").get()
ChorusType is a Literal["system", "project", "team", "department", "custom"] — set at construction and readable via proxy.
Nested choruses: a Chorus can be a member of another Chorus. broadcast() and memory propagation cascade correctly through nesting levels.
Workflow
Workflow implements a state-machine that orchestrates Chorus and actor instances. Transitions fire either when a reply matches a guard predicate or when a named event is dispatched. States and transitions can be added or replaced at runtime on a live workflow.
from actor_ai import Workflow, WorkflowState, WorkflowTransition
wf = Workflow.start(
states={
"draft": WorkflowState(draft_chorus, instruction="{input}"),
"review": WorkflowState(review_chorus, instruction="Review this:\n{output}"),
# Parallel actors — fire simultaneously, replies combined:
"analyse": WorkflowState(
chorus={"researcher": r_ref, "critic": c_ref},
instruction="Analyse: {input}",
),
},
transitions=[
WorkflowTransition("draft", "review", guard=lambda r: "ready" in r),
WorkflowTransition("review", "draft", on_event="reject"),
],
initial_state="draft",
)
# Blocking run — follows guard transitions until terminal:
output = wf.proxy().run("Draft a proposal.").get()
# Fire a named event:
wf.proxy().event("reject").get()
# Non-blocking run — actor mailbox stays free for events during execution:
wf.proxy().run_detached(
"Draft again.",
on_complete=lambda out: print("Done:", out),
).get()
# Add a state at runtime:
wf.proxy().add_state("approve", WorkflowState(approve_chorus, "{output}")).get()
wf.proxy().add_transition(WorkflowTransition("review", "approve", on_event="approve")).get()
Key features
- Agent factory —
make_agent()creates agents in one call;sub_agentsauto-wires delegation tools - Multi-turn sessions — rolling conversation history, configurable window (
max_history) - Long-term memory —
remember(key, value)/forget(key)facts injected into every system prompt - Tool calling — decorate methods with
@toolto expose them to the LLM - SharedContext — thread-safe shared memory (long-term + working) and append-only conversation log across any number of agents; attach with
context=ctxonmake_agent()or as a class attribute - Chorus — group actors as a named team with
broadcast,pipeline,join/leave, typed (ChorusType), nestable - Workflow — state-machine orchestration with guard and event transitions, parallel actor states, runtime modification, non-blocking
run_detached() - Accounting — track token usage (input, output, reasoning, cache read/write) and cost per actor, model, and session (
Ledger,Rates,UsageSummary) - Monitoring — forward per-call metadata to LiteLLM callbacks (Langfuse, Helicone, custom)
Examples
The examples/ directory contains self-contained, runnable scripts. Each example uses a fake scripted provider so no API key is required.
examples/
01_hello_world.py — One-turn interaction
02_session.py — Session history, max_history, clear_session
03_memory.py — remember() / forget()
04_tools.py — @tool decorator and tool calling
05_providers.py — All provider configurations
06_chorus.py — Chorus: instruct, broadcast, pipeline, memory
07_accounting.py — Ledger and Rates
08_monitoring.py — LiteLLM monitoring
09_messages.py — Instruct / Remember / Forget message objects
10_advanced.py — Full pipeline combining all features
11_copilot.py — Copilot provider with token auto-resolution
12_chorus_advanced.py — ChorusType, join/leave, nested choruses, non-AI actors
13_workflow.py — Workflow state machine: run, step, event, runtime modification
14_workflow_parallel.py — Parallel actor states and run_detached()
15_make_agent.py — make_agent() factory: simple agents, tools, sub-agents
16_shared_context.py — SharedContext: shared memory, working memory, conversation log
17_copilot_sdk.py — Copilot native SDK session API (use_sdk=True)
18_copilot_sdk_memory_logs.py — Copilot SDK with memory, working memory, logs, and ledger
Run any example:
uv run python examples/01_hello_world.py
User Manual
See MANUAL.md for the complete reference.
License
MIT
Project details
Release history Release notifications | RSS feed
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 actor_ai-0.3.6.tar.gz.
File metadata
- Download URL: actor_ai-0.3.6.tar.gz
- Upload date:
- Size: 30.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.10.12 {"installer":{"name":"uv","version":"0.10.12","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Pop!_OS","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b303eafdbeb7fdf584d388053a8d8e4d4b45fb97c2115f9a52ebd0718e09f9cf
|
|
| MD5 |
da9cba25e3af7d60cf7c4404c84b3251
|
|
| BLAKE2b-256 |
34f96dd7e9ca44e665ecac7d6bdc3878da2b2308a82115d05ae4b92e1db741d7
|
File details
Details for the file actor_ai-0.3.6-py3-none-any.whl.
File metadata
- Download URL: actor_ai-0.3.6-py3-none-any.whl
- Upload date:
- Size: 38.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.10.12 {"installer":{"name":"uv","version":"0.10.12","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Pop!_OS","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b9b5c0dac4445427997d30583b2c2aa916bc86376d19c7bcc1f25decb41e8e0f
|
|
| MD5 |
1e8ce6af7d853bb44fee1eadc6067404
|
|
| BLAKE2b-256 |
bb0d892cabf144c7543cbaab426652f0bf9cee014e9f4db80d1d590cc0815f40
|