RPC: calling into other actors#
Every spawn call from Spawning actors hands you back
a Portal: a live handle for calling into
another actor’s memory domain. The name is borrowed from
trio’s portal concept — an object you use to submit work
into a separate concurrency domain — except here that domain
is a whole other process.
The layers a portal.run() request rides through.#
There are no proxy objects and no special calling
conventions: you pass a plain function reference plus keyword
args, and Python’s normal await-able semantics apply. The
function just happens to run somewhere else; from the calling
task it looks as though it was called locally. And since this is
all structured concurrency (SC) under the hood, the remote task
runs inside the callee’s supervised task tree while its result
— or its failure, as a boxed
RemoteActorError — always comes back to you.
Portal.run(): pass the function, not a string#
run() schedules an async function as
a new task in the remote actor and waits on its result:
async with tractor.open_nursery() as an:
portal = await an.start_actor(
'service',
enable_modules=[__name__],
)
answer = await portal.run(movie_theatre_question)
The rules of engagement:
the target must be an async function and its defining module must be in the callee’s
enable_modulesallowlist, else anModuleNotExposederror is relayed back (see Spawning actors for the capability-allowlist story).arguments are passed by keyword only; they ride the IPC layer as msgspec-encoded msgs, so keep them serializable.
every call schedules a fresh task remotely — call it twice and the callee runs two tasks, each supervised in its own right.
remote exceptions re-raise locally as
RemoteActorErrorwith the original type preserved via.boxed_type.
Note
Passing dotted-path strings to run() is an ancient,
deprecated form; always pass the function reference. If you
really need name-based addressing use run_from_ns()
below.
Namespaced daemons: run_from_ns()#
Sometimes the calling process can’t (or shouldn’t) import the
target function — think a long-running rpc-daemon serving
modules your client never loads. For that,
run_from_ns() takes the explicit
namespace path:
await portal.run_from_ns('mypkg.service', 'ping')
This is literally how .run() works underneath: the pair is
encoded as a 'mod.path:func' style msg and resolved against
the callee’s enabled modules.
One special namespace exists: 'self' resolves to the remote
Actor instance, i.e. the runtime itself. It’s
how internal machinery (cancel requests, registry ops) travels;
don’t build your app on it.
One-shot results: wait_for_result()#
A portal returned from
run_in_actor() has exactly one
“main” task running remotely; that task’s return value is
delivered as the portal’s final result:
portal = await an.run_in_actor(fib, n=10)
final = await portal.wait_for_result()
Semantics worth knowing:
it blocks until the remote task returns, re-raising any remote error in the usual boxed form.
once resolved it’s idempotent: later calls return the same cached value.
a daemon portal (from
start_actor()) has no main task, so there’s no final result to wait for: you’ll get a warning plus aNoResultsentinel. Results of individual daemon calls come straight back from eachawait portal.run().
Pure RPC daemons: run_daemon()#
When a process’s only job is to sit at the root of its own
tree and serve RPC, skip the boilerplate with
tractor.run_daemon():
import tractor
tractor.run_daemon(
['mypkg.service'],
name='service',
)
It’s a blocking convenience (it calls trio.run() for you):
boot a root actor with the given modules enabled for RPC, then
sleep until cancelled. Pair it with the discovery system —
tractor.find_actor() / tractor.wait_for_actor()
from a separate program — and you’ve got a tiny service
architecture with zero framework ceremony; see
examples/service_daemon_discovery.py for the full pattern.
Fan-out: RPC through nested trees#
Portals compose. An RPC task is just a trio task, so it can
open its own ActorNursery and portal into
its children — one inbound call fanning out into a whole
sub-tree of work. The mid-tier function from the nested-tree
example:
@tractor.context
async def fan_out_squares(
ctx: tractor.Context,
vals: list[int],
) -> list[int]:
'''
Spawn a (nested) pair of leaf actors, fan the input vals
out across them round-robin style, then return the
aggregated squares to our parent.
'''
async with tractor.open_nursery() as an:
portals: list[tractor.Portal] = []
for i in (1, 2):
portals.append(
await an.start_actor(
f'leaf_{i}',
enable_modules=[__name__],
)
)
# unblock the parent's `.open_context()` entry and
# report which leaves came up.
await ctx.started(
[p.chan.aid.name for p in portals]
)
squares: dict[int, int] = {}
async def run_in_leaf(
portal: tractor.Portal,
x: int,
) -> None:
squares[x] = await portal.run(
compute_square,
x=x,
)
# fan out one sub-RPC per input val, concurrently.
async with trio.open_nursery() as tn:
for i, x in enumerate(vals):
tn.start_soon(
run_in_leaf,
portals[i % len(portals)],
x,
)
# graceful inside-out teardown: leaves go first!
for portal in portals:
leaf_name: str = portal.chan.aid.name
print(f'supervisor: cancelling {leaf_name}')
await portal.cancel_actor()
return [squares[x] for x in vals]
The root portals into the supervisor actor; the
supervisor’s RPC task spawns the leaf workers, portals into
each, and returns the combined result back up. Failures at any
depth relay hop-by-hop as boxed errors, and cancelling the root
call tears down the entire sub-tree — SC, transitively.
When to graduate to Context#
portal.run() is great for one-shot, request-response calls.
Reach for open_context() with an
@tractor.context endpoint as soon as you want:
a long-lived dialog with state held on both sides,
bidirectional streaming via
ctx.open_stream(),typed payload contracts (
pld_spec) enforced at the msg layer,or task-scoped cancellation:
Context.cancel()cancels just the linked remote task, whereascancel_actor()nukes the entire remote runtime and its process.
In fact the source plans for Portal.run() itself to be
rebuilt on top of open_context() — contexts are the core
inter-actor protocol. Take the full tour in
The Context: a cross-actor task pair.
See also
Spawning actors — where portals come from and how their actors are supervised.
The Context: a cross-actor task pair — the structured cross-actor task API: handshake, streaming, typed payloads.
Cancellation and error propagation — what happens to in-flight RPC when trees get torn down.