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.

app, tractor runtime, IPC channel and OS process layers

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_modules allowlist, else an ModuleNotExposed error 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 RemoteActorError with 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 a NoResult sentinel. Results of individual daemon calls come straight back from each await 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:

examples/nested_actor_tree.py (supervisor fan-out)#
@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, whereas cancel_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