Skip to main content

Inter-canister calls

Overview

One of the most important features of ICP for developers is the ability to call functions in one canister from another canister. This capability to make calls between canisters, also sometimes referred to as inter-canister calls, enables you to reuse and share functionality in multiple dapps.

For example, you might want to create a dapp for professional networking, organizing community events, or hosting fundraising activities. Each of these dapps might have a social component that enables users to identify social relationships based on some criteria or shared interest, such as friends and family or current and former colleagues.

To address this social component, you might want to create a single canister for storing user relationships then write your professional networking, community organizing, or fundraising application to import and call functions that are defined in the canister for social connections. You could then build additional applications to use the social connections canister or extend the features provided by the social connections canister to make it useful to an even broader community of other developers.

This example will showcase a simple way to configure inter-canister calls that can be used as the foundation for more elaborate projects and use-cases such as those mentioned above.

Example

Consider the following code for Canister1:

import Canister2 "canister:canister2";

actor Canister1 {
public func main() : async Nat {
return await Canister2.getValue();
};
};

Then, consider the following code for Canister2:

import Debug "mo:base/Debug";

actor Canister2 {
public func getValue() : async Nat {
Debug.print("Hello from canister 2!");
return 10;
};
};

To make an inter-canister call from canister1 to canister2, you can use the dfx command:

dfx canister call canister1 main

The output should resemble the following:

2023-06-15 15:53:39.567801 UTC: [Canister ajuq4-ruaaa-aaaaa-qaaga-cai] Hello from canister 2!
(10 : nat)

Alternatively, you can also use a canister id to access a previously deployed canister by using the following piece of code in canister1:

actor {
public func main(canisterId: Text) : async Nat {
let canister2 = actor(canisterId): actor { getValue: () -> async Nat };
return await canister2.getValue();
};
};

Then, use the following call, replacing canisterID with the principal ID of a previously deployed canister:

dfx canister call canister1 main "canisterID"

Advanced usage

If the method name or input types are unknown at compile time, it's possible to call arbitrary canister methods using the ExperimentalInternetComputer module.

Here is an example which you can modify for your specific use case:

import IC "mo:base/ExperimentalInternetComputer";
import Debug "mo:base/Debug";

actor AdvancedCanister1 {
public func main(canisterId : Principal) : async Nat {
// Define the method name and input args
let name = "getValue";
let args = (123);

// Call the method
let encodedArgs = to_candid (args);
let encodedValue = await IC.call(canisterId, name, encodedArgs);

// Decode the return value
let ?value : ?Nat = from_candid encodedValue
else Debug.trap("Unexpected return value");
return value;
}
}
import Debug "mo:base/Debug";

actor AdvancedCanister2 {
public func getValue(number: Nat) : async Nat {
Debug.print("Hello from advanced canister 2!");
return number * 2;
};
};
Logo