Now that you have added the possible actions, including their return values, use events to notify players. Your blockchain can now create and play games. However, it does not inform the outside world about this in a convenient way. That is where events come in - but what do you need to emit them?
Imagine a potential or current player waiting for their turn. It is not practical to look at all the transactions and search for the ones signifying the player's turn. It is better to listen to known events that let clients determine which player's turn it is.
Adding events to your application is as simple as:
Start with the event that announces the creation of a new game. The goal is to:
Inform the players about the game.
Make it easy for the players to find the relevant game.
Define new keys in x/checkers/types/keys.go:
Copy
const(
GameCreatedEventType ="new-game-created"// Indicates what event type to listen to
GameCreatedEventCreator ="creator"// Subsidiary information
GameCreatedEventGameIndex ="game-index"// What game is relevant
GameCreatedEventBlack ="black"// Is it relevant to me?
GameCreatedEventRed ="red"// Is it relevant to me?) x checkers types keys.go View source
Emit the event in your handler file x/checkers/keeper/msg_server_create_game.go:
The created transaction to play a move informs the opponent about:
Which player is relevant.
Which game the move relates to.
When the move happened.
The move's outcome.
Whether the game was won.
Contrary to the create game event, which alerted the players about a new game, the players now know which game IDs to watch for. There is no need to repeat the players' addresses, the game ID is information enough.
You define new keys in x/checkers/types/keys.go similarly:
The unit tests you have created so far still pass. However you also want to confirm that the events have been emitted in both situations. The events are recorded in the context, so the test is a little bit different. In msg_server_create_game_test.go, add this test:
How can you guess the order of elements? Easily, as you created them in this order. Alternatively, you can peek by using Visual Studio Code:
Put a breakpoint on the line after event := events[0].
Run this test in debug mode: right-click the green arrow next to the test name.
Observe the live values on the left.
As for the events emitted during the play move test, there are two of them: one for the creation and the other for the play. Because this is a unit test and each action is not isolated into individual transactions, the context collects all events emitted during the test. It just so happens that the context prepends them - the newest one is at index 0. Which is why, when you fetch them, the play event is at events[0].
When two players play one after the other, the context collates the attributes of move-played all together in a single array in an appending fashion, with the older attributes at the lower indices, starting at 0. For instance, you have to rely on array slices like event.Attributes[5:] to test the attributes of the second move-played event:
This confirms that the play event is emitted as expected. You can confirm the same for the game created event.
When you are done with this exercise you can stop Ignite's chain serve.
synopsis
To summarize, this section has explored:
How to define event types and then emit events to cause the UI to notify players of game actions as they occur, such as creating games and playing moves.
How listening to known events which let clients determine which player must move next is better than the impractical alternative of examining all transactions to search for the ones which signify a player's turn.
How to define a Game-created event that will notify the participating players and make it easy for them to find the game.
How to define a Player-moved event that will indicate which player and game is involved, when the move occurred, the move's outcome, and whether the game was won as a result.
How to test your code to ensure that it functions as desired.
How to interact with the CLI to check the effectiveness of an emitted event.