Use RpcTarget class to handle Durable Object metadata
Access the name from within a Durable Object using RpcTarget.
When working with Durable Objects, you will need to access the name that was used to create the Durable Object via idFromName()
. This name is typically a meaningful identifier that represents what the Durable Object is responsible for (like a user ID, room name, or resource identifier).
However, there is a limitation in the current implementation: even though you can create a Durable Object with .idFromName(name)
, you cannot directly access this name inside the Durable Object via this.ctx.id.name
.
The RpcTarget
pattern shown below offers a solution by creating a communication layer that automatically carries the name with each method call. This keeps your API clean while ensuring the Durable Object has access to its own name.
Based on your needs, you can either store the metadata temporarily in the RpcTarget
class, or use Durable Object storage to persist the metadata for the lifetime of the object.
This example does not persist the Durable Object metadata. It demonstrates how to:
- Create an
RpcTarget
class - Set the Durable Object metadata (identifier in this example) in the
RpcTarget
class - Pass the metadata to a Durable Object method
- Clean up the
RpcTarget
class after use
import { DurableObject, RpcTarget } from "cloudflare:workers";
// * Create an RpcDO class that extends RpcTarget// * Use this class to set the Durable Object metadata// * Pass the metadata in the Durable Object methods// * @param mainDo - The main Durable Object class// * @param doIdentifier - The identifier of the Durable Object
export class RpcDO extends RpcTarget { constructor( private mainDo: MyDurableObject, private doIdentifier: string, ) { super(); }
// * Pass the user's name to the Durable Object method // * @param userName - The user's name to pass to the Durable Object method
async computeMessage(userName: string): Promise<string> { // Call the Durable Object method and pass the user's name and the Durable Object identifier return this.mainDo.computeMessage(userName, this.doIdentifier); }
// * Call the Durable Object method without using the Durable Object identifier // * @param userName - The user's name to pass to the Durable Object method
async simpleGreeting(userName: string) { return this.mainDo.simpleGreeting(userName); }}
// * Create a Durable Object class// * You can use the RpcDO class to set the Durable Object metadata
export class MyDurableObject extends DurableObject<Env> { constructor(ctx: DurableObjectState, env: Env) { super(ctx, env); }
// * Initialize the RpcDO class // * You can set the Durable Object metadata here // * It returns an instance of the RpcDO class // * @param doIdentifier - The identifier of the Durable Object
async setMetaData(doIdentifier: string) { return new RpcDO(this, doIdentifier); }
// * Function that computes a greeting message using the user's name and DO identifier // * @param userName - The user's name to include in the greeting // * @param doIdentifier - The identifier of the Durable Object
async computeMessage( userName: string, doIdentifier: string, ): Promise<string> { console.log({ userName: userName, durableObjectIdentifier: doIdentifier, }); return `Hello, ${userName}! The identifier of this DO is ${doIdentifier}`; }
// * Function that is not in the RpcTarget // * Not every function has to be in the RpcTarget
private async notInRpcTarget() { return "This is not in the RpcTarget"; }
// * Function that takes the user's name and does not use the Durable Object identifier // * @param userName - The user's name to include in the greeting
async simpleGreeting(userName: string) { // Call the private function that is not in the RpcTarget console.log(this.notInRpcTarget());
return `Hello, ${userName}! This doesn't use the DO identifier.`; }}
export default { async fetch(request, env, ctx): Promise<Response> { let id: DurableObjectId = env.MY_DURABLE_OBJECT.idFromName( new URL(request.url).pathname, ); let stub = env.MY_DURABLE_OBJECT.get(id);
// * Set the Durable Object metadata using the RpcTarget // * Notice that no await is needed here
const rpcTarget = stub.setMetaData(id.name ?? "default");
// Call the Durable Object method using the RpcTarget. // The DO identifier is passed in the RpcTarget const greeting = await rpcTarget.computeMessage("world");
// Call the Durable Object method that does not use the Durable Object identifier const simpleGreeting = await rpcTarget.simpleGreeting("world");
// Clean up the RpcTarget. try { (await rpcTarget)[Symbol.dispose]?.(); console.log("RpcTarget cleaned up."); } catch (e) { console.error({ message: "RpcTarget could not be cleaned up.", error: String(e), errorProperties: e, }); }
return new Response(greeting, { status: 200 }); },} satisfies ExportedHandler<Env>;
This example persists the Durable Object metadata. It demonstrates similar steps as the previous example, but uses Durable Object storage to store the identifier, eliminating the need to pass it through the RpcTarget.
import { DurableObject, RpcTarget } from "cloudflare:workers";
// * Create an RpcDO class that extends RpcTarget// * Use this class to set the Durable Object metadata// * Pass the metadata in the Durable Object methods// * @param mainDo - The main Durable Object class// * @param doIdentifier - The identifier of the Durable Object
export class RpcDO extends RpcTarget { constructor( private mainDo: MyDurableObject, private doIdentifier: string, ) { super(); }
// * Pass the user's name to the Durable Object method // * @param userName - The user's name to pass to the Durable Object method
async computeMessage(userName: string): Promise<string> { // Call the Durable Object method and pass the user's name and the Durable Object identifier return this.mainDo.computeMessage(userName, this.doIdentifier); }
// * Call the Durable Object method without using the Durable Object identifier // * @param userName - The user's name to pass to the Durable Object method
async simpleGreeting(userName: string) { return this.mainDo.simpleGreeting(userName); }}
// * Create a Durable Object class// * You can use the RpcDO class to set the Durable Object metadata
export class MyDurableObject extends DurableObject<Env> { constructor(ctx: DurableObjectState, env: Env) { super(ctx, env); }
// * Initialize the RpcDO class // * You can set the Durable Object metadata here // * It returns an instance of the RpcDO class // * @param doIdentifier - The identifier of the Durable Object
async setMetaData(doIdentifier: string) { // Use DO storage to store the Durable Object identifier await this.ctx.storage.put("doIdentifier", doIdentifier); return new RpcDO(this, doIdentifier); }
// * Function that computes a greeting message using the user's name and DO identifier // * @param userName - The user's name to include in the greeting
async computeMessage(userName: string): Promise<string> { // Get the DO identifier from storage const doIdentifier = await this.ctx.storage.get("doIdentifier"); console.log({ userName: userName, durableObjectIdentifier: doIdentifier, }); return `Hello, ${userName}! The identifier of this DO is ${doIdentifier}`; }
// * Function that is not in the RpcTarget // * Not every function has to be in the RpcTarget
private async notInRpcTarget() { return "This is not in the RpcTarget"; }
// * Function that takes the user's name and does not use the Durable Object identifier // * @param userName - The user's name to include in the greeting
async simpleGreeting(userName: string) { // Call the private function that is not in the RpcTarget console.log(this.notInRpcTarget());
return `Hello, ${userName}! This doesn't use the DO identifier.`; }}
export default { async fetch(request, env, ctx): Promise<Response> { let id: DurableObjectId = env.MY_DURABLE_OBJECT.idFromName( new URL(request.url).pathname, ); let stub = env.MY_DURABLE_OBJECT.get(id);
// * Set the Durable Object metadata using the RpcTarget // * Notice that no await is needed here
const rpcTarget = stub.setMetaData(id.name ?? "default");
// Call the Durable Object method using the RpcTarget. // The DO identifier is stored in the Durable Object's storage const greeting = await rpcTarget.computeMessage("world");
// Call the Durable Object method that does not use the Durable Object identifier const simpleGreeting = await rpcTarget.simpleGreeting("world");
// Clean up the RpcTarget. try { (await rpcTarget)[Symbol.dispose]?.(); console.log("RpcTarget cleaned up."); } catch (e) { console.error({ message: "RpcTarget could not be cleaned up.", error: String(e), errorProperties: e, }); }
return new Response(greeting, { status: 200 }); },} satisfies ExportedHandler<Env>;
Was this helpful?
- Resources
- API
- New to Cloudflare?
- Products
- Sponsorships
- Open Source
- Support
- Help Center
- System Status
- Compliance
- GDPR
- Company
- cloudflare.com
- Our team
- Careers
- 2025 Cloudflare, Inc.
- Privacy Policy
- Terms of Use
- Report Security Issues
- Trademark