Modules connected by a channel must agree on what application data they are sending over the
channel, as well as how they will encode/decode it. This process is not specified by IBC as it is up
to each application module to determine how to implement this agreement. However, for most
applications this will happen as a version negotiation during the channel handshake. While more
complex version negotiation is possible to implement inside the channel opening handshake, a very
simple version negotiation is implemented in the ibc-transfer module.Thus, a module must define its custom packet data structure, along with a well-defined way to
encode and decode it to and from []byte.
/ Custom packet data defined in application moduletype CustomPacketData struct { / Custom fields ...}EncodePacketData(packetData CustomPacketData) []byte { / encode packetData to bytes}DecodePacketData(encoded []byte) (CustomPacketData) { / decode from bytes to packet data}
Note that the CustomPacketData struct is defined in the proto definition and then compiled by the protobuf compiler.
Then a module must encode its packet data before sending it through IBC.
/ retrieve the dynamic capability for this channel channelCap := scopedKeeper.GetCapability(ctx, channelCapName)/ Sending custom application packet data data := EncodePacketData(customPacketData)/ Send packet to IBC, authenticating with channelCapsequence, err := IBCChannelKeeper.SendPacket( ctx, channelCap, sourcePort, sourceChannel, timeoutHeight, timeoutTimestamp, data,)
A module receiving a packet must decode the PacketData into a structure it expects so that it can
act on it.
/ Receiving custom application packet data (in OnRecvPacket) packetData := DecodePacketData(packet.Data)/ handle received custom packet data
Modules may commit an acknowledgement upon receiving and processing a packet in the case of synchronous packet processing.
In the case where a packet is processed at some later point after the packet has been received (asynchronous execution), the acknowledgement
will be written once the packet has been processed by the application which may be well after the packet receipt.NOTE: Most blockchain modules will want to use the synchronous execution model in which the module processes and writes the acknowledgement
for a packet as soon as it has been received from the IBC module.This acknowledgement can then be relayed back to the original sender chain, which can take action
depending on the contents of the acknowledgement.Just as packet data was opaque to IBC, acknowledgements are similarly opaque. Modules must pass and
receive acknowledegments with the IBC modules as byte strings.Thus, modules must agree on how to encode/decode acknowledgements. The process of creating an
acknowledgement struct along with encoding and decoding it, is very similar to the packet data
example above. ICS 04
specifies a recommended format for acknowledgements. This acknowledgement type can be imported from
channel types.While modules may choose arbitrary acknowledgement structs, a default acknowledgement types is provided by IBC here:
/ Acknowledgement is the recommended acknowledgement format to be used by/ app-specific protocols./ NOTE: The field numbers 21 and 22 were explicitly chosen to avoid accidental/ conflicts with other protobuf message formats used for acknowledgements./ The first byte of any message with this format will be the non-ASCII values/ `0xaa` (result) or `0xb2` (error). Implemented as defined by ICS:/ https://github.com/cosmos/ibc/tree/master/spec/core/ics-004-channel-and-packet-semantics#acknowledgement-envelopemessage Acknowledgement { / response contains either a result or an error and must be non-empty oneof response { bytes result = 21; string error = 22; }}