z2aaa_on_ack_packet_filetest.gno
20.11 Kb · 589 lines
1// PKGPATH: gno.land/r/aib/main
2package main
3
4import (
5 "encoding/base64"
6 "encoding/hex"
7 "testing"
8 "time"
9
10 "gno.land/p/aib/ibc/lightclient/tendermint"
11 tmtesting "gno.land/p/aib/ibc/lightclient/tendermint/testing"
12 "gno.land/p/aib/ibc/types"
13 "gno.land/p/aib/ics23"
14 "gno.land/r/aib/ibc/apps/transfer"
15 "gno.land/r/aib/ibc/core"
16)
17
18// OnAcknowledgementPacket: success w/ refund of IBC voucher token
19func main(cur realm) {
20 var (
21 chainID = "chain-id-2"
22 trustedHeight = types.NewHeight(2, 2)
23 clientState = tmtesting.NewClientState(chainID, trustedHeight)
24 // NOTE this apphash was provided by the gen-proof command below (RecvPacket proof).
25 apphash, _ = hex.DecodeString("948f52451fd35947b960d2a8d316e3c1d4cae9318a3d71923efeea9d7e3db57b")
26 // priv=8a6cAbQSpDbebmcTEhCMPhhr/SkL/2pizo60yzHRkN9Uyk7RHOZm7g4xW+yeJh147/Z4/6HXF6gBwcFNkLsZ/A==
27 val1 = tendermint.NewValidator("9DIBYr64rywKO3Kk6+743xDHcEU=",
28 "VMpO0RzmZu4OMVvsniYdeO/2eP+h1xeoAcHBTZC7Gfw=", 10)
29 // priv=nWg6ETc62tyxd94lh8fFaQnZKaAW6vlS0L/4lfseJuI14ZXUKp7AZROkflLFVF+SBg4wJVfzgzIKyWq3D066+g==
30 val2 = tendermint.NewValidator("y+naL3ubs9q1bXrY9+uRxY9c+J8=",
31 "NeGV1CqewGUTpH5SxVRfkgYOMCVX84MyCslqtw9Ouvo=", 10)
32 trustedValset = tendermint.NewValset(val1, val2)
33 consensusState = tmtesting.GenConsensusState(time.Now(), apphash, trustedValset.Hash())
34 counterpartyID = "07-tendermint-42"
35 )
36 clientID := core.CreateClient(cross(cur), clientState, consensusState)
37 core.RegisterCounterparty(cross(cur), clientID, [][]byte{[]byte("iavlStoreKey"), []byte("prefix2")}, counterpartyID)
38 relayer := cur.Previous().Address()
39
40 signer := cur.Address()
41 // ibc/F9A67CB19B2CAD2ADEC20AD475BE86DF851DEC2B6F6CABC9B7B781BD9131D18F
42 voucherDenom := transfer.NewDenom("uatone", transfer.NewHop(transfer.PortID, clientID)).IBCDenom()
43
44 // Mint voucher tokens via a real RecvPacket flow.
45 recvPayload := transfer.NewFungibleTokenPacketData("uatone", "100", "atone1user", signer.String(), "")
46 recvPayloadBz := recvPayload.ProtoMarshal()
47 // NOTE this base64 value is used in payload.value in the gen-proof command below.
48 println("Payload proto:", base64.StdEncoding.EncodeToString(recvPayloadBz))
49
50 specs := ics23.IavlSpec()
51 // NOTE code generated by:
52 // go run -C ./cmd/gen-proof . 'prefix2' '07-tendermint-42' 'packet' '{"sequence":1,"source_client":"07-tendermint-42","destination_client":"07-tendermint-1","timeout_timestamp":1234571490,"payloads":[{"source_port":"transfer","destination_port":"transfer","encoding":"application/x-protobuf","value":"CgZ1YXRvbmUSAzEwMBoKYXRvbmUxdXNlciIoZzFnanM4bmZkcmQ1NXozeHU5Z2pjYXRuajI5dmt3dXl4NGZmcXhkOQ==","version":"ics20-1"}]}'
53 recvProof := []ics23.CommitmentProof{
54
55 // iavl proof
56 ics23.CommitmentProof_Exist{
57 Exist: &ics23.ExistenceProof{
58 Key: []byte("\x70\x72\x65\x66\x69\x78\x32\x30\x37\x2d\x74\x65\x6e\x64\x65\x72\x6d\x69\x6e\x74\x2d\x34\x32\x01\x00\x00\x00\x00\x00\x00\x00\x01"),
59 Value: []byte("\x2a\xec\xd3\x00\x8e\x1c\xaa\x6d\x4c\x80\xec\x03\xb7\x0b\x11\xc7\xfb\x00\x4a\x23\x0e\x95\x1e\x30\x00\x32\x3e\x30\xfb\xa1\x33\xde"),
60 Leaf: &ics23.LeafOp{
61 Hash: specs.LeafSpec.Hash,
62 PrehashKey: specs.LeafSpec.PrehashKey,
63 PrehashValue: specs.LeafSpec.PrehashValue,
64 Length: specs.LeafSpec.Length,
65 Prefix: []byte("\x00\x02\x02"),
66 },
67 Path: []*ics23.InnerOp{
68 {
69 Hash: specs.InnerSpec.Hash,
70 Prefix: []byte("\x02\x04\x02\x20\x35\xf8\xea\x80\x53\x90\xe0\x84\x85\x4f\x39\x9b\x42\xcc\xde\xae\xa3\x3a\x1d\xed\xc1\x15\x63\x8a\xc4\x8d\x06\x00\x63\x7d\xba\x1f\x20"),
71 Suffix: []byte(""),
72 },
73 {
74 Hash: specs.InnerSpec.Hash,
75 Prefix: []byte("\x04\x08\x02\x20"),
76 Suffix: []byte("\x20\x79\x8e\x2c\xaa\x96\xfd\xfb\xa3\x76\xdd\xeb\x47\x99\x99\x54\xd2\xf4\x7e\x65\x16\x22\x64\xb0\x53\x6a\xb5\xdf\xf7\xfc\x0a\x2e\x07"),
77 },
78 {
79 Hash: specs.InnerSpec.Hash,
80 Prefix: []byte("\x06\x0c\x02\x20\x9a\xf3\x7d\xd5\x95\xa0\x19\x08\x03\xb5\xe0\x5a\xae\xf4\x2a\xe3\xfa\xd4\x99\xe4\xfb\xe3\x7f\x7c\xd3\x1c\xad\xff\x22\xa9\xee\x74\x20"),
81 Suffix: []byte(""),
82 },
83 },
84 },
85 },
86
87 // rootmulti proof
88 ics23.CommitmentProof_Exist{
89 Exist: &ics23.ExistenceProof{
90 Key: []byte("\x69\x61\x76\x6c\x53\x74\x6f\x72\x65\x4b\x65\x79"),
91 Value: []byte("\xf5\xf0\x9d\x7b\x1d\x33\x59\xf7\x09\x21\x4a\x5d\x28\x91\xb8\x63\xf2\xd4\x7c\xf3\x97\xb6\x55\x89\x82\x05\x00\x31\x14\x0e\x93\x53"),
92 Leaf: &ics23.LeafOp{
93 Hash: specs.LeafSpec.Hash,
94 PrehashKey: specs.LeafSpec.PrehashKey,
95 PrehashValue: specs.LeafSpec.PrehashValue,
96 Length: specs.LeafSpec.Length,
97 Prefix: []byte("\x00"),
98 },
99 Path: []*ics23.InnerOp{},
100 },
101 },
102 }
103
104 recvMsg := types.MsgRecvPacket{
105 Packet: types.Packet{
106 Sequence: 1,
107 SourceClient: counterpartyID,
108 DestinationClient: clientID,
109 TimeoutTimestamp: uint64(time.Now().Add(time.Hour).Unix()),
110 Payloads: []types.Payload{{
111 SourcePort: transfer.PortID,
112 DestinationPort: transfer.PortID,
113 Encoding: transfer.EncodingProtobuf,
114 Value: recvPayloadBz,
115 Version: transfer.V1,
116 }},
117 },
118 ProofCommitment: recvProof,
119 ProofHeight: trustedHeight,
120 }
121 core.RecvPacket(cross(cur), recvMsg)
122
123 // Now send the voucher tokens back (burn)
124 testing.SetOriginCaller(signer)
125 println("signer voucher balance before Transfer:", transfer.VoucherBalanceOf(voucherDenom, signer))
126 sendPacket, sequence := transfer.Transfer(cross(cur), clientID, "atone1user", voucherDenom, 100, uint64(time.Now().Add(time.Hour).Unix()), "")
127 println("\nsigner voucher balance after Transfer:", transfer.VoucherBalanceOf(voucherDenom, signer))
128
129 // UpdateClient to advance to height containing the ack proof
130 testing.SetOriginCaller(relayer)
131 // NOTE code generated by:
132 // go run -C ./cmd/gen-block-signatures . -apphash-hex=5467ca2d19a7b74dbcd20e33bda91932f912d76595872ec856945af96e743b82 -chainid=chain-id-2 -header-time-shift=1 -height=12 -privkeys=8a6cAbQSpDbebmcTEhCMPhhr/SkL/2pizo60yzHRkN9Uyk7RHOZm7g4xW+yeJh147/Z4/6HXF6gBwcFNkLsZ/A==,nWg6ETc62tyxd94lh8fFaQnZKaAW6vlS0L/4lfseJuI14ZXUKp7AZROkflLFVF+SBg4wJVfzgzIKyWq3D066+g==
133 // NOTE apphash generated by the gen-proof command below
134 apphash, _ = hex.DecodeString("5467ca2d19a7b74dbcd20e33bda91932f912d76595872ec856945af96e743b82")
135 trustedHeight = clientState.LatestHeight
136 var (
137 commitTimestamp = tmtesting.ToTime("2025-09-25T07:55:57.306746166Z")
138 newHeight = uint64(12)
139 newTimestamp = consensusState.Timestamp.Add(time.Minute * time.Duration(1))
140 valset = tendermint.NewValset(val1, val2)
141 nextValset = tendermint.NewValset(val1, val2)
142
143 signatures = []tendermint.CommitSig{
144 {
145 BlockIDFlag: tendermint.BlockIDFlagCommit,
146 ValidatorAddress: valset.Validators[0].Address,
147 Timestamp: commitTimestamp,
148 Signature: []byte("\x6b\x64\x5e\xa3\x44\x73\x69\x6b\x4f\x3a\x63\xb0\x9f\x80\x58\xa5\x8f\x2e\xc5\x9b\x01\x9d\x56\x52\x32\xf1\x19\x79\x66\x40\x71\x2c\xb0\x21\xa1\xc7\x0a\x77\xe7\xa9\x25\x32\x67\x2a\x9a\xbb\xd1\xfa\xc5\x32\x37\x8d\x55\x48\xdd\x2f\x26\xfb\xc3\x61\xc3\x7d\x75\x04"),
149 },
150 {
151 BlockIDFlag: tendermint.BlockIDFlagCommit,
152 ValidatorAddress: valset.Validators[1].Address,
153 Timestamp: commitTimestamp,
154 Signature: []byte("\x93\x86\xf3\x29\x39\x59\x06\xa3\x86\x09\x08\xb4\xc4\xa6\x89\xd5\xae\x54\x92\xb1\xcb\x8e\xce\x53\xe5\x93\xfc\x93\xbc\x1b\x4b\x41\x47\x24\xe9\xfd\x1c\x65\xf5\x33\x08\xcc\x5f\x39\x8a\x3d\x5b\x0d\xf3\x5a\xa8\x7e\xde\x0c\xbf\xcd\xac\x49\xeb\xca\xf2\x95\xbf\x06"),
155 },
156 }
157
158 msgHeader = tmtesting.NewMsgHeader(
159 chainID, newTimestamp, apphash, newHeight, trustedHeight, valset,
160 nextValset, trustedValset, signatures,
161 )
162 )
163 core.UpdateClient(cross(cur), clientID, msgHeader)
164
165 // Acknowledge the packet
166 // Generate the proof of acknowledgement written during the RecvPacket of the
167 // counterparty client.
168 // NOTE proof generated by:
169 // go run -C ./cmd/gen-proof . prefix2 07-tendermint-42 acknowledgement $(echo -n UNIVERSAL_ERROR_ACKNOWLEDGEMENT | sha256sum | cut -d' ' -f1)
170 proofAcked := []ics23.CommitmentProof{
171
172 // iavl proof
173 ics23.CommitmentProof_Exist{
174 Exist: &ics23.ExistenceProof{
175 Key: []byte("\x70\x72\x65\x66\x69\x78\x32\x30\x37\x2d\x74\x65\x6e\x64\x65\x72\x6d\x69\x6e\x74\x2d\x34\x32\x03\x00\x00\x00\x00\x00\x00\x00\x01"),
176 Value: []byte("\xe2\xfb\x30\xdf\xbf\x7a\xbd\xea\xca\x82\xd4\x26\x53\x4d\x2b\x3a\x9d\x54\x44\xdd\x2a\x87\xfa\x16\xd3\x8b\x77\xba\x1a\x13\xce\xd7"),
177 Leaf: &ics23.LeafOp{
178 Hash: specs.LeafSpec.Hash,
179 PrehashKey: specs.LeafSpec.PrehashKey,
180 PrehashValue: specs.LeafSpec.PrehashValue,
181 Length: specs.LeafSpec.Length,
182 Prefix: []byte("\x00\x02\x02"),
183 },
184 Path: []*ics23.InnerOp{
185 {
186 Hash: specs.InnerSpec.Hash,
187 Prefix: []byte("\x02\x04\x02\x20\x35\xf8\xea\x80\x53\x90\xe0\x84\x85\x4f\x39\x9b\x42\xcc\xde\xae\xa3\x3a\x1d\xed\xc1\x15\x63\x8a\xc4\x8d\x06\x00\x63\x7d\xba\x1f\x20"),
188 Suffix: []byte(""),
189 },
190 {
191 Hash: specs.InnerSpec.Hash,
192 Prefix: []byte("\x04\x08\x02\x20"),
193 Suffix: []byte("\x20\x79\x8e\x2c\xaa\x96\xfd\xfb\xa3\x76\xdd\xeb\x47\x99\x99\x54\xd2\xf4\x7e\x65\x16\x22\x64\xb0\x53\x6a\xb5\xdf\xf7\xfc\x0a\x2e\x07"),
194 },
195 {
196 Hash: specs.InnerSpec.Hash,
197 Prefix: []byte("\x06\x0c\x02\x20\x9a\xf3\x7d\xd5\x95\xa0\x19\x08\x03\xb5\xe0\x5a\xae\xf4\x2a\xe3\xfa\xd4\x99\xe4\xfb\xe3\x7f\x7c\xd3\x1c\xad\xff\x22\xa9\xee\x74\x20"),
198 Suffix: []byte(""),
199 },
200 },
201 },
202 },
203
204 // rootmulti proof
205 ics23.CommitmentProof_Exist{
206 Exist: &ics23.ExistenceProof{
207 Key: []byte("\x69\x61\x76\x6c\x53\x74\x6f\x72\x65\x4b\x65\x79"),
208 Value: []byte("\x91\x57\xa5\x00\xd6\xc8\xde\x17\x14\xba\xdf\x44\x6d\x17\xb6\xb8\xd3\xa5\x8b\x30\xf8\x54\x83\x18\x66\xd4\x33\x53\x6f\xd2\xea\xb9"),
209 Leaf: &ics23.LeafOp{
210 Hash: specs.LeafSpec.Hash,
211 PrehashKey: specs.LeafSpec.PrehashKey,
212 PrehashValue: specs.LeafSpec.PrehashValue,
213 Length: specs.LeafSpec.Length,
214 Prefix: []byte("\x00"),
215 },
216 Path: []*ics23.InnerOp{},
217 },
218 },
219 }
220
221 ackPacket := types.MsgAcknowledgement{
222 Packet: types.Packet{
223 Sequence: sequence,
224 SourceClient: clientID,
225 DestinationClient: counterpartyID,
226 TimeoutTimestamp: sendPacket.TimeoutTimestamp,
227 Payloads: sendPacket.Payloads,
228 },
229 Acknowledgement: types.Acknowledgement{
230 // This UNIVERSAL_ERROR_ACKNOWLEDGEMENT error should trigger a refund
231 AppAcknowledgements: [][]byte{types.UniversalErrorAcknowledgement()},
232 },
233 ProofAcked: proofAcked,
234 ProofHeight: msgHeader.GetHeight(),
235 }
236
237 res := core.Acknowledgement(cross(cur), ackPacket)
238
239 println("\nack res:", res)
240 println("\nsigner voucher balance after Acknowledgement:", transfer.VoucherBalanceOf(voucherDenom, signer))
241}
242
243// Output:
244// Payload proto: CgZ1YXRvbmUSAzEwMBoKYXRvbmUxdXNlciIoZzFnanM4bmZkcmQ1NXozeHU5Z2pjYXRuajI5dmt3dXl4NGZmcXhkOQ==
245// signer voucher balance before Transfer: 100
246//
247// signer voucher balance after Transfer: 0
248//
249// ack res: (2 gno.land/p/aib/ibc/types.ResponseResultType)
250//
251// signer voucher balance after Acknowledgement: 100
252
253// Events:
254// [
255// {
256// "type": "create_client",
257// "attrs": [
258// {
259// "key": "client_id",
260// "value": "07-tendermint-1"
261// },
262// {
263// "key": "client_type",
264// "value": "07-tendermint"
265// },
266// {
267// "key": "consensus_heights",
268// "value": "2/2"
269// }
270// ],
271// "pkg_path": "gno.land/r/aib/ibc/core"
272// },
273// {
274// "type": "recv_packet",
275// "attrs": [
276// {
277// "key": "packet_source_client",
278// "value": "07-tendermint-42"
279// },
280// {
281// "key": "packet_dest_client",
282// "value": "07-tendermint-1"
283// },
284// {
285// "key": "packet_sequence",
286// "value": "1"
287// },
288// {
289// "key": "packet_timeout_timestamp",
290// "value": "1234571490"
291// },
292// {
293// "key": "encoded_packet_hex",
294// "value": "0801121030372d74656e6465726d696e742d34321a0f30372d74656e6465726d696e742d3120e2a1d8cc042a7a0a087472616e7366657212087472616e736665721a0769637332302d3122166170706c69636174696f6e2f782d70726f746f6275662a430a067561746f6e6512033130301a0a61746f6e65317573657222286731676a73386e6664726435357a33787539676a6361746e6a3239766b7775797834666671786439"
295// }
296// ],
297// "pkg_path": "gno.land/r/aib/ibc/core"
298// },
299// {
300// "type": "denomination",
301// "attrs": [
302// {
303// "key": "denom_hash",
304// "value": "F9A67CB19B2CAD2ADEC20AD475BE86DF851DEC2B6F6CABC9B7B781BD9131D18F"
305// },
306// {
307// "key": "denom",
308// "value": "{\"base\":\"uatone\",\"path\":\"transfer/07-tendermint-1/uatone\",\"denom\":\"ibc/F9A67CB19B2CAD2ADEC20AD475BE86DF851DEC2B6F6CABC9B7B781BD9131D18F\"}"
309// }
310// ],
311// "pkg_path": "gno.land/r/aib/ibc/apps/transfer"
312// },
313// {
314// "type": "register",
315// "attrs": [
316// {
317// "key": "pkgpath",
318// "value": "gno.land/r/aib/ibc/apps/transfer"
319// },
320// {
321// "key": "slug",
322// "value": "F9A67CB19B2CAD2ADEC20AD475BE86DF851DEC2B6F6CABC9B7B781BD9131D18F"
323// }
324// ],
325// "pkg_path": "gno.land/r/demo/defi/grc20reg"
326// },
327// {
328// "type": "Transfer",
329// "attrs": [
330// {
331// "key": "token",
332// "value": "gno.land/r/aib/ibc/apps/transfer.F9A67CB19B2"
333// },
334// {
335// "key": "from",
336// "value": ""
337// },
338// {
339// "key": "to",
340// "value": "g1gjs8nfdrd55z3xu9gjcatnj29vkwuyx4ffqxd9"
341// },
342// {
343// "key": "value",
344// "value": "100"
345// }
346// ],
347// "pkg_path": "gno.land/p/demo/tokens/grc20"
348// },
349// {
350// "type": "fungible_token_packet",
351// "attrs": [
352// {
353// "key": "sender",
354// "value": "atone1user"
355// },
356// {
357// "key": "receiver",
358// "value": "g1gjs8nfdrd55z3xu9gjcatnj29vkwuyx4ffqxd9"
359// },
360// {
361// "key": "denom",
362// "value": "transfer/07-tendermint-1/uatone"
363// },
364// {
365// "key": "amount",
366// "value": "100"
367// },
368// {
369// "key": "memo",
370// "value": ""
371// },
372// {
373// "key": "success",
374// "value": "true"
375// }
376// ],
377// "pkg_path": "gno.land/r/aib/ibc/apps/transfer"
378// },
379// {
380// "type": "write_acknowledgement",
381// "attrs": [
382// {
383// "key": "packet_source_client",
384// "value": "07-tendermint-42"
385// },
386// {
387// "key": "packet_dest_client",
388// "value": "07-tendermint-1"
389// },
390// {
391// "key": "packet_sequence",
392// "value": "1"
393// },
394// {
395// "key": "packet_timeout_timestamp",
396// "value": "1234571490"
397// },
398// {
399// "key": "encoded_packet_hex",
400// "value": "0801121030372d74656e6465726d696e742d34321a0f30372d74656e6465726d696e742d3120e2a1d8cc042a7a0a087472616e7366657212087472616e736665721a0769637332302d3122166170706c69636174696f6e2f782d70726f746f6275662a430a067561746f6e6512033130301a0a61746f6e65317573657222286731676a73386e6664726435357a33787539676a6361746e6a3239766b7775797834666671786439"
401// },
402// {
403// "key": "encoded_acknowledgement_hex",
404// "value": "0a117b22726573756c74223a2241513d3d227d"
405// }
406// ],
407// "pkg_path": "gno.land/r/aib/ibc/core"
408// },
409// {
410// "type": "send_packet",
411// "attrs": [
412// {
413// "key": "packet_source_client",
414// "value": "07-tendermint-1"
415// },
416// {
417// "key": "packet_dest_client",
418// "value": "07-tendermint-42"
419// },
420// {
421// "key": "packet_sequence",
422// "value": "1"
423// },
424// {
425// "key": "packet_timeout_timestamp",
426// "value": "1234571490"
427// },
428// {
429// "key": "encoded_packet_hex",
430// "value": "0801120f30372d74656e6465726d696e742d311a1030372d74656e6465726d696e742d343220e2a1d8cc042a93010a087472616e7366657212087472616e736665721a0769637332302d3122166170706c69636174696f6e2f782d70726f746f6275662a5c0a1f7472616e736665722f30372d74656e6465726d696e742d312f7561746f6e6512033130301a286731676a73386e6664726435357a33787539676a6361746e6a3239766b7775797834666671786439220a61746f6e653175736572"
431// }
432// ],
433// "pkg_path": "gno.land/r/aib/ibc/core"
434// },
435// {
436// "type": "Transfer",
437// "attrs": [
438// {
439// "key": "token",
440// "value": "gno.land/r/aib/ibc/apps/transfer.F9A67CB19B2"
441// },
442// {
443// "key": "from",
444// "value": "g1gjs8nfdrd55z3xu9gjcatnj29vkwuyx4ffqxd9"
445// },
446// {
447// "key": "to",
448// "value": ""
449// },
450// {
451// "key": "value",
452// "value": "100"
453// }
454// ],
455// "pkg_path": "gno.land/p/demo/tokens/grc20"
456// },
457// {
458// "type": "ibc_transfer",
459// "attrs": [
460// {
461// "key": "sender",
462// "value": "g1gjs8nfdrd55z3xu9gjcatnj29vkwuyx4ffqxd9"
463// },
464// {
465// "key": "receiver",
466// "value": "atone1user"
467// },
468// {
469// "key": "denom",
470// "value": "transfer/07-tendermint-1/uatone"
471// },
472// {
473// "key": "amount",
474// "value": "100"
475// },
476// {
477// "key": "memo",
478// "value": ""
479// }
480// ],
481// "pkg_path": "gno.land/r/aib/ibc/apps/transfer"
482// },
483// {
484// "type": "update_client",
485// "attrs": [
486// {
487// "key": "client_id",
488// "value": "07-tendermint-1"
489// },
490// {
491// "key": "client_type",
492// "value": "07-tendermint"
493// },
494// {
495// "key": "consensus_heights",
496// "value": "2/12"
497// }
498// ],
499// "pkg_path": "gno.land/r/aib/ibc/core"
500// },
501// {
502// "type": "acknowledge_packet",
503// "attrs": [
504// {
505// "key": "packet_source_client",
506// "value": "07-tendermint-1"
507// },
508// {
509// "key": "packet_dest_client",
510// "value": "07-tendermint-42"
511// },
512// {
513// "key": "packet_sequence",
514// "value": "1"
515// },
516// {
517// "key": "packet_timeout_timestamp",
518// "value": "1234571490"
519// },
520// {
521// "key": "encoded_packet_hex",
522// "value": "0801120f30372d74656e6465726d696e742d311a1030372d74656e6465726d696e742d343220e2a1d8cc042a93010a087472616e7366657212087472616e736665721a0769637332302d3122166170706c69636174696f6e2f782d70726f746f6275662a5c0a1f7472616e736665722f30372d74656e6465726d696e742d312f7561746f6e6512033130301a286731676a73386e6664726435357a33787539676a6361746e6a3239766b7775797834666671786439220a61746f6e653175736572"
523// }
524// ],
525// "pkg_path": "gno.land/r/aib/ibc/core"
526// },
527// {
528// "type": "Transfer",
529// "attrs": [
530// {
531// "key": "token",
532// "value": "gno.land/r/aib/ibc/apps/transfer.F9A67CB19B2"
533// },
534// {
535// "key": "from",
536// "value": ""
537// },
538// {
539// "key": "to",
540// "value": "g1gjs8nfdrd55z3xu9gjcatnj29vkwuyx4ffqxd9"
541// },
542// {
543// "key": "value",
544// "value": "100"
545// }
546// ],
547// "pkg_path": "gno.land/p/demo/tokens/grc20"
548// },
549// {
550// "type": "fungible_token_packet",
551// "attrs": [
552// {
553// "key": "sender",
554// "value": "g1gjs8nfdrd55z3xu9gjcatnj29vkwuyx4ffqxd9"
555// },
556// {
557// "key": "receiver",
558// "value": "atone1user"
559// },
560// {
561// "key": "denom",
562// "value": "transfer/07-tendermint-1/uatone"
563// },
564// {
565// "key": "amount",
566// "value": "100"
567// },
568// {
569// "key": "memo",
570// "value": ""
571// },
572// {
573// "key": "acknowledgement",
574// "value": "Gtԥu\ufffd?\ufffd;\u001c\u0006W76azEz\ufffd\ufffdX\ufffdxۍ\u0010\ufffdKJ\ufffd\u0011\ufffd"
575// }
576// ],
577// "pkg_path": "gno.land/r/aib/ibc/apps/transfer"
578// },
579// {
580// "type": "fungible_token_packet",
581// "attrs": [
582// {
583// "key": "error",
584// "value": "receive packet failed"
585// }
586// ],
587// "pkg_path": "gno.land/r/aib/ibc/apps/transfer"
588// }
589// ]