title |
---|
Runtime Events |
A Substrate runtime module can emit events when it wants to notify external entities about changes or conditions in the runtime to external entities like users, chain explorers, or dApps.
You can define what events your module emits, what information is contained within those events, and when those events are emitted.
Runtime events are created with the decl_event!
macro.
decl_event!(
pub enum Event<T> where AccountId = <T as Trait>::AccountId {
/// Set a value.
ValueSet(u32, AccountId),
}
);
The Event
enum needs to be declared in your runtime's configuration trait.
pub trait Trait: system::Trait {
type Event: From<Event<Self>> + Into<<Self as system::Trait>::Event>;
}
The events for your module need to be exposed to your Substrate's runtime (/runtime/src/lib.rs
).
First you need to implement the Event type in your module's configuration trait:
// runtime/src/lib.rs
impl template::Trait for Runtime {
type Event = Event;
}
Then you need to add the Event
type to your construct_runtime!
macro:
// runtime/src/lib.rs
construct_runtime!(
pub enum Runtime where
Block = Block,
NodeBlock = opaque::Block,
UncheckedExtrinsic = UncheckedExtrinsic
{
// --snip--
TemplateModule: template::{Module, Call, Storage, Event<T>},
//--add-this------------------------------------->^^^^^^^^
}
);
Note: You may or may not need the
<T>
parameter depending on whether your events use generic types. In our example it does, and is included above.
Substrate provides a default implementation of how to deposit an event that is defined in the
decl_module!
macro.
decl_module! {
pub struct Module<T: Trait> for enum Call where origin: T::Origin {
// Default implementation of `deposit_event`
fn deposit_event() = default;
fn set_value(origin, value: u64) {
let sender = ensure_signed(origin)?;
// --snip--
Self::deposit_event(RawEvent::ValueSet(value, sender));
}
}
}
The default behavior of this function is to call
deposit_event
from the FRAME system, which writes the event to storage.
This function places the event in the System module's runtime storage for that block. At the beginning of a new block, the System module automatically removes all events that were stored from the previous block.
Events deposited using the default implementation will be directly supported by downstream libraries
like the Polkadot-JS api, however you can implement your own
deposit_event
function if you want to handle events differently.
Events can emit any type which supports the Parity SCALE codec.
In the case where you want to use Runtime generic types like AccountId
or Balances
, you need to
include a where
clause to define
those types as shown in the example above.
The Substrate RPC does not directly expose an endpoint for querying events. If you used the default implementation, you can see the list of events for the current block by querying the storage of the System module. Otherwise, the Polkadot-JS api supports a WebSocket subscription on runtime events.
- Learn more about the macros used in Substrate runtime development.
- Learn more about using the Polkadot JS api.
These Substrate Recipes offer examples of how runtime events are used: