Implement the Order Books
In this chapter, you implement the logic to create order books.
In the Cosmos SDK, the state is stored in a key-value store. Each order book is stored under a unique key that is composed of four values:
- Port ID
- Channel ID
- Source denom
- Target denom
For example, an order book for marscoin and venuscoin could be stored under dex-channel-4-marscoin-venuscoin
.
First, define a function that returns an order book store key:
// x/dex/types/keys.go
package types
import "fmt"
// ...
func OrderBookIndex(portID string, channelID string, sourceDenom string, targetDenom string) string {
return fmt.Sprintf("%s-%s-%s-%s", portID, channelID, sourceDenom, targetDenom)
}
The send-create-pair
command is used to create order books. This command:
- Creates and broadcasts a transaction with a message of type
SendCreatePair
. - The message gets routed to the
dex
module. - Finally, a
SendCreatePair
keeper method is called.
You need the send-create-pair
command to do the following:
- When processing
SendCreatePair
message on the source chain:- Check that an order book with the given pair of denoms does not yet exist.
- Transmit an IBC packet with information about port, channel, source denoms, and target denoms.
- After the packet is received on the target chain:
- Check that an order book with the given pair of denoms does not yet exist on the target chain.
- Create a new order book for buy orders.
- Transmit an IBC acknowledgement back to the source chain.
- After the acknowledgement is received on the source chain:
- Create a new order book for sell orders.
Message Handling in SendCreatePair
The SendCreatePair
function was created during the IBC packet scaffolding. The function creates an IBC packet,
populates it with source and target denoms, and transmits this packet over IBC.
Now, add the logic to check for an existing order book for a particular pair of denoms:
// x/dex/keeper/msg_server_create_pair.go
package keeper
import (
"errors"
// ...
)
func (k msgServer) SendCreatePair(goCtx context.Context, msg *types.MsgSendCreatePair) (*types.MsgSendCreatePairResponse, error) {
ctx := sdk.UnwrapSDKContext(goCtx)
// Get an order book index
pairIndex := types.OrderBookIndex(msg.Port, msg.ChannelID, msg.SourceDenom, msg.TargetDenom)
// If an order book is found, return an error
_, found := k.GetSellOrderBook(ctx, pairIndex)
if found {
return &types.MsgSendCreatePairResponse{}, errors.New("the pair already exist")
}
// Construct the packet
var packet types.CreatePairPacketData
packet.SourceDenom = msg.SourceDenom
packet.TargetDenom = msg.TargetDenom
// Transmit the packet
_, err := k.TransmitCreatePairPacket(
ctx,
packet,
msg.Port,
msg.ChannelID,
clienttypes.ZeroHeight(),
msg.TimeoutTimestamp,
)
if err != nil {
return nil, err
}
return &types.MsgSendCreatePairResponse{}, nil
}
Lifecycle of an IBC Packet
During a successful transmission, an IBC packet goes through these stages:
- Message processing before packet transmission on the source chain
- Reception of a packet on the target chain
- Acknowledgment of a packet on the source chain
- Timeout of a packet on the source chain
In the following section, implement the packet reception logic in the OnRecvCreatePairPacket
function and the packet
acknowledgement logic in the OnAcknowledgementCreatePairPacket
function.
Leave the Timeout function empty.
Receive an IBC packet
The protocol buffer definition defines the data that an order book contains.
Add the OrderBook
and Order
messages to the order.proto
file.
First, add the proto buffer files to build the Go code files. You can modify these files for the purpose of your app.
Create a new order.proto
file in the proto/interchange/dex
directory and add the content:
// proto/interchange/dex/order.proto
syntax = "proto3";
package interchange.dex;
option go_package = "interchange/x/dex/types";
message OrderBook {
int32 idCount = 1;
repeated Order orders = 2;
}
message Order {
int32 id = 1;
string creator = 2;
int32 amount = 3;
int32 price = 4;
}
Modify the buy_order_book.proto
file to have the fields for creating a buy order on the order book.
Don't forget to add the import as well.
Tip: Don't forget to add the import as well.
// proto/interchange/dex/buy_order_book.proto
// ...
import "interchange/dex/order.proto";
message BuyOrderBook {
// ...
OrderBook book = 4;
}
Modify the sell_order_book.proto
file to add the order book into the buy order book.
The proto definition for the SellOrderBook
looks like:
// proto/interchange/dex/sell_order_book.proto
// ...
import "interchange/dex/order.proto";
message SellOrderBook {
// ...
OrderBook book = 4;
}
Now, use Ignite CLI to build the proto files for the send-create-pair
command:
ignite generate proto-go --yes
Start enhancing the functions for the IBC packets.
Create a new file x/dex/types/order_book.go
.
Add the new order book function to the corresponding Go file:
// x/dex/types/order_book.go
package types
func NewOrderBook() OrderBook {
return OrderBook{
IdCount: 0,
}
}
To create a new buy order book type, define NewBuyOrderBook
in a new file x/dex/types/buy_order_book.go
:
// x/dex/types/buy_order_book.go
package types
func NewBuyOrderBook(AmountDenom string, PriceDenom string) BuyOrderBook {
book := NewOrderBook()
return BuyOrderBook{
AmountDenom: AmountDenom,
PriceDenom: PriceDenom,
Book: &book,
}
}
When an IBC packet is received on the target chain, the module must check whether a book already exists. If not, then create a buy order book for the specified denoms.
// x/dex/keeper/create_pair.go
package keeper
// ...
func (k Keeper) OnRecvCreatePairPacket(ctx sdk.Context, packet channeltypes.Packet, data types.CreatePairPacketData) (packetAck types.CreatePairPacketAck, err error) {
// ...
// Get an order book index
pairIndex := types.OrderBookIndex(packet.SourcePort, packet.SourceChannel, data.SourceDenom, data.TargetDenom)
// If an order book is found, return an error
_, found := k.GetBuyOrderBook(ctx, pairIndex)
if found {
return packetAck, errors.New("the pair already exist")
}
// Create a new buy order book for source and target denoms
book := types.NewBuyOrderBook(data.SourceDenom, data.TargetDenom)
// Assign order book index
book.Index = pairIndex
// Save the order book to the store
k.SetBuyOrderBook(ctx, book)
return packetAck, nil
}
Receive an IBC Acknowledgement
When an IBC acknowledgement is received on the source chain, the module must check whether a book already exists. If not, create a sell order book for the specified denoms.
Create a new file x/dex/types/sell_order_book.go
.
Insert the NewSellOrderBook
function which creates a new sell order book.
// x/dex/types/sell_order_book.go
package types
func NewSellOrderBook(AmountDenom string, PriceDenom string) SellOrderBook {
book := NewOrderBook()
return SellOrderBook{
AmountDenom: AmountDenom,
PriceDenom: PriceDenom,
Book: &book,
}
}
Modify the Acknowledgement function in the x/dex/keeper/create_pair.go
file:
// x/dex/keeper/create_pair.go
package keeper
// ...
func (k Keeper) OnAcknowledgementCreatePairPacket(ctx sdk.Context, packet channeltypes.Packet, data types.CreatePairPacketData, ack channeltypes.Acknowledgement) error {
switch dispatchedAck := ack.Response.(type) {
case *channeltypes.Acknowledgement_Error:
return nil
case *channeltypes.Acknowledgement_Result:
// Decode the packet acknowledgment
var packetAck types.CreatePairPacketAck
if err := types.ModuleCdc.UnmarshalJSON(dispatchedAck.Result, &packetAck); err != nil {
// The counter-party module doesn't implement the correct acknowledgment format
return errors.New("cannot unmarshal acknowledgment")
}
// Set the sell order book
pairIndex := types.OrderBookIndex(packet.SourcePort, packet.SourceChannel, data.SourceDenom, data.TargetDenom)
book := types.NewSellOrderBook(data.SourceDenom, data.TargetDenom)
book.Index = pairIndex
k.SetSellOrderBook(ctx, book)
return nil
default:
// The counter-party module doesn't implement the correct acknowledgment format
return errors.New("invalid acknowledgment format")
}
}
In this section, you implemented the logic behind the new send-create-pair
command:
- When an IBC packet is received on the target chain,
send-create-pair
command creates a buy order book. - When an IBC acknowledgement is received on the source chain, the
send-create-pair
command creates a sell order book.
Implement the appendOrder Function to Add Orders to the Order Book
// x/dex/types/order_book.go
package types
import (
"errors"
"sort"
)
func NewOrderBook() OrderBook {
return OrderBook{
IdCount: 0,
}
}
const (
MaxAmount = int32(100000)
MaxPrice = int32(100000)
)
type Ordering int
const (
Increasing Ordering = iota
Decreasing
)
var (
ErrMaxAmount = errors.New("max amount reached")
ErrMaxPrice = errors.New("max price reached")
ErrZeroAmount = errors.New("amount is zero")
ErrZeroPrice = errors.New("price is zero")
ErrOrderNotFound = errors.New("order not found")
)
The AppendOrder
function initializes and appends a new order to an order book from the order information:
// x/dex/types/order_book.go
func (book *OrderBook) appendOrder(creator string, amount int32, price int32, ordering Ordering) (int32, error) {
if err := checkAmountAndPrice(amount, price); err != nil {
return 0, err
}
// Initialize the order
var order Order
order.Id = book.GetNextOrderID()
order.Creator = creator
order.Amount = amount
order.Price = price
// Increment ID tracker
book.IncrementNextOrderID()
// Insert the order
book.insertOrder(order, ordering)
return order.Id, nil
}
Implement the checkAmountAndPrice Function For an Order
The checkAmountAndPrice
function checks for the correct amount or price:
// x/dex/types/order_book.go
func checkAmountAndPrice(amount int32, price int32) error {
if amount == int32(0) {
return ErrZeroAmount
}
if amount > MaxAmount {
return ErrMaxAmount
}
if price == int32(0) {
return ErrZeroPrice
}
if price > MaxPrice {
return ErrMaxPrice
}
return nil
}
Implement the GetNextOrderID Function
The GetNextOrderID
function gets the ID of the next order to append:
// x/dex/types/order_book.go
func (book OrderBook) GetNextOrderID() int32 {
return book.IdCount
}
Implement the IncrementNextOrderID Function
The IncrementNextOrderID
function updates the ID count for orders:
// x/dex/types/order_book.go
func (book *OrderBook) IncrementNextOrderID() {
// Even numbers to have different ID than buy orders
book.IdCount++
}
Implement the insertOrder Function
The insertOrder
function inserts the order in the book with the provided order:
// x/dex/types/order_book.go
func (book *OrderBook) insertOrder(order Order, ordering Ordering) {
if len(book.Orders) > 0 {
var i int
// get the index of the new order depending on the provided ordering
if ordering == Increasing {
i = sort.Search(len(book.Orders), func(i int) bool { return book.Orders[i].Price > order.Price })
} else {
i = sort.Search(len(book.Orders), func(i int) bool { return book.Orders[i].Price < order.Price })
}
// insert order
orders := append(book.Orders, &order)
copy(orders[i+1:], orders[i:])
orders[i] = &order
book.Orders = orders
} else {
book.Orders = append(book.Orders, &order)
}
}
This completes the order book setup.
Now is a good time to save the state of your implementation. Because your project is in a local repository, you can use git. Saving your current state lets you jump back and forth in case you introduce errors or need a break.
git add .
git commit -m "Create Order Books"
In the next chapter, you learn how to deal with vouchers by minting and burning vouchers and locking and unlocking native blockchain token in your app.