So far you’ve built a single agent: one LLM, one context, one set of tools. But many problems are better served by several agents working together, each owning its own LLM. A greeter hands off to a support agent. A researcher runs in the background while the main agent keeps talking. A screen-driving agent acts on the UI while a voice agent converses. Giving each agent its own LLM keeps every context small and focused. Instead of one model juggling every instruction and tool, each agent reasons over just its own job. That’s cheaper, faster, and less prone to the model getting lost.Documentation Index
Fetch the complete documentation index at: https://docs.pipecat.ai/llms.txt
Use this file to discover all available pages before exploring further.
LLMWorker overview
Multi-agent systems often run several LLM-backed agents — a greeter, a support agent, a researcher — each with its own instructions, tools, and (optionally) its own conversation context.LLMWorker is the building block for each one. It extends PipelineWorker with everything you need to run an LLM-powered agent:
- A pipeline with your LLM service, automatically built
- Tool registration via the
@tooldecorator - Activation handling that injects messages and runs the LLM
LLMWorker so you can host @tool methods, then instantiate it with its own LLM service. Pass bridged=() so the agent receives frames from the bus:
bus= argument to the constructor — the worker gets its bus when you register it with runner.add_workers(...). The default pipeline is Pipeline([llm]), with tools from build_tools() automatically registered. When bridged=() is set, the framework wraps this pipeline with edge processors that connect it to the bus.
The @tool decorator
The@tool decorator marks a method as an LLM-callable tool. The framework automatically collects all @tool-decorated methods and registers them with the LLM service.
Args section in the docstring.
Tool options
The@tool decorator accepts options:
| Option | Default | Description |
|---|---|---|
cancel_on_interruption | True | Cancel the tool if the user interrupts |
timeout | None | Maximum execution time in seconds |
Tool parameters
Every tool method receivesself and params: FunctionCallParams as the first two arguments. Additional arguments are the tool’s parameters that the LLM fills in.
The params object gives you access to:
params.result_callback(result)— return the result to the LLMparams.llm— the LLM service instance, useful for queuing frames
Returning results
Always callparams.result_callback() to return the tool result to the LLM:
Activation with messages
When anLLMWorker is activated, you can inject messages into its context. Pass an LLMWorkerActivationArgs via the args parameter:
on_activated() implementation:
- Sets the tools from
build_tools() - Injects the provided messages into the LLM context
- Runs the LLM if
run_llmisTrue(the default whenmessagesis set)
Managing context with LLMContextWorker
A plainLLMWorker runs an LLM but doesn’t manage conversation context on its own — it relies on context coming from elsewhere (for example, the main agent’s aggregators bridged in). When an agent needs to keep its own history, use LLMContextWorker. It extends LLMWorker with a built-in LLMContext and the user/assistant aggregator pair, building the pipeline as [user_aggregator, llm, assistant_aggregator] for you.
LLMContextWorker gets its own context by default, so agents don’t see each other’s history. To give several agents a shared conversation, pass the same context= to each:
self.user_aggregator and self.assistant_aggregator.
Custom pipelines
If you need more control, you can pass a custompipeline= to the LLMWorker constructor. For example, to add TTS to the agent’s own pipeline:
What’s next
Now that your agents can run LLMs and call tools, here’s a powerful one: an agent that sees and drives the user’s screen.Controlling the UI
A UIWorker that reads the screen and acts on it over a two-way RTVI interface