2019-12-17 03:44:44 -08:00
<!--
2019-12-10 06:29:46 -08:00
order: 7
2019-12-17 03:44:44 -08:00
-->
2019-12-10 06:29:46 -08:00
# Events
2020-03-24 09:20:30 -07:00
`Event` s are objects that contain information about the execution of the application. They are mainly used by service providers like block explorers and wallet to track the execution of various messages and index transactions. {synopsis}
## Pre-requisite Readings
2019-12-10 06:29:46 -08:00
- [Anatomy of an SDK application ](../basics/app-anatomy.md ) {prereq}
## Events
2020-01-06 08:49:31 -08:00
Events are implemented in the Cosmos SDK as an alias of the ABCI `Event` type and
take the form of: `{eventType}.{eventAttribute}={value}` .
2019-12-10 06:29:46 -08:00
2020-02-26 23:37:16 -08:00
+++ https://github.com/tendermint/tendermint/blob/bc572217c07b90ad9cee851f193aaa8e9557cbc7/abci/types/types.pb.go#L2187-L2193
2019-12-10 06:29:46 -08:00
2020-01-06 08:49:31 -08:00
Events contain:
2019-12-10 06:29:46 -08:00
2020-01-06 08:49:31 -08:00
- A `type` , which is meant to categorize an event at a high-level (e.g. by module or action).
- A list of `attributes` , which are key-value pairs that give more information about
the categorized `event` .
+++ https://github.com/cosmos/cosmos-sdk/blob/7d7821b9af132b0f6131640195326aa02b6751db/types/events.go#L51-L56
2019-12-10 06:29:46 -08:00
2020-01-06 08:49:31 -08:00
Events are returned to the underlying consensus engine in the response of the following ABCI messages:
2019-12-10 06:29:46 -08:00
2020-01-06 08:49:31 -08:00
- [`BeginBlock` ](./baseapp.md#beginblock )
- [`EndBlock` ](./baseapp.md#endblock )
- [`CheckTx` ](./baseapp.md#checktx )
- [`DeliverTx` ](./baseapp.md#delivertx )
Events, the `type` and `attributes` , are defined on a **per-module basis** in the module's
2020-04-17 07:01:42 -07:00
`/types/events.go` file, and triggered from the module's [`handler` ](../building-modules/handler.md )
2020-01-06 08:49:31 -08:00
via the [`EventManager` ](#eventmanager ). In addition, each module documents its events under
`spec/xx_events.md` .
2019-12-10 06:29:46 -08:00
## EventManager
2020-01-06 08:49:31 -08:00
In Cosmos SDK applications, events are managed by an abstraction called the `EventManager` .
Internally, the `EventManager` tracks a list of `Events` for the entire execution flow of a
transaction or `BeginBlock` /`EndBlock`.
2019-12-10 06:29:46 -08:00
+++ https://github.com/cosmos/cosmos-sdk/blob/7d7821b9af132b0f6131640195326aa02b6751db/types/events.go#L16-L20
2020-01-06 08:49:31 -08:00
The `EventManager` comes with a set of useful methods to manage events. Among them, the one that is
used the most by module and application developers is the `EmitEvent` method, which tracks
an `event` in the `EventManager` .
2019-12-10 06:29:46 -08:00
+++ https://github.com/cosmos/cosmos-sdk/blob/7d7821b9af132b0f6131640195326aa02b6751db/types/events.go#L29-L31
2020-01-06 08:49:31 -08:00
Module developers should handle event emission via the `EventManager#EmitEvent` in each message
`Handler` and in each `BeginBlock` /`EndBlock` handler. The `EventManager` is accessed via
the [`Context` ](./context.md ), where event emission generally follows this pattern:
2019-12-10 06:29:46 -08:00
```go
ctx.EventManager().EmitEvent(
2020-01-06 08:49:31 -08:00
sdk.NewEvent(eventType, sdk.NewAttribute(attributeKey, attributeValue)),
)
2019-12-10 06:29:46 -08:00
```
2020-01-29 05:43:48 -08:00
Module's `handler` function should also set a new `EventManager` to the `context` to isolate emitted events per `message` :
```go
func NewHandler(keeper Keeper) sdk.Handler {
return func(ctx sdk.Context, msg sdk.Msg) (*sdk.Result, error) {
ctx = ctx.WithEventManager(sdk.NewEventManager())
switch msg := msg.(type) {
```
2020-01-06 08:49:31 -08:00
See the [`Handler` ](../building-modules/handler.md ) concept doc for a more detailed
view on how to typically implement `Events` and use the `EventManager` in modules.
2019-12-10 06:29:46 -08:00
2020-01-06 08:49:31 -08:00
## Subscribing to Events
2019-12-10 06:29:46 -08:00
2020-01-06 08:49:31 -08:00
It is possible to subscribe to `Events` via Tendermint's [Websocket ](https://tendermint.com/docs/app-dev/subscribing-to-events-via-websocket.html#subscribing-to-events-via-websocket ).
This is done by calling the `subscribe` RPC method via Websocket:
2019-12-10 06:29:46 -08:00
2020-01-06 08:49:31 -08:00
```json
2019-12-10 06:29:46 -08:00
{
"jsonrpc": "2.0",
"method": "subscribe",
"id": "0",
"params": {
2020-01-06 08:49:31 -08:00
"query": "tm.event='eventCategory' AND eventType.eventAttribute='attributeValue'"
2019-12-10 06:29:46 -08:00
}
}
```
The main `eventCategory` you can subscribe to are:
- `NewBlock` : Contains `events` triggered during `BeginBlock` and `EndBlock` .
- `Tx` : Contains `events` triggered during `DeliverTx` (i.e. transaction processing).
2020-01-06 08:49:31 -08:00
- `ValidatorSetUpdates` : Contains validator set updates for the block.
2019-12-10 06:29:46 -08:00
2020-01-06 08:49:31 -08:00
These events are triggered from the `state` package after a block is committed. You can get the
full list of `event` categories [here ](https://godoc.org/github.com/tendermint/tendermint/types#pkg-constants ).
2019-12-10 06:29:46 -08:00
2020-04-17 07:01:42 -07:00
The `type` and `attribute` value of the `query` allow you to filter the specific `event` you are looking for. For example, a `transfer` transaction triggers an `event` of type `Transfer` and has `Recipient` and `Sender` as `attributes` (as defined in the [`events` file of the `bank` module ](https://github.com/cosmos/cosmos-sdk/blob/master/x/bank/types/events.go )). Subscribing to this `event` would be done like so:
2019-12-10 06:29:46 -08:00
2020-01-06 08:49:31 -08:00
```json
2019-12-10 06:29:46 -08:00
{
"jsonrpc": "2.0",
"method": "subscribe",
"id": "0",
"params": {
"query": "tm.event='Tx' AND transfer.sender='senderAddress'"
}
}
```
2020-07-07 03:19:36 -07:00
where `senderAddress` is an address following the [`AccAddress` ](../basics/accounts.md#addresses ) format.
2019-12-10 06:29:46 -08:00
## Next {hide}
2020-07-07 03:19:36 -07:00
Learn about SDK [telemetry ](./telemetry.md ) {hide}