Compare commits
902 commits
Author | SHA1 | Date | |
---|---|---|---|
|
b66cea5ccc | ||
|
e5eb20bbae | ||
|
cf4d4a2611 | ||
|
7a7a5d0322 | ||
|
4d6333866d | ||
|
4a5e8f8592 | ||
|
b4fdf8c3c9 | ||
|
3c471f0b7e | ||
|
41109f442a | ||
|
5c408f7fe4 | ||
|
cd525a1df5 | ||
|
836183ecb6 | ||
|
3c1c650ddf | ||
|
45b8af359d | ||
|
4945145b09 | ||
|
f48e992a78 | ||
|
0b136c1c9c | ||
|
f4aeaa6387 | ||
|
0ae5e7ea83 | ||
|
228052360e | ||
|
5cbfe215a4 | ||
|
72161fe902 | ||
|
aceaabe9d6 | ||
|
9ba532ce9c | ||
|
2aa3c93acd | ||
|
aa5a0cb49b | ||
|
7868578571 | ||
|
6a5833759c | ||
|
6be757af3e | ||
|
67b6d5da31 | ||
|
b5bb605651 | ||
|
d91a0945df | ||
|
c90f6785b9 | ||
|
c30e7ec8d7 | ||
|
ce442a1942 | ||
|
e92606a7ae | ||
|
df1ed68d98 | ||
|
82a7c1bd9c | ||
|
c43cfae24c | ||
|
2d4993a837 | ||
|
647b8c7fb1 | ||
|
0b69901d1e | ||
|
502f024e7c | ||
|
b1bb12df48 | ||
|
31a99d44eb | ||
|
dc8b2f639a | ||
|
2ad4c86712 | ||
|
9a1075d332 | ||
|
58a086ea91 | ||
|
59c98c4d09 | ||
|
f4731eab91 | ||
|
13020ccd02 | ||
|
bce94dfa66 | ||
|
244ca6f438 | ||
|
a32217fac8 | ||
|
0435551eee | ||
|
0a3bf01a8f | ||
|
0aecddea10 | ||
|
02ecbeb519 | ||
|
e275495850 | ||
|
6e0926e59f | ||
|
988440949b | ||
|
701ea8d5f3 | ||
|
3d1e33a502 | ||
|
5053a073f9 | ||
|
71aa32406d | ||
|
3acb132e9a | ||
|
bd2f9c75e1 | ||
|
34eef47a18 | ||
|
7995229f6b | ||
|
b21db99747 | ||
|
bc158eed51 | ||
|
ac70f069ff | ||
|
b54fcbcdd9 | ||
|
8da7c0a671 | ||
|
956fd08adb | ||
|
1292a00ef9 | ||
|
7c8d2c3ec5 | ||
|
df2a56908b | ||
|
198af7f3ae | ||
|
874ed1ac2e | ||
|
4a8a43b983 | ||
|
6cf785ce54 | ||
|
9b8d579bce | ||
|
36e128516b | ||
|
4ca2686583 | ||
|
235f4398c6 | ||
|
ec6fc54bc6 | ||
|
2d3d52e3d0 | ||
|
82993dab2b | ||
|
5c75ee13d0 | ||
|
3a2e301267 | ||
|
d74dc368e0 | ||
|
73c742a466 | ||
|
d62fad1268 | ||
|
a13dc59bd9 | ||
|
678c1982f9 | ||
|
c84adf856a | ||
|
7345dbcab4 | ||
|
fbe514ca91 | ||
|
83fdcc8568 | ||
|
4b9024fa45 | ||
|
9b688a9ee9 | ||
|
1786136a23 | ||
|
d3993ad54c | ||
|
db37101e34 | ||
|
567f9f9c6e | ||
|
fb4a4f074f | ||
|
97a19df6f0 | ||
|
c5dbecb754 | ||
|
fc79d38ad2 | ||
|
ae3515e819 | ||
|
8913542f93 | ||
|
69bcd4b7d9 | ||
|
51537dfd29 | ||
|
ee1ee56f97 | ||
|
8c1d061446 | ||
|
a28e5b24b2 | ||
|
7aa5983543 | ||
|
fcbfc3b807 | ||
|
cf49e8c4c8 | ||
|
78cefca5c9 | ||
|
f2bb4d455b | ||
|
f553f77d20 | ||
|
700a550973 | ||
|
f84f072a29 | ||
|
927dbb6dc4 | ||
|
e5c919f701 | ||
|
22af33a14e | ||
|
ef3ec190d0 | ||
|
b1a986fba8 | ||
|
f6901806a8 | ||
|
686c77aeea | ||
|
89ef26164c | ||
|
4c288720cd | ||
|
f8c2d269fe | ||
|
698bdc7eea | ||
|
0280ceefe6 | ||
|
b6a9c64c55 | ||
|
be1b97d04e | ||
|
29deba45ca | ||
|
c81ed22698 | ||
|
c5ba53986f | ||
|
fff25bd85c | ||
|
182762af5a | ||
|
e7a0ded65e | ||
|
7202518fcf | ||
|
e0e7fdf810 | ||
|
f95004ad83 | ||
|
309016561d | ||
|
65dbd537ec | ||
|
b947d09552 | ||
|
5c995e71b5 | ||
|
5b30d15f8e | ||
|
57d82c1281 | ||
|
e341e64684 | ||
|
cd215fe096 | ||
|
d7ab7634ef | ||
|
00ac38b545 | ||
|
56e6119f78 | ||
|
d3c8916827 | ||
|
86f16bb931 | ||
|
e27a822d4b | ||
|
687a724d2c | ||
|
8d0d5cb7cd | ||
|
627a0abb57 | ||
|
beee2605a3 | ||
|
708b439f4a | ||
|
3e6dfff503 | ||
|
f8ca51db93 | ||
|
85a2a9a989 | ||
|
37d7a3a2d5 | ||
|
c090628ea5 | ||
|
6d1a22dcb3 | ||
|
1840c057bd | ||
|
fa1c07e7e6 | ||
|
d5a5e4c125 | ||
|
f8ebed2da4 | ||
|
72484a33eb | ||
|
e0328cc1f8 | ||
|
900ef462df | ||
|
d80e14dcbb | ||
|
f409fc36e2 | ||
|
b028c772f2 | ||
|
308475f746 | ||
|
31548dc28f | ||
|
7c2575568d | ||
|
270515de6a | ||
|
a81dc5c795 | ||
|
da4e80e7a0 | ||
|
1a3a494459 | ||
|
7e12cc11e4 | ||
|
27f79b8c63 | ||
|
445ab97906 | ||
|
82e52c4f6d | ||
|
335550fb1e | ||
|
a6f52a7180 | ||
|
bfc3aa6b67 | ||
|
0e6fbadfd5 | ||
|
9c83beffbb | ||
|
37d4012306 | ||
|
67d39177d8 | ||
|
1679b162b8 | ||
|
45c6a7dcd7 | ||
|
077b4f9097 | ||
|
b015c7b05d | ||
|
57d828c4ab | ||
|
b78b715259 | ||
|
86faf7b343 | ||
|
319880e201 | ||
|
0016b6b630 | ||
|
b12ef701f0 | ||
|
588d7c6418 | ||
|
0e47d662b5 | ||
|
54c53cf6ca | ||
|
025bf228a2 | ||
|
bdcc036073 | ||
|
d31b01d4b1 | ||
|
ee39b5ca18 | ||
|
46f2da0fb1 | ||
|
b7efccc040 | ||
|
a256eac619 | ||
|
261b389e24 | ||
|
9e99e35a63 | ||
|
b2936e602c | ||
|
d65b1fd66d | ||
|
72ebcbec72 | ||
|
d2a7162217 | ||
|
49edf9e498 | ||
|
8162e9033d | ||
|
b030c331b0 | ||
|
1560df6913 | ||
|
0c6627f13d | ||
|
13ff95a3d3 | ||
|
5485cf60da | ||
|
3f772c2711 | ||
|
c660a6e8f3 | ||
|
90db72abc9 | ||
|
8d67f17943 | ||
|
46a3ff3348 | ||
|
9525bc7785 | ||
|
685d095fa4 | ||
|
1f4e1922ee | ||
|
2404145133 | ||
|
e1e2b5304f | ||
|
9d7357c9fd | ||
|
132531fabe | ||
|
6a560b9a95 | ||
|
630bd29b8d | ||
|
5fcae176b8 | ||
|
cc38221d77 | ||
|
5cf0c75744 | ||
|
5bb7c6b715 | ||
|
9b540770cd | ||
|
775c56e87e | ||
|
4715e523e0 | ||
|
f8dc5ec44f | ||
|
224808975b | ||
|
8ab08c6221 | ||
|
e55809f198 | ||
|
a81804e296 | ||
|
9d3258495f | ||
|
e09fc00188 | ||
|
be17924ef8 | ||
|
6147bf452a | ||
|
327e766cd9 | ||
|
b6a65e4d80 | ||
|
00a1a8da56 | ||
|
07b2de73cd | ||
|
f94df6c013 | ||
|
5128b84e1d | ||
|
5d425a6e45 | ||
|
71fb759c0d | ||
|
588e911ad6 | ||
|
5fe4778b2e | ||
|
939631c3f9 | ||
|
51ac153a7b | ||
|
9d95a2691b | ||
|
57aa05add3 | ||
|
000109bfd1 | ||
|
36b89215ab | ||
|
a19f85ba34 | ||
|
161b83fd7f | ||
|
9b7eff18c0 | ||
|
a22a7177e3 | ||
|
b90c0ece87 | ||
|
ef99a7a9e3 | ||
|
5f745dff72 | ||
|
ed73628100 | ||
|
d20775dd30 | ||
|
d428cfc4f1 | ||
|
32fab1adf1 | ||
|
2c0c111fd5 | ||
|
ef9f75547a | ||
|
f9a82ae45f | ||
|
db2e214f7e | ||
|
baac738ec3 | ||
|
188a5e6a8f | ||
|
d901697615 | ||
|
2da728d176 | ||
|
012da4501d | ||
|
d8320372b8 | ||
|
f5b1bd3978 | ||
|
7c9b1d05d2 | ||
|
1ca7b0bbeb | ||
|
0ffa24932b | ||
|
0d3f749b5e | ||
|
78050e8b2c | ||
|
4b7b71ce25 | ||
|
402a73b7f3 | ||
|
ba1417397f | ||
|
4f5e3f363a | ||
|
4c6dca876c | ||
|
28f1beff64 | ||
|
3176f72878 | ||
|
cc06674253 | ||
|
275e814271 | ||
|
3283fda790 | ||
|
6c88fccc64 | ||
|
1237c719e3 | ||
|
9015215228 | ||
|
5d514538cf | ||
|
56d32a010e | ||
|
a872c200b0 | ||
|
47aefad7ea | ||
|
d7d850ac7d | ||
|
b62541b6a3 | ||
|
c8ff44560e | ||
|
962b161652 | ||
|
de98b39a95 | ||
|
2df0ba7c10 | ||
|
bcee8c268c | ||
|
0baa35ed77 | ||
|
ad122aec41 | ||
|
ead5b3dc09 | ||
|
3766206f44 | ||
|
2012c56031 | ||
|
a5a87a72c2 | ||
|
d5dee8fc94 | ||
|
6f3a219d67 | ||
|
bbbb12af20 | ||
|
3fd48a743e | ||
|
afca64f164 | ||
|
21ccb1a02a | ||
|
44e21c9a8e | ||
|
8bdbac3c5e | ||
|
d7cab3b82c | ||
|
f93e2fbba4 | ||
|
441eb8aa86 | ||
|
af0d1fe3c9 | ||
|
cab358dba1 | ||
|
0639cedd05 | ||
|
9a270ae30c | ||
|
7b53a0c239 | ||
|
6c0c2a6a98 | ||
|
26f96fe7ee | ||
|
daa2b3b920 | ||
|
b31d39836a | ||
|
34be76efb7 | ||
|
90e18ec743 | ||
|
e3aa6bba4d | ||
|
81a439a0d6 | ||
|
6fdda0f15d | ||
|
8e8936912d | ||
|
76261f1605 | ||
|
42b84ad905 | ||
|
98de5b9ccb | ||
|
b637048122 | ||
|
e8e964e3da | ||
|
d7b84c0b47 | ||
|
29f344fd9b | ||
|
2f9b4a2dbf | ||
|
2eab743a5b | ||
|
a159054f10 | ||
|
b35f351f0b | ||
|
fd7b7ffc13 | ||
|
a9803b3477 | ||
|
7179efec35 | ||
|
fc77754098 | ||
|
7a2eb32c42 | ||
|
20a57d8337 | ||
|
7a1bf77585 | ||
|
15b4e0a8cd | ||
|
11ec0db350 | ||
|
4216efbe6c | ||
|
626f1ee198 | ||
|
5d1e0192f1 | ||
|
1f84756c21 | ||
|
61d5b6eb0c | ||
|
f457d50773 | ||
|
d511f6e5a9 | ||
|
802d8d24b9 | ||
|
195bb742ca | ||
|
251b8c57b9 | ||
|
62b710868b | ||
|
7362fd94e6 | ||
|
341d978e5b | ||
|
22c654b200 | ||
|
2f6ba1fded | ||
|
25ef2c7f16 | ||
|
910d53b27b | ||
|
1d189fd90c | ||
|
6824bd9f40 | ||
|
387c411da0 | ||
|
00c6a811ab | ||
|
e21664cd7d | ||
|
23e553f647 | ||
|
4c9cd438f8 | ||
|
eade327b9b | ||
|
8aeb30e019 | ||
|
fced6a27ba | ||
|
5370034955 | ||
|
a5d041a1ac | ||
|
7f9e2e5047 | ||
|
2ae44e074d | ||
|
ff958706f4 | ||
|
28ee1621b8 | ||
|
58102a9a80 | ||
|
4d8601297d | ||
|
64b5ba24e9 | ||
|
3b11f98cd0 | ||
|
c6d259b285 | ||
|
0493ddbd70 | ||
|
474225d425 | ||
|
82cb2e718d | ||
|
80fcf81102 | ||
|
8e5a724f5d | ||
|
59baecd39b | ||
|
f10d2d2be7 | ||
|
63de821ff4 | ||
|
a4779de375 | ||
|
0fef119f5f | ||
|
16b7a83f30 | ||
|
406c9f8b92 | ||
|
b5cf3f592f | ||
|
f6cb698cd1 | ||
|
7fb077e999 | ||
|
48bd8057dd | ||
|
90705e37e9 | ||
|
06c99fb2af | ||
|
e63a93b8a0 | ||
|
2fba04490a | ||
|
90b01bcea6 | ||
|
14d98811a5 | ||
|
b2205940fa | ||
|
fdcc369d60 | ||
|
dd48f810cc | ||
|
f88df92925 | ||
|
de2a445088 | ||
|
5e220754d7 | ||
|
b0cdae4666 | ||
|
963a22e280 | ||
|
dbd647e3a6 | ||
|
ed4610f8b4 | ||
|
7045d3bd63 | ||
|
f8d8b03f64 | ||
|
fca7eb35a1 | ||
|
b4866bd69e | ||
|
c5e284c117 | ||
|
ba4c58780c | ||
|
de38163f89 | ||
|
bbbc6805a8 | ||
|
91c8aa21cc | ||
|
8ed6d97085 | ||
|
c9803f5448 | ||
|
9de9320d55 | ||
|
b9297a4b2c | ||
|
a755a2be30 | ||
|
0e458f473f | ||
|
d89c8801d6 | ||
|
dffd87c6c8 | ||
|
1605496002 | ||
|
7a6b908cae | ||
|
c3955f87d1 | ||
|
3544d65ecd | ||
|
6e0af5ac90 | ||
|
39ce3a9d1d | ||
|
c93658b1c1 | ||
|
890f64007a | ||
|
8cc32a91b6 | ||
|
065bd3f0be | ||
|
1815bc8a32 | ||
|
61a74ab331 | ||
|
04d778612e | ||
|
cb0ef2de3d | ||
|
3353599b38 | ||
|
6476241b3a | ||
|
1841d818ec | ||
|
ef2bc7ca04 | ||
|
6a706d5c5a | ||
|
c42486587d | ||
|
d24579748e | ||
|
19cfbc8d85 | ||
|
b1ad19864f | ||
|
0bd618bee4 | ||
|
38f77c39d0 | ||
|
1aabdc9672 | ||
|
545dbbc668 | ||
|
ff19189937 | ||
|
a9a848a306 | ||
|
69a0104d05 | ||
|
f31e3f70aa | ||
|
c8ed629b5d | ||
|
645076d04f | ||
|
3c13a3d74d | ||
|
e06051e14c | ||
|
3aa858a69f | ||
|
241d5f593e | ||
|
dc3d1300dd | ||
|
7ca4ce0f79 | ||
|
9cd5c88077 | ||
|
45dcf38563 | ||
|
995d774ff8 | ||
|
e6b4fbc999 | ||
|
6e9ac29d5c | ||
|
b284b902d6 | ||
|
32bfed4741 | ||
|
fce98a0af8 | ||
|
2f747dd8d9 | ||
|
8fcf1eecfa | ||
|
bb9b8368a8 | ||
|
460362ab2e | ||
|
9f7b19f886 | ||
|
6865848ca7 | ||
|
de69560c7d | ||
|
a59afa8ea3 | ||
|
eb995c8600 | ||
|
413caac941 | ||
|
d9644204ee | ||
|
f78f915071 | ||
|
27dddb4d50 | ||
|
794658b54c | ||
|
312534af7c | ||
|
07e1bc7cd7 | ||
|
96449d803a | ||
|
19c59c3a59 | ||
|
eeb439f548 | ||
|
cef70091f6 | ||
|
f1c6b9fe8f | ||
|
b78bea17c0 | ||
|
32eb1e2606 | ||
|
96ee2e2b2d | ||
|
d27d2a8561 | ||
|
91c928e8d3 | ||
|
fc6e87af2f | ||
|
a2d28272ef | ||
|
b5a75b1485 | ||
|
1745b21142 | ||
|
e2abb5cb7c | ||
|
ca71bd51d3 | ||
|
b7e019e7ef | ||
|
8c46517522 | ||
|
ff260a6a9b | ||
|
03951c94b0 | ||
|
9e74fc5b47 | ||
|
0a3260c22c | ||
|
4598f3d3c2 | ||
|
49a44b0b9d | ||
|
fd2774ea91 | ||
|
3628b824e1 | ||
|
2581146a01 | ||
|
f66abd020d | ||
|
fda0e8364f | ||
|
eb571a1356 | ||
|
a0f9743515 | ||
|
0f57ad4a12 | ||
|
2d88ed9fe5 | ||
|
31cc349725 | ||
|
eb9735750c | ||
|
26c97a58ed | ||
|
ee12520cbd | ||
|
a703e54563 | ||
|
9bdf66a5e0 | ||
|
98b946dcd3 | ||
|
c6850b49d9 | ||
|
4c015b30d5 | ||
|
2f54f176fb | ||
|
7cedcb9197 | ||
|
e9d582f525 | ||
|
fff7e91709 | ||
|
f3c1283ac6 | ||
|
a59fd50844 | ||
|
259cbc3356 | ||
|
5180305240 | ||
|
cb1432233b | ||
|
59f72429b4 | ||
|
acd821948f | ||
|
0d30c834d5 | ||
|
7afa950eb7 | ||
|
d06f135792 | ||
|
9e31e42bd9 | ||
|
d285342d54 | ||
|
1e6ef0c0e6 | ||
|
edf21e0b79 | ||
|
e83916cef1 | ||
|
abb35ad6fd | ||
|
f330aaf313 | ||
|
c80702cc4a | ||
|
5463a91edc | ||
|
4a648692a2 | ||
|
ad24aad9c0 | ||
|
c63289a564 | ||
|
2d4e1b598f | ||
|
0b67fa9bca | ||
|
1863b79c5c | ||
|
b89078c42a | ||
|
f5b0489d74 | ||
|
dd7c762ff9 | ||
|
aeb7ee1021 | ||
|
97a57de82d | ||
|
b4dff7b040 | ||
|
5d3938ae23 | ||
|
7d75526c20 | ||
|
7b64b693bd | ||
|
c7d5ee0898 | ||
|
1fb0c96e2c | ||
|
97d523ceed | ||
|
617c628c24 | ||
|
87e4b6beaa | ||
|
b9e957a05a | ||
|
124c3df2ff | ||
|
227b0c5480 | ||
|
66a80017d3 | ||
|
c5d9bf6e39 | ||
|
bb2a99d451 | ||
|
ed2c4b0319 | ||
|
2872c1c668 | ||
|
35c3b65c8a | ||
|
380de580a7 | ||
|
1b1b454ce4 | ||
|
415d44792a | ||
|
87aaaf1a67 | ||
|
60795a899f | ||
|
5c6a111d00 | ||
|
90f1b0fd17 | ||
|
f3760c1a98 | ||
|
f557959c24 | ||
|
2598257628 | ||
|
1e6372e6d9 | ||
|
d3fe92de14 | ||
|
3b29720298 | ||
|
3178c8ff78 | ||
|
31ceb568cb | ||
|
3abddc78c0 | ||
|
9fb154a376 | ||
|
d1fe64470d | ||
|
4e61e8238a | ||
|
16d1d1e5eb | ||
|
b4c0fcfaad | ||
|
b2f84c83b3 | ||
|
1356721862 | ||
|
11c0f13d06 | ||
|
624f193f94 | ||
|
d90608ddbf | ||
|
562293c74b | ||
|
3608314c15 | ||
|
158d8e69a4 | ||
|
5b12be2ac7 | ||
|
de43b39c2a | ||
|
11bb733f1a | ||
|
e7f77e052f | ||
|
2493400525 | ||
|
c39153756a | ||
|
25f61b45dd | ||
|
50ee241377 | ||
|
bd937bc500 | ||
|
6c1240d023 | ||
|
5fc61be5f6 | ||
|
4e6b1c4a38 | ||
|
b5f17215db | ||
|
afa4530c7d | ||
|
e82fcb48c4 | ||
|
50a77f3fae | ||
|
7e867aefc4 | ||
|
18c42f7993 | ||
|
80fdb1c90c | ||
|
e2c6bbb6b1 | ||
|
8d5a41de6e | ||
|
6615cce81d | ||
|
c0abc61613 | ||
|
bf871760c6 | ||
|
96aa10bb80 | ||
|
8db997c58a | ||
|
e30e262e66 | ||
|
ebe4c4ce2b | ||
|
b560475e67 | ||
|
4383832aa2 | ||
|
6abf5f6388 | ||
|
d560395985 | ||
|
2c984e2393 | ||
|
d0718a680f | ||
|
97b93c6833 | ||
|
288dee8871 | ||
|
08b273266b | ||
|
ab64f7cfe4 | ||
|
6e990f39de | ||
|
465d3f43d2 | ||
|
f8227aa5f7 | ||
|
758c455c7e | ||
|
c880873105 | ||
|
d5f964f181 | ||
|
6cd0f78649 | ||
|
56b93d279e | ||
|
7028930bd6 | ||
|
eeab9c1b2e | ||
|
01a5816f04 | ||
|
03b9b4a4a1 | ||
|
4be692193e | ||
|
3bbb21bf8f | ||
|
e57967b11c | ||
|
081f9d3ac5 | ||
|
ee4b8f883b | ||
|
7a7d0a06ce | ||
|
0d17273476 | ||
|
0a2be89964 | ||
|
c926d53869 | ||
|
862c2e4ed3 | ||
|
35ebdc1c91 | ||
|
966111f4a8 | ||
|
8253025c90 | ||
|
165525b7e9 | ||
|
9f69522ff5 | ||
|
e7a83cbef9 | ||
|
8fcfa2c11a | ||
|
ee885bafdf | ||
|
146775213a | ||
|
762a8da76a | ||
|
f611a793a7 | ||
|
68ad542639 | ||
|
48c8ae359f | ||
|
ca80cfea2b | ||
|
6fa4bcdc1d | ||
|
bfe8867e63 | ||
|
abb58fb7de | ||
|
91b57745ac | ||
|
e993003b46 | ||
|
3bf8192f98 | ||
|
60e6fe05a7 | ||
|
70aed34d77 | ||
|
50c8805034 | ||
|
84b00d46aa | ||
|
a30f098f73 | ||
|
20b19d4599 | ||
|
9185820289 | ||
|
31e2076810 | ||
|
ea13fbe94a | ||
|
71bcb8bade | ||
|
a2daad6ba6 | ||
|
0d470edf21 | ||
|
4b2fc32462 | ||
|
b86975b061 | ||
|
802a2b3879 | ||
|
f96ccf2bfe | ||
|
c7836ed6e7 | ||
|
805157b7a0 | ||
|
67ce9de181 | ||
|
a0117042e8 | ||
|
f97eaddfd1 | ||
|
50fad8455f | ||
|
68b9ff1f17 | ||
|
772e723e8e | ||
|
9a15d2d781 | ||
|
8beb3f4515 | ||
|
9f9cec53bd | ||
|
8ae4a1e957 | ||
|
2ce1454ef5 | ||
|
44dfe8342d | ||
|
865bd6c9cc | ||
|
6379bcc15a | ||
|
0195aae824 | ||
|
3e2755e66d | ||
|
a0d991a500 | ||
|
37af2031bb | ||
|
19cc6c6369 | ||
|
41938ffa78 | ||
|
2c36802e23 | ||
|
e2580187a1 | ||
|
194639bb15 | ||
|
044ae477ca | ||
|
ae52b2c2fa | ||
|
871ee18981 | ||
|
db812f7fa5 | ||
|
0f0f7b364f | ||
|
8beb9f23c3 | ||
|
36af361c2b | ||
|
b5ec1271d5 | ||
|
7a06cea885 | ||
|
aca12b58c0 | ||
|
ddfbf7a434 | ||
|
2f4fb3a079 | ||
|
b6b80f3abf | ||
|
35c3518b37 | ||
|
7b86a54fc0 | ||
|
fc6029d006 | ||
|
d463bad96f | ||
|
aace790651 | ||
|
ed2b59e077 | ||
|
a2449ae8ae | ||
|
3dbb733324 | ||
|
15138b2004 | ||
|
8e085d3ca3 | ||
|
d493afc941 | ||
|
8599e25fc3 | ||
|
4afc71ab6d | ||
|
4be9e4347a | ||
|
abf3ef5af7 | ||
|
649b9ac7b0 | ||
|
230f54c38a | ||
|
778ddfb277 | ||
|
b47a891b9e | ||
|
433275265f | ||
|
8e6025fbc8 | ||
|
fcaa24f928 | ||
|
8bd9a7d420 | ||
|
cef7ac2510 | ||
|
a0aef0e92f | ||
|
98d29570d3 | ||
|
9c0a45c65e | ||
|
0a160ee93b | ||
|
2567c4c672 | ||
|
dcea3f6107 | ||
|
1bd22adcb3 | ||
|
67d4d891ef | ||
|
edb2d46d5b | ||
|
33c971b0e4 | ||
|
29b3df10b6 | ||
|
e2782aef05 | ||
|
bbdfdc3099 | ||
|
b1f9ba193b | ||
|
3f2e0e5441 | ||
|
6e8328415c | ||
|
dab13a4e2d | ||
|
4a49bf5de4 | ||
|
f0a5430b5b | ||
|
d90626d556 | ||
|
6bcdf32c31 | ||
|
a8f05d9833 | ||
|
dd8218f87a | ||
|
c50ab95164 | ||
|
fb7fce0775 | ||
|
7b109586ca | ||
|
77836ea1d0 | ||
|
6eaa76520f | ||
|
45b353781f | ||
|
7bc9b6e323 | ||
|
ef925b2c0b | ||
|
a4cc6da766 | ||
|
e29c33e449 | ||
|
e126bcc462 | ||
|
a113940f0b | ||
|
649877d8f3 | ||
|
e0abe2b858 | ||
|
b9f95a820c | ||
|
3a71aafc43 | ||
|
7bcc62d99c | ||
|
49cea72e41 | ||
|
a875409055 | ||
|
a74454aaca | ||
|
0c049f620f | ||
|
8149d33fef | ||
|
8f9d101eab | ||
|
55ab38ed81 | ||
|
37f7c9d9a7 | ||
|
541da7e83b | ||
|
fdec8ea3f7 | ||
|
1962dd956c | ||
|
d5bea0ad4c | ||
|
4f827de5cd | ||
|
400620a9fb | ||
|
cb0f786b28 | ||
|
e3747b1d57 | ||
|
41caeed5c0 | ||
|
5322a3535b | ||
|
601c26bbe6 | ||
|
f479b2bc57 | ||
|
ce7658ef82 | ||
|
a80d7bef80 | ||
|
e2cf5b868a | ||
|
7e92b4a694 | ||
|
01ac2d9f31 | ||
|
66b82d0a7d | ||
|
68cb07999b | ||
|
69102a6aa3 | ||
|
115ec4d8dd | ||
|
c053f1a4af | ||
|
4ab6d1e31a | ||
|
6b21ad9922 | ||
|
83545b8451 | ||
|
ec002f31df | ||
|
3f6a88c239 | ||
|
110356857d | ||
|
856edcca39 | ||
|
1633ae32df | ||
|
d36df15878 | ||
|
9224878196 | ||
|
f21e618be5 | ||
|
de7bff9cea | ||
|
f672a97d1b | ||
|
6e27883c10 | ||
|
4671fbb3be | ||
|
7e709313d3 | ||
|
2093bf806f |
615 changed files with 30744 additions and 11251 deletions
|
@ -16,7 +16,7 @@
|
||||||
"deployed": false
|
"deployed": false
|
||||||
},
|
},
|
||||||
"lock": false,
|
"lock": false,
|
||||||
"isdefault": false
|
"isDefault": false
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"address": "NVTiAjNgagDkTr5HTzDmQP9kPwPHN5BgVq",
|
"address": "NVTiAjNgagDkTr5HTzDmQP9kPwPHN5BgVq",
|
||||||
|
@ -41,7 +41,7 @@
|
||||||
"deployed": false
|
"deployed": false
|
||||||
},
|
},
|
||||||
"lock": false,
|
"lock": false,
|
||||||
"isdefault": false
|
"isDefault": false
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"scrypt": {
|
"scrypt": {
|
||||||
|
|
|
@ -16,7 +16,7 @@
|
||||||
"deployed": false
|
"deployed": false
|
||||||
},
|
},
|
||||||
"lock": false,
|
"lock": false,
|
||||||
"isdefault": false
|
"isDefault": false
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"address": "NVTiAjNgagDkTr5HTzDmQP9kPwPHN5BgVq",
|
"address": "NVTiAjNgagDkTr5HTzDmQP9kPwPHN5BgVq",
|
||||||
|
@ -41,7 +41,7 @@
|
||||||
"deployed": false
|
"deployed": false
|
||||||
},
|
},
|
||||||
"lock": false,
|
"lock": false,
|
||||||
"isdefault": false
|
"isDefault": false
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"address": "NfgHwwTi3wHAS8aFAN243C5vGbkYDpqLHP",
|
"address": "NfgHwwTi3wHAS8aFAN243C5vGbkYDpqLHP",
|
||||||
|
@ -58,7 +58,7 @@
|
||||||
"deployed": false
|
"deployed": false
|
||||||
},
|
},
|
||||||
"lock": false,
|
"lock": false,
|
||||||
"isdefault": false
|
"isDefault": false
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"scrypt": {
|
"scrypt": {
|
||||||
|
|
|
@ -16,7 +16,7 @@
|
||||||
"deployed": false
|
"deployed": false
|
||||||
},
|
},
|
||||||
"lock": false,
|
"lock": false,
|
||||||
"isdefault": false
|
"isDefault": false
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"address": "NVTiAjNgagDkTr5HTzDmQP9kPwPHN5BgVq",
|
"address": "NVTiAjNgagDkTr5HTzDmQP9kPwPHN5BgVq",
|
||||||
|
@ -41,7 +41,7 @@
|
||||||
"deployed": false
|
"deployed": false
|
||||||
},
|
},
|
||||||
"lock": false,
|
"lock": false,
|
||||||
"isdefault": false
|
"isDefault": false
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"scrypt": {
|
"scrypt": {
|
||||||
|
|
|
@ -16,7 +16,7 @@
|
||||||
"deployed": false
|
"deployed": false
|
||||||
},
|
},
|
||||||
"lock": false,
|
"lock": false,
|
||||||
"isdefault": false
|
"isDefault": false
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"address": "NVTiAjNgagDkTr5HTzDmQP9kPwPHN5BgVq",
|
"address": "NVTiAjNgagDkTr5HTzDmQP9kPwPHN5BgVq",
|
||||||
|
@ -41,7 +41,7 @@
|
||||||
"deployed": false
|
"deployed": false
|
||||||
},
|
},
|
||||||
"lock": false,
|
"lock": false,
|
||||||
"isdefault": false
|
"isDefault": false
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"scrypt": {
|
"scrypt": {
|
||||||
|
|
|
@ -16,7 +16,7 @@
|
||||||
"deployed": false
|
"deployed": false
|
||||||
},
|
},
|
||||||
"lock": false,
|
"lock": false,
|
||||||
"isdefault": false
|
"isDefault": false
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"address": "NVTiAjNgagDkTr5HTzDmQP9kPwPHN5BgVq",
|
"address": "NVTiAjNgagDkTr5HTzDmQP9kPwPHN5BgVq",
|
||||||
|
@ -41,7 +41,7 @@
|
||||||
"deployed": false
|
"deployed": false
|
||||||
},
|
},
|
||||||
"lock": false,
|
"lock": false,
|
||||||
"isdefault": false
|
"isDefault": false
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"scrypt": {
|
"scrypt": {
|
||||||
|
|
2
.github/CODEOWNERS
vendored
2
.github/CODEOWNERS
vendored
|
@ -1 +1 @@
|
||||||
* @AnnaShaleva @roman-khimov @fyrchik
|
* @AnnaShaleva @roman-khimov
|
||||||
|
|
43
.github/workflows/build.yml
vendored
43
.github/workflows/build.yml
vendored
|
@ -37,29 +37,23 @@ jobs:
|
||||||
runs-on: ${{matrix.os.name}}
|
runs-on: ${{matrix.os.name}}
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
os: [{ name: ubuntu-20.04, bin-name: linux }, { name: windows-2022, bin-name: windows }, { name: macos-12, bin-name: darwin }]
|
os: [{ name: ubuntu-22.04, bin-name: linux }, { name: windows-2022, bin-name: windows }, { name: macos-12, bin-name: darwin }]
|
||||||
arch: [amd64, arm64]
|
arch: [amd64, arm64]
|
||||||
exclude:
|
exclude:
|
||||||
- os: { name: windows-2022, bin-name: windows }
|
- os: { name: windows-2022, bin-name: windows }
|
||||||
arch: 'arm64'
|
arch: 'arm64'
|
||||||
- os: { name: macos-12, bin-name: darwin }
|
|
||||||
arch: 'amd64'
|
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
ref: ${{ github.event.inputs.ref }}
|
ref: ${{ github.event.inputs.ref }}
|
||||||
# Allows to fetch all history for all branches and tags. Need this for proper versioning.
|
# Allows to fetch all history for all branches and tags. Need this for proper versioning.
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
|
|
||||||
- name: Set up Go
|
- name: Set up Go
|
||||||
uses: actions/setup-go@v3
|
uses: actions/setup-go@v5
|
||||||
with:
|
with:
|
||||||
go-version: 1.19
|
go-version: '1.22'
|
||||||
cache: true
|
|
||||||
|
|
||||||
- name: Update Go modules
|
|
||||||
run: go mod download -json
|
|
||||||
|
|
||||||
- name: Build CLI
|
- name: Build CLI
|
||||||
run: make build
|
run: make build
|
||||||
|
@ -70,32 +64,38 @@ jobs:
|
||||||
run: mv ./bin/neo-go* ./bin/neo-go-${{ matrix.os.bin-name }}-${{ matrix.arch }}${{ (matrix.os.bin-name == 'windows' && '.exe') || '' }}
|
run: mv ./bin/neo-go* ./bin/neo-go-${{ matrix.os.bin-name }}-${{ matrix.arch }}${{ (matrix.os.bin-name == 'windows' && '.exe') || '' }}
|
||||||
|
|
||||||
- name: Upload artifact
|
- name: Upload artifact
|
||||||
uses: actions/upload-artifact@v3
|
uses: actions/upload-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: neo-go-${{ matrix.os.bin-name }}-${{ matrix.arch }}
|
name: neo-go-${{ matrix.os.bin-name }}-${{ matrix.arch }}
|
||||||
path: ./bin/neo-go*
|
path: ./bin/neo-go*
|
||||||
if-no-files-found: error
|
if-no-files-found: error
|
||||||
|
|
||||||
|
- name: Attach binary to the release as an asset
|
||||||
|
if: ${{ github.event_name == 'release' }}
|
||||||
|
run: gh release upload ${{ github.event.release.tag_name }} ./bin/neo-go-${{ matrix.os.bin-name }}-${{ matrix.arch }}${{ (matrix.os.bin-name == 'windows' && '.exe') || '' }}
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
build_image:
|
build_image:
|
||||||
needs: build_cli
|
needs: build_cli
|
||||||
name: Build and push docker image
|
name: Build and push docker image
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-22.04
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
ref: ${{ github.event.inputs.ref }}
|
ref: ${{ github.event.inputs.ref }}
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
|
|
||||||
- name: Set up QEMU
|
- name: Set up QEMU
|
||||||
uses: docker/setup-qemu-action@v2
|
uses: docker/setup-qemu-action@v3
|
||||||
|
|
||||||
- name: Set up Docker Buildx
|
- name: Set up Docker Buildx
|
||||||
uses: docker/setup-buildx-action@v2
|
uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
- name: Login to DockerHub
|
- name: Login to DockerHub
|
||||||
if: ${{ github.event_name == 'release' || (github.event_name == 'workflow_dispatch' && github.event.inputs.push_image == 'true') }}
|
if: ${{ github.event_name == 'release' || (github.event_name == 'workflow_dispatch' && github.event.inputs.push_image == 'true') }}
|
||||||
uses: docker/login-action@v2
|
uses: docker/login-action@v3
|
||||||
with:
|
with:
|
||||||
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
||||||
password: ${{ secrets.DOCKERHUB_PASSWORD }}
|
password: ${{ secrets.DOCKERHUB_PASSWORD }}
|
||||||
|
@ -110,7 +110,7 @@ jobs:
|
||||||
run: echo "latest=,${{ steps.setvars.outputs.repo }}:latest" >> $GITHUB_OUTPUT
|
run: echo "latest=,${{ steps.setvars.outputs.repo }}:latest" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: Build and push
|
- name: Build and push
|
||||||
uses: docker/build-push-action@v3
|
uses: docker/build-push-action@v5
|
||||||
with:
|
with:
|
||||||
context: .
|
context: .
|
||||||
push: ${{ github.event_name == 'release' || (github.event_name == 'workflow_dispatch' && github.event.inputs.push_image == 'true') }}
|
push: ${{ github.event_name == 'release' || (github.event_name == 'workflow_dispatch' && github.event.inputs.push_image == 'true') }}
|
||||||
|
@ -126,21 +126,20 @@ jobs:
|
||||||
runs-on: windows-2022
|
runs-on: windows-2022
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
ref: ${{ github.event.inputs.ref }}
|
ref: ${{ github.event.inputs.ref }}
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
|
|
||||||
# For proper `deps` make target execution.
|
# For proper `deps` make target execution.
|
||||||
- name: Set up Go
|
- name: Set up Go
|
||||||
uses: actions/setup-go@v3
|
uses: actions/setup-go@v5
|
||||||
with:
|
with:
|
||||||
go-version: 1.19
|
go-version: '1.22'
|
||||||
cache: true
|
|
||||||
|
|
||||||
- name: Login to DockerHub
|
- name: Login to DockerHub
|
||||||
if: ${{ github.event_name == 'release' || (github.event_name == 'workflow_dispatch' && github.event.inputs.push_image == 'true') }}
|
if: ${{ github.event_name == 'release' || (github.event_name == 'workflow_dispatch' && github.event.inputs.push_image == 'true') }}
|
||||||
uses: docker/login-action@v2
|
uses: docker/login-action@v3
|
||||||
with:
|
with:
|
||||||
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
||||||
password: ${{ secrets.DOCKERHUB_PASSWORD }}
|
password: ${{ secrets.DOCKERHUB_PASSWORD }}
|
||||||
|
|
11
.github/workflows/contribution_guidelines.yml
vendored
Normal file
11
.github/workflows/contribution_guidelines.yml
vendored
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
name: Contribution guidelines
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
commits_check_job:
|
||||||
|
name: DCO check
|
||||||
|
uses: nspcc-dev/.github/.github/workflows/dco.yml@master
|
|
@ -8,7 +8,7 @@ on:
|
||||||
- master
|
- master
|
||||||
types: [opened, synchronize]
|
types: [opened, synchronize]
|
||||||
paths-ignore:
|
paths-ignore:
|
||||||
- 'scripts/**'
|
- 'scripts/*.sh'
|
||||||
- '**/*.md'
|
- '**/*.md'
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
|
|
||||||
|
@ -18,26 +18,61 @@ jobs:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/setup-go@v3
|
- uses: actions/checkout@v4
|
||||||
|
- uses: actions/setup-go@v5
|
||||||
with:
|
with:
|
||||||
go-version: 1.19
|
go-version-file: 'go.mod'
|
||||||
- uses: actions/checkout@v3
|
|
||||||
- name: golangci-lint
|
- name: golangci-lint
|
||||||
uses: golangci/golangci-lint-action@v3
|
uses: golangci/golangci-lint-action@v4
|
||||||
with:
|
with:
|
||||||
version: latest
|
version: latest
|
||||||
|
skip-pkg-cache: true # golangci-lint can't work with this cache enabled, ref. https://github.com/golangci/golangci-lint-action/issues/135.
|
||||||
|
|
||||||
gomodcheck:
|
gomodcheck:
|
||||||
name: Check internal dependencies
|
name: Check internal dependencies
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
- name: Check dependencies
|
- name: Check dependencies
|
||||||
run: |
|
run: |
|
||||||
./scripts/check_deps.sh
|
./scripts/check_deps.sh
|
||||||
|
- name: Check go.mod is tidy
|
||||||
|
run: |
|
||||||
|
go mod tidy
|
||||||
|
if [[ $(git diff --name-only go.* | grep '' -c) != 0 ]]; then
|
||||||
|
echo "go mod tidy should be executed before the merge, following packages are unused or out of date:";
|
||||||
|
git diff go.*;
|
||||||
|
exit 1;
|
||||||
|
fi
|
||||||
|
|
||||||
|
codegencheck:
|
||||||
|
name: Check code generated with 'go generate' is up-to-date
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
- uses: actions/setup-go@v5
|
||||||
|
with:
|
||||||
|
go-version-file: 'go.mod'
|
||||||
|
|
||||||
|
- name: Install stringer
|
||||||
|
run: go install golang.org/x/tools/cmd/stringer@latest
|
||||||
|
|
||||||
|
- name: Run go generate
|
||||||
|
run: go generate ./...
|
||||||
|
|
||||||
|
- name: Check that autogenerated code is up-to-date
|
||||||
|
run: |
|
||||||
|
if [[ $(git diff --name-only | grep '' -c) != 0 ]]; then
|
||||||
|
echo "Fresh version of autogenerated code should be committed for the following files:";
|
||||||
|
git diff --name-only;
|
||||||
|
exit 1;
|
||||||
|
fi
|
||||||
|
|
||||||
codeql:
|
codeql:
|
||||||
name: CodeQL
|
name: CodeQL
|
||||||
|
@ -53,11 +88,11 @@ jobs:
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v3
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
# Initializes the CodeQL tools for scanning.
|
# Initializes the CodeQL tools for scanning.
|
||||||
- name: Initialize CodeQL
|
- name: Initialize CodeQL
|
||||||
uses: github/codeql-action/init@v2
|
uses: github/codeql-action/init@v3
|
||||||
with:
|
with:
|
||||||
languages: ${{ matrix.language }}
|
languages: ${{ matrix.language }}
|
||||||
# If you wish to specify custom queries, you can do so here or in a config file.
|
# If you wish to specify custom queries, you can do so here or in a config file.
|
||||||
|
@ -68,7 +103,7 @@ jobs:
|
||||||
# Autobuild attempts to build any compiled languages (C/C++, C#, or Java).
|
# Autobuild attempts to build any compiled languages (C/C++, C#, or Java).
|
||||||
# If this step fails, then you should remove it and run the build manually (see below)
|
# If this step fails, then you should remove it and run the build manually (see below)
|
||||||
- name: Autobuild
|
- name: Autobuild
|
||||||
uses: github/codeql-action/autobuild@v2
|
uses: github/codeql-action/autobuild@v3
|
||||||
|
|
||||||
# ℹ️ Command-line programs to run using the OS shell.
|
# ℹ️ Command-line programs to run using the OS shell.
|
||||||
# 📚 https://git.io/JvXDl
|
# 📚 https://git.io/JvXDl
|
||||||
|
@ -82,41 +117,37 @@ jobs:
|
||||||
# make release
|
# make release
|
||||||
|
|
||||||
- name: Perform CodeQL Analysis
|
- name: Perform CodeQL Analysis
|
||||||
uses: github/codeql-action/analyze@v2
|
uses: github/codeql-action/analyze@v3
|
||||||
|
|
||||||
test_cover:
|
test_cover:
|
||||||
name: Coverage
|
name: Coverage
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-22.04
|
||||||
|
|
||||||
env:
|
env:
|
||||||
CGO_ENABLED: 0
|
CGO_ENABLED: 0
|
||||||
|
GOEXPERIMENT: nocoverageredesign
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
|
submodules: 'true'
|
||||||
- name: Sync VM submodule
|
|
||||||
run: |
|
|
||||||
git submodule sync
|
|
||||||
git submodule update --init
|
|
||||||
|
|
||||||
- name: Set up Go
|
- name: Set up Go
|
||||||
uses: actions/setup-go@v3
|
uses: actions/setup-go@v5
|
||||||
with:
|
with:
|
||||||
go-version: 1.19
|
go-version: '1.22'
|
||||||
cache: true
|
cache: true
|
||||||
|
|
||||||
- name: Update Go modules
|
|
||||||
run: go mod download -json
|
|
||||||
|
|
||||||
- name: Write coverage profile
|
- name: Write coverage profile
|
||||||
run: go test -v ./... -coverprofile=./coverage.txt -covermode=atomic -coverpkg=./pkg...,./cli/...
|
run: go test -timeout 15m -v ./... -coverprofile=./coverage.txt -covermode=atomic -coverpkg=./pkg...,./cli/...
|
||||||
|
|
||||||
- name: Upload coverage results to Codecov
|
- name: Upload coverage results to Codecov
|
||||||
uses: codecov/codecov-action@v2
|
uses: codecov/codecov-action@v4
|
||||||
with:
|
with:
|
||||||
fail_ci_if_error: true # if something is wrong on uploading codecov results, then this job will fail
|
fail_ci_if_error: true # if something is wrong on uploading codecov results, then this job will fail
|
||||||
files: ./coverage.txt
|
files: ./coverage.txt
|
||||||
|
slug: nspcc-dev/neo-go
|
||||||
|
token: ${{ secrets.CODECOV_TOKEN }}
|
||||||
verbose: true
|
verbose: true
|
||||||
|
|
||||||
tests:
|
tests:
|
||||||
|
@ -124,40 +155,36 @@ jobs:
|
||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
os: [ubuntu-20.04, windows-2022, macos-12]
|
os: [ubuntu-22.04, windows-2022, macos-12, macos-14]
|
||||||
go_versions: [ '1.17', '1.18', '1.19' ]
|
go_versions: [ '1.20', '1.21', '1.22' ]
|
||||||
exclude:
|
exclude:
|
||||||
# Only latest Go version for Windows and MacOS.
|
# Only latest Go version for Windows and MacOS.
|
||||||
- os: windows-2022
|
- os: windows-2022
|
||||||
go_versions: '1.17'
|
go_versions: '1.20'
|
||||||
- os: windows-2022
|
- os: windows-2022
|
||||||
go_versions: '1.18'
|
go_versions: '1.21'
|
||||||
- os: macos-12
|
- os: macos-12
|
||||||
go_versions: '1.17'
|
go_versions: '1.20'
|
||||||
- os: macos-12
|
- os: macos-12
|
||||||
go_versions: '1.18'
|
go_versions: '1.21'
|
||||||
|
- os: macos-14
|
||||||
|
go_versions: '1.20'
|
||||||
|
- os: macos-14
|
||||||
|
go_versions: '1.21'
|
||||||
# Exclude latest Go version for Ubuntu as Coverage uses it.
|
# Exclude latest Go version for Ubuntu as Coverage uses it.
|
||||||
- os: ubuntu-20.04
|
- os: ubuntu-22.04
|
||||||
go_versions: '1.19'
|
go_versions: '1.22'
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
|
submodules: 'true'
|
||||||
|
|
||||||
- name: Set up Go
|
- name: Set up Go
|
||||||
uses: actions/setup-go@v3
|
uses: actions/setup-go@v5
|
||||||
with:
|
with:
|
||||||
go-version: '${{ matrix.go_versions }}'
|
go-version: '${{ matrix.go_versions }}'
|
||||||
cache: true
|
|
||||||
|
|
||||||
- name: Update Go modules
|
|
||||||
run: go mod download -json
|
|
||||||
|
|
||||||
- name: Sync VM submodule
|
|
||||||
run: |
|
|
||||||
git submodule sync
|
|
||||||
git submodule update --init
|
|
||||||
|
|
||||||
- name: Run tests
|
- name: Run tests
|
||||||
run: go test -v -race ./...
|
run: go test -timeout 15m -v -race ./...
|
4
.gitignore
vendored
4
.gitignore
vendored
|
@ -7,9 +7,6 @@
|
||||||
# Test binary, build with `go test -c`
|
# Test binary, build with `go test -c`
|
||||||
*.test
|
*.test
|
||||||
|
|
||||||
# Output of the go coverage tool, specifically when used with LiteIDE
|
|
||||||
*.out
|
|
||||||
|
|
||||||
# Added by CoZ developers
|
# Added by CoZ developers
|
||||||
vendor/
|
vendor/
|
||||||
bin/
|
bin/
|
||||||
|
@ -54,6 +51,7 @@ testdata/
|
||||||
!pkg/services/notary/testdata
|
!pkg/services/notary/testdata
|
||||||
!pkg/services/oracle/testdata
|
!pkg/services/oracle/testdata
|
||||||
!pkg/smartcontract/testdata
|
!pkg/smartcontract/testdata
|
||||||
|
!cli/smartcontract/testdata
|
||||||
pkg/vm/testdata/fuzz
|
pkg/vm/testdata/fuzz
|
||||||
!pkg/vm/testdata
|
!pkg/vm/testdata
|
||||||
!pkg/wallet/testdata
|
!pkg/wallet/testdata
|
||||||
|
|
|
@ -69,3 +69,7 @@ issues:
|
||||||
- EXC0003 # test/Test ... consider calling this
|
- EXC0003 # test/Test ... consider calling this
|
||||||
- EXC0004 # govet
|
- EXC0004 # govet
|
||||||
- EXC0005 # C-style breaks
|
- EXC0005 # C-style breaks
|
||||||
|
exclude-rules:
|
||||||
|
- linters:
|
||||||
|
- revive
|
||||||
|
text: "unused-parameter"
|
||||||
|
|
512
CHANGELOG.md
512
CHANGELOG.md
|
@ -2,6 +2,516 @@
|
||||||
|
|
||||||
This document outlines major changes between releases.
|
This document outlines major changes between releases.
|
||||||
|
|
||||||
|
## 0.106.1 "Implication" (3 Jun 2024)
|
||||||
|
|
||||||
|
An urgent release that fixes mainnet state difference at block 5462944 which halts
|
||||||
|
blocks processing starting from the height 5468658. This release requires full node
|
||||||
|
DB resynchronization for mainnet nodes. T5 testnet DB state is not affected by this
|
||||||
|
bug (at least up to the current 4087361 height). Thus, DB resynchronisation may be
|
||||||
|
skipped for testnet nodes. No configuration changes implied.
|
||||||
|
|
||||||
|
Bugs fixed:
|
||||||
|
* mainnet state difference at block 5462944 caused by runtime notification
|
||||||
|
permissions check against the updated contract state instead of executing state
|
||||||
|
(#3472)
|
||||||
|
* unused neofs-contract dependency in the node modules (#3458)
|
||||||
|
|
||||||
|
## 0.106.0 "Zephyranthes" (21 May 2024)
|
||||||
|
|
||||||
|
We're rolling out a large set of updates including all of Neo 3.7.4 protocol changes:
|
||||||
|
native contracts update functionality and extended native contract APIs united under
|
||||||
|
the upcoming Cockatrice hardfork. For smart contract developers an ability to
|
||||||
|
generate smart contract bindings with dynamic hash is supported as well as a number
|
||||||
|
of useful extensions for `unwrap` package, compatible NNS smart contract RPC binding
|
||||||
|
and a lot of other handy enhancements. This release also includes a couple of severe
|
||||||
|
regression bug fixes affecting the node state. As a bonus of fixing a wide range of
|
||||||
|
failing tests, we've refactored the node services start and shutdown procedures to
|
||||||
|
make it more stable. This version drops support for Go 1.19 and requires 1.20+ to
|
||||||
|
build (with Go 1.22 supported).
|
||||||
|
|
||||||
|
Notice that this release requires full node resynchronization for both mainnet
|
||||||
|
and testnet nodes. Please pay a special attention to the Cockatrice hardfork schedule
|
||||||
|
and check your configurations. It should be configured for 3967000 of T5 testnet and
|
||||||
|
5450000 of mainnet. To ensure compatibility your node must be configured
|
||||||
|
appropriately.
|
||||||
|
|
||||||
|
New features:
|
||||||
|
* native contracts update functionality bound to hardforks (#3402, #3444)
|
||||||
|
* Cockatrice hardfork planned for 5450000 block of mainnet and 3967000 block of T5
|
||||||
|
testnet (#3402, #3448, #3402, #3446)
|
||||||
|
* `CommitteeChanged` events are emitted by NeoToken native contract starting from
|
||||||
|
Cockatrice hardfork (#3351, #3448)
|
||||||
|
* `getCommitteeAddress` method of NeoToken native contract is available starting
|
||||||
|
from Cockatrice hardfork (#3362, #3402)
|
||||||
|
* `keccak256` method of CryptoLib native contract is available starting from
|
||||||
|
Cockatrice hardfork (#3301, #3402)
|
||||||
|
* dynamic contract hash support for smart contract bindings (#3405)
|
||||||
|
* support `StringCompressed` API for `crypto.PublicKey` (#3408)
|
||||||
|
* support `Copy` API for `transaction.Transaction` and `payload.P2PNotaryRequest`
|
||||||
|
(#3407)
|
||||||
|
* extend `verifyWithECDsa` method of native CryptoLib contract with Keccak256 hasher
|
||||||
|
starting from Cockatrice and add an example of custom Koblitz-based and
|
||||||
|
Keccak256-based transaction witness verification (#3425)
|
||||||
|
* autogenerated `nativehashes` package (#3402, #3431)
|
||||||
|
* introduce `unwrap.Exception` type for better RPC invocation result exceptions
|
||||||
|
detection (#3438)
|
||||||
|
|
||||||
|
Behavior changes:
|
||||||
|
* Neo Name Service smart contract RPC binding follows the official N3 implementation
|
||||||
|
(#3291)
|
||||||
|
* clear LastGasPerVote NeoToken account info on unvoting (#3349)
|
||||||
|
* return fault exception (if any) in `unwrap` RPC client helpers (#3356)
|
||||||
|
* move P2PNotary designation role out of P2PSigExtensions (#3452)
|
||||||
|
|
||||||
|
Improvements:
|
||||||
|
* support `smartcontract.Convertible` interface as RPC Actor and Invoker call
|
||||||
|
parameters (#3297)
|
||||||
|
* allow to import multisignature account into wallet without WIF and password
|
||||||
|
specified (#3293)
|
||||||
|
* upgrade go-ordered-json dependency to avoid possible node state incompatibility
|
||||||
|
issues caused by smart contract manifest marshalling (#3331, #3333)
|
||||||
|
* Go 1.22 support, bump minimum required Go version up to Go 1.20 (#3336)
|
||||||
|
* a number of dependent libraries are updated to the fresh versions (#3338, #3418)
|
||||||
|
* improve error message of `System.Crypto.CheckMultisig` interop API handler (#3374)
|
||||||
|
* upgrade dBFT library to v0.2.0 (#3371, #3413)
|
||||||
|
* increase `ValidUntilBlock` value for transactions generated by CLI commands
|
||||||
|
(#3376)
|
||||||
|
* documentation updates (#3375, #3382)
|
||||||
|
* let default Notary Actor options be customizable (#3394)
|
||||||
|
* extend `getversion` RPC response with RPC server settings (#3386)
|
||||||
|
* make errors related to wallet opening more detailed (#3389)
|
||||||
|
* adjust error messages of `setExecFeeFactor` and `setStoragePrice` methods of
|
||||||
|
native PolicyContract (#3406)
|
||||||
|
* make neotest chain constructor options customizable (#3404)
|
||||||
|
* format improvements for CLI commands description (#3410)
|
||||||
|
* make state dumps comparator script more verbose (#3411)
|
||||||
|
* refactor storage `Get` implementation for BoltDB (#3441)
|
||||||
|
* add `updatecounter` field to the resulting items of `getnativecontracts` RPC API
|
||||||
|
call (#3439)
|
||||||
|
|
||||||
|
Bugs fixed:
|
||||||
|
* node panic on committee missing from node configuration (#3294)
|
||||||
|
* autogenerated RPC bindings do not follow Go naming convention (#3299)
|
||||||
|
* a batch of failing tests is fixed (#3306, #3313, #3321, #3332, #3337, #3335,
|
||||||
|
#3344, #3340, #3350, #3355, #3364, #3368, #3377, #3393, #3397, #3392, #3398,
|
||||||
|
#3400)
|
||||||
|
* outdated minimum required Go version of boilerplate contract generated by
|
||||||
|
`neo-go contract init` (#3318)
|
||||||
|
* outdated address used as an example in RPC client documentation (#3327)
|
||||||
|
* ungraceful node services shutdown procedure (#3307)
|
||||||
|
* logging data race on node services shutdown (#3307)
|
||||||
|
* Map parameter support is missing from RPC server handlers (#3329)
|
||||||
|
* smart contract storage iterator prefix wasn't copied while iterating over values
|
||||||
|
(#3336)
|
||||||
|
* RPC error with -511 code (insufficient funds) returned on `sendrawtransaction` RPC
|
||||||
|
request should cover multiple cases of sender's insufficient funds (#3360, #3361)
|
||||||
|
* reentrancy possibility for Notary deposit withdrawal (#3357)
|
||||||
|
* null `findstorage` RPC response in case of missing storage items (#3385)
|
||||||
|
* uninitialized named return variables in compiler (#3401)
|
||||||
|
* wrong debug sequence points after `JUMP*` instructions shortening by compiler
|
||||||
|
(#3412)
|
||||||
|
* corrupted genesis block record and application log caused by improper Conflicts
|
||||||
|
attribute processing (#3437)
|
||||||
|
* false positive DB-based blocked accounts detection in native PolicyContract leaded
|
||||||
|
to invalid committee computations and reward distribution made by NeoToken (#3443)
|
||||||
|
|
||||||
|
## 0.105.1 "Enumeration" (12 Jan 2024)
|
||||||
|
|
||||||
|
This is another v3.6.2-compatible release that fixes mainnet state difference at
|
||||||
|
block 4688591. It is confirmed to have the same state up to 4723K height (which
|
||||||
|
is current), but to get proper mainnet state you need to resynchronize your node
|
||||||
|
from the genesis. T5 testnet state is not affected (at least up to the current
|
||||||
|
3323K height), thus DB resynchronisation may be skipped for testnet nodes.
|
||||||
|
|
||||||
|
Improvements:
|
||||||
|
* better network services logging (#3287, #3290)
|
||||||
|
|
||||||
|
Bugs fixed:
|
||||||
|
* state difference at block 4688591 of N3 mainnet caused by difference at
|
||||||
|
characters escaping during manifest's `Extra` field JSON serialisation (#3286)
|
||||||
|
|
||||||
|
## 0.105.0 "Designation" (29 Dec 2023)
|
||||||
|
|
||||||
|
We're rolling out an update for NeoGo nodes that contains a number of user-facing
|
||||||
|
API improvements for RPC web-socket notification subsystem, CLI utility, wallet
|
||||||
|
related packages and not only. Try out our new `--await` CLI option for those
|
||||||
|
commands that relay transactions to the network to automatically wait for the
|
||||||
|
on-chain transaction execution result. Subscribe for new block headers with
|
||||||
|
extended RPC notification subsystem interface. Use contract-based accounts
|
||||||
|
provided by `wallet` package and `neotest` framework to sign transactions. These
|
||||||
|
and a set of other improvements are available to our users while this release is
|
||||||
|
staying compatible with 3.6.2 version of C# node.
|
||||||
|
|
||||||
|
This version also delivers a bug fix for consensus node panic caused by improper
|
||||||
|
native NeoToken cache initialisation. Moreover, there's a set of RPC server side
|
||||||
|
improvements, thus, we recommend to upgrade both consensus and RPC nodes to
|
||||||
|
provide more stable consensus node functioning and extended user APIs functionality.
|
||||||
|
No database resynchronisation is needed.
|
||||||
|
|
||||||
|
New features:
|
||||||
|
* block headers RPC web-socket subscription (#3252)
|
||||||
|
* --await option to synchronize on transaction execution for CLI commands (#3265)
|
||||||
|
* partial session-based RPC iterator unwrapping (#3274)
|
||||||
|
* contract-based transaction signers in neotest framework (#3233)
|
||||||
|
* AMD64 release binaries for macOS (#3251)
|
||||||
|
* complex contract signature schemes in wallet.Account (#3256)
|
||||||
|
|
||||||
|
Behavior changes:
|
||||||
|
* basic RPC subscription filter validity checks are implemented on both RPC
|
||||||
|
client and RPC server sides (#3258)
|
||||||
|
* filter of notary request event RPC subscription is extended with `type` field
|
||||||
|
(#3236)
|
||||||
|
* if available, use block headers RPC web-socket subscription for transaction
|
||||||
|
awaiting via `waiter` package API (#3283)
|
||||||
|
|
||||||
|
Improvements:
|
||||||
|
* add smart contract storage limits to interop utilities (#3232)
|
||||||
|
* extend ZKP examples documentation with additional links to PoT ceremony
|
||||||
|
response files (#3234)
|
||||||
|
* support Go types in VM emitter API (#3237)
|
||||||
|
* documentation update (#3239, #3242, #3246)
|
||||||
|
* BoltDB (go.etcd.io/bbolt) dependency upgrade (#3250)
|
||||||
|
* CLI code refactoring (#2682)
|
||||||
|
* extend wallet package to work with byte slice based wallets (#3255)
|
||||||
|
* export RPC client side transaction awaiting functionality via `waiter` package
|
||||||
|
(#3265, #3283)
|
||||||
|
|
||||||
|
Bugs fixed:
|
||||||
|
* remove stale `updatehistory` section of `getnativecontracts` RPC response (#3240)
|
||||||
|
* immediately check RPC client initialisation on access to blocks RPC subscription
|
||||||
|
API (#3257, #3261)
|
||||||
|
* fix CN panic caused by unexpected call to native NeoToken cache (#3253)
|
||||||
|
* make "automatically generated" warning of all automatically generated files
|
||||||
|
follow the standard (#3280)
|
||||||
|
|
||||||
|
## 0.104.0 "Globalization" (27 Nov 2023)
|
||||||
|
|
||||||
|
We're updating NeoGo to push out a number of useful updates and protocol
|
||||||
|
extensions as well as make it compatible with 3.6.2 version of C# node. The most
|
||||||
|
invasive behaviour changes are VM-level protocol constraints imposed on the size
|
||||||
|
of serialized stackitems and `NativeActivation` node setting removal. This version
|
||||||
|
also delivers a set of smaller useful changes in the interoperability layer and
|
||||||
|
native contract functionality including System.Runtime.CurrentSigners interop,
|
||||||
|
`strLen` StdLib method and PolicyContract-based transaction attributes pricing
|
||||||
|
as far as a user-facing `canceltx` CLI command and community-requested
|
||||||
|
`--relative-path` CLI option.
|
||||||
|
|
||||||
|
Node operators must resynchronize their nodes to get fully compatible state
|
||||||
|
(which is confirmed to be compatible with 3.6.2 for current mainnet up to
|
||||||
|
block 4483627 and T5 testnet up to block 3078762). Please, ensure your node
|
||||||
|
configuration doesn't contain `NativeActivations` protocol configuration section
|
||||||
|
as this logic is hidden under Hardforks starting from the current release.
|
||||||
|
|
||||||
|
New features:
|
||||||
|
* System.Runtime.CurrentSigners interop allowing to get signers of the currently
|
||||||
|
loaded transaction (#3058)
|
||||||
|
* `strLen` method to native StdLib contract (#3208)
|
||||||
|
* transaction attributes pricing regulation via native PolicyContract (#3155)
|
||||||
|
* `canceltx` CLI command as an alternative to unsupported `canceltransaction` RPC
|
||||||
|
request (#3223, #3214)
|
||||||
|
|
||||||
|
Behaviour changes:
|
||||||
|
* reduce maximum allowed stackitem.Item size (#3185)
|
||||||
|
* restrict maximum allowed NEF file size and prohibit large contracts deployment (#3186)
|
||||||
|
* bind `NativeActivations` node setting to the `Hardforks` setting (#3212)
|
||||||
|
* introduce serialization limit to stackitem.Item and fail large contracts deploy
|
||||||
|
wrt this setting (#3218)
|
||||||
|
* add customizable `MaxRequestBodyBytes` and `MaxRequestHeaderBytes` RPC server
|
||||||
|
configuration setting (#3221)
|
||||||
|
|
||||||
|
Improvements:
|
||||||
|
* provide more detailed error on attempt to read non-existent service wallet
|
||||||
|
provided via node configuration file (#3210)
|
||||||
|
* optimize emit of imported code for autogenerated RPC bindings (#3215)
|
||||||
|
* documentation update (#3203, #3222)
|
||||||
|
* add `--relative-path` CLI flag allowing to override configuration-specific relative
|
||||||
|
paths (#3206)
|
||||||
|
* update code owners (#3216, @fyrchik will live in our hearts forever)
|
||||||
|
|
||||||
|
Bugs fixed:
|
||||||
|
* unify messages of RPC errors according to the RPC errors NEP (#3199)
|
||||||
|
* limit maximum allowed number ad depth of transaction signers and witnesses per
|
||||||
|
RPC request that accept transaction (#3207, #3221)
|
||||||
|
* forbid unknown fields usage in the node configuration file (#3209)
|
||||||
|
* enable hardfork-dependant code starting exactly from the block height specified
|
||||||
|
in the node configuration (#3211)
|
||||||
|
* require Notary native deposit to be valid for at least one subsequent block after
|
||||||
|
deposit transaction acceptance (#3211)
|
||||||
|
* deduplicate unnamed event types for autogenerated RPC bindings and make the
|
||||||
|
binding generation order strictly defined and stable (#3215, #3220)
|
||||||
|
* do not panic on trying to compile an import cycle (#3215)
|
||||||
|
* state difference at block 3002333 of T5 testnet caused by difference at characters
|
||||||
|
escaping during manifest's `Extra` field JSON serialisation (#3225)
|
||||||
|
* properly start node services that depend on native RoleManagement contract data
|
||||||
|
with genesis `Roles` protocol configuration setting specified (#3229)
|
||||||
|
|
||||||
|
## 0.103.1 "Verification" (09 Nov 2023)
|
||||||
|
|
||||||
|
An urgent 3.6.0-compatible version that contains a hotfix for the bug that
|
||||||
|
prevents node from starting from the existing database every new dBFT epoch.
|
||||||
|
Also, the release includes a bugfix that fails any non-zero NEO and GAS
|
||||||
|
roundtrips from accounts with zero balance.
|
||||||
|
|
||||||
|
Although the DB format and storage states were not affected by this release,
|
||||||
|
the node resynchronization is still recommended for those who want to keep
|
||||||
|
correct application logs. Resynchronization can be skipped if you're ok with
|
||||||
|
wrong application logs for some transactions (e.g. CN doesn't care, and RPC node
|
||||||
|
cares a lot).
|
||||||
|
|
||||||
|
Bugs fixed:
|
||||||
|
* properly initialize cache of native NeoToken contract every new dBFT epoch (#3187)
|
||||||
|
* forbid non-zero NEO and GAS roundtrips from accounts with zero balance (#3191)
|
||||||
|
|
||||||
|
## 0.103.0 "Backwardation" (20 Oct 2023)
|
||||||
|
|
||||||
|
A minor 3.6.0-compatible version of NeoGo with new salty ZKP-related
|
||||||
|
functionality, a large batch of deprecated APIs removal and an experimental
|
||||||
|
genesis block extension. Build Groth-16 circuits, generate proofs and deploy
|
||||||
|
verification contracts easily with the new end-to-end ZKP example and `zkpbinding`
|
||||||
|
NeoGo API. Setup node roles via the node configuration starting from the genesis
|
||||||
|
block without any painful multisignature transaction forming and invoke any
|
||||||
|
custom script in the genesis-level transaction with the new `Genesis` protocol
|
||||||
|
configuration extension. And don't forget to move from deprecated RPC client
|
||||||
|
APIs, as a lot of them have been removed.
|
||||||
|
|
||||||
|
New node configuration format is finally adapted and the deprecated configuration
|
||||||
|
sections are permanently removed according to the schedule. Pay attention to the
|
||||||
|
`SecondsPerBlock` protocol configuration section that was replaced by
|
||||||
|
`TimePerBlock`, a set of node and services address- and port- related configuration
|
||||||
|
section that were replaced by the new `Addresses` format, direct `UnlockWallet`
|
||||||
|
consensus configuration that was replaced by a separate `Consensus` section and
|
||||||
|
a set of node-specific protocol configurations that were moved under a separate
|
||||||
|
`P2P` section.
|
||||||
|
|
||||||
|
This release fixes a set of bugs including a vulnerability introduced by changes
|
||||||
|
in the transaction conflicts storage scheme implemented in #3061 (a part of
|
||||||
|
0.102.0 release). With the patch presented, it's impossible to uncontrollably
|
||||||
|
pollute the storage with malicious conflicting records.
|
||||||
|
|
||||||
|
This version is fully compatible with C# node 3.6.0. However, it requires
|
||||||
|
complete resynchronization on upgrade due to the database storage scheme changes.
|
||||||
|
|
||||||
|
New features:
|
||||||
|
* API for Groth-16 verification contracts autogeneration and end-to-end example for
|
||||||
|
proving and verifying statements on NeoGo (#3043, #3146)
|
||||||
|
* introduce genesis protocol extensions: default node roles designation and genesis
|
||||||
|
transactions (#3168)
|
||||||
|
|
||||||
|
Behaviour changes:
|
||||||
|
* a lot of deprecated functionality is dropped: RPC client old APIs, shared
|
||||||
|
Notifications channel of WebSocket client, SecondsPerBlock protocol
|
||||||
|
configuration, old way of services and node address and port configuration, old
|
||||||
|
P2P related application settings configuration, direct UnlockWallet consensus
|
||||||
|
configuration, direct node-specific protocol configuration (#3150)
|
||||||
|
* database scheme is changed by new way of storing the transaction conflicting
|
||||||
|
records (#3138)
|
||||||
|
|
||||||
|
Improvements:
|
||||||
|
* NeoFS SDK dependency upgrade (#3129)
|
||||||
|
* gnark and gnark-crypto dependencies upgrade (#3145, #3162, #3163)
|
||||||
|
* introduce timeout restriction for BoltDB opening (#3148)
|
||||||
|
* bump code coverage uploading action version (#3153)
|
||||||
|
* Go 1.21 support, bump minimum required Go version up to Go 1.19 (#3157)
|
||||||
|
* upgrade golang.org/x/net version (#3158)
|
||||||
|
* add protocol hardforks configuration to the `getversion` RPC response (#3160)
|
||||||
|
* format autogenerated smart contract bindings with accordance to standard `go fmt` rules (#3164)
|
||||||
|
* add "DO NOT EDIT" warning to the autogenerated smart contract bindings (#3167)
|
||||||
|
|
||||||
|
Bugs fixed:
|
||||||
|
* avoid race between `getnextblockvalidators` RPC call handler and blockchain's
|
||||||
|
block handler routine, significantly refactor native validators caching scheme
|
||||||
|
(#3110)
|
||||||
|
* do not panic on failed NeoFS oracle requests (#3129)
|
||||||
|
* enable Notary subsystem in NeoFS mainnet configuration (#3136)
|
||||||
|
* reorganize storage scheme for transaction conflicting records to avoid possible attack (#3138)
|
||||||
|
* properly deserialize wildcard trusts field of smart contract manifest (#3140)
|
||||||
|
* use more strict GAS limit for transaction network fee calculation via RPC call (#3141)
|
||||||
|
* avoid WebSocket client request blocking (#3142)
|
||||||
|
* properly emit big uint64 constants during smart contract compilation (#3147)
|
||||||
|
* avoid race access to the node version by tests (#3149)
|
||||||
|
* make FindStorage* RPC client APIs to be always compatible with NeoC# RPC server (#3166)
|
||||||
|
|
||||||
|
## 0.102.0 "Aberration" (06 Sep 2023)
|
||||||
|
|
||||||
|
Long-awaited feature-packed 3.6.0-compatible version of NeoGo with all the
|
||||||
|
appropriate protocol updates and a set of tasty improvements and bug
|
||||||
|
fixes. Groth16 ZKP proof checks are there for contract developers as well as
|
||||||
|
new opcodes. A huge number of improvements went into the RPC server and
|
||||||
|
client, new RPCs, improved contract binding generator and standardized
|
||||||
|
extended error codes make developing dApps much easier.
|
||||||
|
|
||||||
|
Users of smart contract utilities and RPC-related Prometheus metrics are
|
||||||
|
advised to take a look at the deprecated APIs removal schedule
|
||||||
|
(ROADMAP.md). This version was delayed for quite some time (waiting for 3.6),
|
||||||
|
so the next minor release (0.103.0) will remove a substantial amount of
|
||||||
|
compatibility code. It's expected to be released in November with 3.6 protocol
|
||||||
|
compatibility (3.7 cycle is likely to require more time).
|
||||||
|
|
||||||
|
This is also the first version that drops support for Go 1.17 and requires
|
||||||
|
1.18+ to build (with Go 1.20 supported). Using generics in smart contracts is
|
||||||
|
not supported yet, but some elements of this support can be implemented in
|
||||||
|
future versions.
|
||||||
|
|
||||||
|
Mainnet and testnet node operators, please pay attention to the Basilisk
|
||||||
|
hardfork schedule and check your configurations. It will happen at block
|
||||||
|
2680000 for T5 testnet and 4120000 for mainnet. To ensure compatibility your
|
||||||
|
node must be configured appropriately. This version requires DB
|
||||||
|
resynchronization, so schedule your updates accordingly.
|
||||||
|
|
||||||
|
New features:
|
||||||
|
* ZKP proof checking support via CryptoLib native contract (operations with
|
||||||
|
BLS12-381 curve points) (#2940, #3042)
|
||||||
|
* System.Storage.Find interop now support "Backwards" flag to iterate in the
|
||||||
|
opposite direction (#2952)
|
||||||
|
* `util ops` CLI utility that pretty-prints VM script (#2965)
|
||||||
|
* `notarypool_unsorted_tx` Prometheus metric for notary-enabled nodes (#2696)
|
||||||
|
* `CloseNotificationChannelIfFull` WSClient option allowing the client to
|
||||||
|
close notification channel on overflow (#2988)
|
||||||
|
* autogenerated RPC bindings for contract events and conversion code from
|
||||||
|
stackitem.Item to structure (#3008, #3035, #3036, #3087)
|
||||||
|
* event type inference from contract code (#3008)
|
||||||
|
* dynamic contract hashes support for RPC bindings (#3012)
|
||||||
|
* "Basilisk" protocol hardfork support (#3056, #3080, #3086, #3104)
|
||||||
|
* ABORTMSG and ASSERTMSG VM opcodes (#3066)
|
||||||
|
* standardized RPC error codes (#3063)
|
||||||
|
* `findstorage` and `findstoragehistoric` RPC support (#3099)
|
||||||
|
* `getstoragehistoric` RPC support (#3099)
|
||||||
|
* `getrawnotarypool` and `getrawnotarytransaction` RPC (#3098)
|
||||||
|
|
||||||
|
Behaviour changes:
|
||||||
|
* deprecated `FromAddress` smart contract helper is dropped from the `util`
|
||||||
|
interop package (#2941)
|
||||||
|
* a number of deprecated RPC-related Prometheus counters are permanently
|
||||||
|
removed (#2941)
|
||||||
|
* NEP-2 account label will be asked on account import via CLI (#2889)
|
||||||
|
* hashes and states of native contracts are now accessible via native
|
||||||
|
ContractManagement API (#2991)
|
||||||
|
* `serv_node_version` gauge Prometheus metric is marked as deprecated and
|
||||||
|
replaced by `neogo_version` and `server_id` (#3009)
|
||||||
|
* strict contract script check is back with Basilisk hardfork, it was
|
||||||
|
previously removed in 0.101.3 for 3.5 protocol compatibility (#3056)
|
||||||
|
* JSON number deserialization (via StdLib.jsonDeserialize) changes with
|
||||||
|
Basilisk hardfork allowing for more precision and handling more corner
|
||||||
|
cases (#3080)
|
||||||
|
* notification type errors are treated as fatal after Basilisk hardfork,
|
||||||
|
before it they're just logged (#3086)
|
||||||
|
* NULL and non-UTF8 items are not allowed for String event types (#3086)
|
||||||
|
* `Conflicts` and `NotValidBefore` transaction attributes are no longer NeoGo
|
||||||
|
extensions, they can be used on regular networks (#2962)
|
||||||
|
|
||||||
|
Improvements:
|
||||||
|
* documentation and example improvements (#2945, #2972, #2979, #3020, #3084,
|
||||||
|
#3099, #3114, #3116, #3119, #3121)
|
||||||
|
* `getproof` and `verifyproof` RPC API support in RPC client (#2942)
|
||||||
|
* Go 1.20 support, bump minimum required Go version up to Go 1.18 (#2908)
|
||||||
|
* faster state reset process (#2819)
|
||||||
|
* optimized voting data storage scheme for NEO contract (#2892, #2893)
|
||||||
|
* NeoFS SDK dependency upgrades (#2995, #3032)
|
||||||
|
* special exported error returned in case of WSClient disconnection (#3000)
|
||||||
|
* automatic guessing of contract and manifest filenames for `contract
|
||||||
|
compile` CLI command and `loadnef` VM CLI command (#3013)
|
||||||
|
* support for pushing stackitem.Convertible objects via VM script emitter (#3016)
|
||||||
|
* economic adjustment for ranking of transactions with `Conflicts` attribute (#3031)
|
||||||
|
* `google.golang.org/grpc` dependency upgrade fixing high severity security
|
||||||
|
vulnerability (#3055, gRPC is only used to communicate with NeoFS nodes in
|
||||||
|
the oracle service)
|
||||||
|
* enforce default RPC server values when it's used as an independent package
|
||||||
|
(not a part of node) (#3107)
|
||||||
|
* unwrap.Nothing function for RPC clients (#3117)
|
||||||
|
* address and reverse hash display in opcode dumps (#3115)
|
||||||
|
|
||||||
|
Bugs fixed:
|
||||||
|
* invalid peer port type returned by `getpeers` RPC response (#2914)
|
||||||
|
* invalid data source for `mempool_unsorted_tx` Prometheus metric (#2969)
|
||||||
|
* dBFT library upgrade fixing the ability of a single node to speed up the
|
||||||
|
process of new blocks creation for the whole network (#3018,
|
||||||
|
nspcc-dev/dbft#75)
|
||||||
|
* failing CALLT instructions in VM CLI for loaded NEF files (#3020)
|
||||||
|
* missing signers check for on-chain conflicting transactions (#3061)
|
||||||
|
* incorrect sequence point boundaries in debug data (#3074)
|
||||||
|
* compiler panic on encountering generic code (#3041)
|
||||||
|
* potential pooled fallback notary transaction changes (#3108)
|
||||||
|
* lost LastGasPerVote value on NEO state deserialization for non-voting
|
||||||
|
accounts (#3122)
|
||||||
|
|
||||||
|
## 0.101.4 "Yarborough" (01 Aug 2023)
|
||||||
|
|
||||||
|
Another one 3.5.0-compatible version that is aimed to fix T5 testnet state
|
||||||
|
difference that has happened at block 2336911 which leads to inability to process
|
||||||
|
new blocks since 2418703. The issue is fixed by allowing JSON numbers
|
||||||
|
unmarshalling from scientific notation to Integer stackitem. Maximum parsing
|
||||||
|
precision for such numbers is currently restricted by 53 bits. This is a
|
||||||
|
temporary C#-compatible solution that is likely to change in the future versions
|
||||||
|
when an appropriate C# node bug is fixed (neo-project/neo#2879).
|
||||||
|
|
||||||
|
A set of minor bug fixes is included as well to flush some of the long-awaited
|
||||||
|
changes that were blocked by the 0.102.0 release delay (caused by v3.6.0 C# node
|
||||||
|
release delay). In particular, invalid headers returned by an RPC server for
|
||||||
|
error responses, invalid format of incremental dumps created by CLI and deadlock
|
||||||
|
on unhealthy RPC server shutdown. Long-awaited `--config-file` CLI option to
|
||||||
|
start the node providing a single configuration file is added.
|
||||||
|
|
||||||
|
T5 testnet chain requires a complete resynchronization for this version. Mainnet
|
||||||
|
chain resynchronization is recommended, but not required.
|
||||||
|
|
||||||
|
New features:
|
||||||
|
* `--config-file` CLI option allowing to start the node with a single configuration file (#3014)
|
||||||
|
|
||||||
|
Improvements:
|
||||||
|
* blockchain Notary and Oracle services documentation improvement (#2972)
|
||||||
|
* BoltDB (`go.etcd.io/bbolt`) dependency upgrade that fixes a number of Windows-related issues (#3034)
|
||||||
|
|
||||||
|
Bugs fixed:
|
||||||
|
* panic on node start with invalid configuration (#2968)
|
||||||
|
* deadlock on unhealthy RPC server shutdown (#2966)
|
||||||
|
* improper WSClient notification channels managing after disconnection (#2980)
|
||||||
|
* missing Prometheus metric initialisation on node start (#2992)
|
||||||
|
* invalid initialisation of native contracts cache (#2994)
|
||||||
|
* incorrect way of incremental DB dumps creation (#3047)
|
||||||
|
* Notary contract is allowed to be a sender of main Notary request transaction (#3065)
|
||||||
|
* discrepancy in signer's witness scope parsing on the RPC server side (#3060)
|
||||||
|
* Invoker calling API isn't allowed to accept nil parameter (#3067)
|
||||||
|
* contract RPC Client unwrapper helper can't handle missing contract case (#3072)
|
||||||
|
* JSON numbers can't be unmarshalled to stackitem from scientific notation (#3073)
|
||||||
|
* invalid content-type header returned by RPC server on error responses (#3075)
|
||||||
|
|
||||||
|
## 0.101.3 "Yuckiness" (08 Jul 2023)
|
||||||
|
|
||||||
|
Yet another 3.5.0-compatible emergency version that removes scrupulous
|
||||||
|
instructions check of smart contract's script on contract deployment or update.
|
||||||
|
Presence of this check leads to the known T5 testnet state incompatibility
|
||||||
|
(since 1670095) which causes inability to process new blocks (since 2272533).
|
||||||
|
It should be noted that the corresponding check was accidentally removed from
|
||||||
|
the reference C# node implementation way back in neo-project/neo#2266 and added
|
||||||
|
again in neo-project/neo#2849 which is planned to be a part of the upcoming
|
||||||
|
3.6.0 C# node release. Thus, changes made in the presented 0.101.3 release will
|
||||||
|
be reverted afterwards and strict contract script check will be present in the
|
||||||
|
next 3.6.0-compatible version of NeoGo node.
|
||||||
|
|
||||||
|
T5 testnet chain requires a complete resynchronization for this version. Mainnet
|
||||||
|
chain resynchronization is recommended.
|
||||||
|
|
||||||
|
Bugs fixed:
|
||||||
|
|
||||||
|
* extra strict contract script check on contract deployment or update is
|
||||||
|
removed (#3052)
|
||||||
|
|
||||||
|
## 0.101.2 "Excavation" (29 Jun 2023)
|
||||||
|
|
||||||
|
One more (and unexpected one!) 3.5.0-compatible version that fixes a VM bug
|
||||||
|
leading to mainnet state incompatibility (since 3672783) which in turn leads
|
||||||
|
to inability to process new blocks (since 3682290).
|
||||||
|
|
||||||
|
Mainnet chain needs to be resynchronized for this version.
|
||||||
|
|
||||||
|
Improvements:
|
||||||
|
* documentation updates (#3029, #2990, #2981)
|
||||||
|
|
||||||
|
Bugs fixed:
|
||||||
|
* incorrect handling of empty Any-type parameter for RPC invocations (#2959)
|
||||||
|
* incorrect state rollbacks in case of exception during cross-contract call
|
||||||
|
when the call is made from non-TRYing context (#3046)
|
||||||
|
|
||||||
## 0.101.1 "Shallowness" (17 Mar 2023)
|
## 0.101.1 "Shallowness" (17 Mar 2023)
|
||||||
|
|
||||||
Another 3.5.0-compatible version that delivers important bug fixes and
|
Another 3.5.0-compatible version that delivers important bug fixes and
|
||||||
|
@ -2099,7 +2609,7 @@ Behavior changes:
|
||||||
* contracts no longer have single entry point, rather they export a set of
|
* contracts no longer have single entry point, rather they export a set of
|
||||||
methods with specific offsets. Go smart contract compiler has been changed
|
methods with specific offsets. Go smart contract compiler has been changed
|
||||||
accordingly to add all exported (as in Go) methods to the manifest
|
accordingly to add all exported (as in Go) methods to the manifest
|
||||||
(but with the first letter being lowercased to match NEP-5 expections,
|
(but with the first letter being lowercased to match NEP-5 expectations,
|
||||||
#1228). Please also refer to examples changes to better see how it affects
|
#1228). Please also refer to examples changes to better see how it affects
|
||||||
contracts, manifests and configuration files (#1296)
|
contracts, manifests and configuration files (#1296)
|
||||||
* native contracts are now called via Neo.Native.Call syscall (#1191)
|
* native contracts are now called via Neo.Native.Call syscall (#1191)
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
# Builder image
|
# Builder image
|
||||||
# Keep go version in sync with Build GA job.
|
# Keep go version in sync with Build GA job.
|
||||||
FROM golang:1.18-alpine as builder
|
FROM golang:1.22-alpine as builder
|
||||||
|
|
||||||
# Display go version for information purposes.
|
# Display go version for information purposes.
|
||||||
RUN go version
|
RUN go version
|
||||||
|
|
|
@ -1,75 +1,6 @@
|
||||||
# Builder image
|
# Builder image
|
||||||
FROM mcr.microsoft.com/windows/servercore:ltsc2022 as builder
|
|
||||||
|
|
||||||
SHELL ["powershell", "-Command", "$ErrorActionPreference = 'Stop';", "$ProgressPreference = 'SilentlyContinue';"]
|
|
||||||
|
|
||||||
ENV GIT_VERSION=2.23.0
|
|
||||||
|
|
||||||
ENV GIT_TAG=v2.23.0.windows.1
|
|
||||||
|
|
||||||
ENV GIT_DOWNLOAD_URL=https://github.com/git-for-windows/git/releases/download/v2.23.0.windows.1/MinGit-2.23.0-64-bit.zip
|
|
||||||
|
|
||||||
ENV GIT_DOWNLOAD_SHA256=8f65208f92c0b4c3ae4c0cf02d4b5f6791d539cd1a07b2df62b7116467724735
|
|
||||||
|
|
||||||
RUN Write-Host ('Downloading {0} ...' -f $env:GIT_DOWNLOAD_URL); \
|
|
||||||
[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12; \
|
|
||||||
Invoke-WebRequest -Uri $env:GIT_DOWNLOAD_URL -OutFile 'git.zip'; \
|
|
||||||
\
|
|
||||||
Write-Host ('Verifying sha256 ({0}) ...' -f $env:GIT_DOWNLOAD_SHA256); \
|
|
||||||
if ((Get-FileHash git.zip -Algorithm sha256).Hash -ne $env:GIT_DOWNLOAD_SHA256) { \
|
|
||||||
Write-Host 'FAILED!'; \
|
|
||||||
exit 1; \
|
|
||||||
}; \
|
|
||||||
\
|
|
||||||
Write-Host 'Expanding ...'; \
|
|
||||||
Expand-Archive -Path git.zip -DestinationPath C:\git\.; \
|
|
||||||
\
|
|
||||||
Write-Host 'Removing ...'; \
|
|
||||||
Remove-Item git.zip -Force; \
|
|
||||||
\
|
|
||||||
Write-Host 'Updating PATH ...'; \
|
|
||||||
$env:PATH = 'C:\git\cmd;C:\git\mingw64\bin;C:\git\usr\bin;' + $env:PATH; \
|
|
||||||
[Environment]::SetEnvironmentVariable('PATH', $env:PATH, [EnvironmentVariableTarget]::Machine); \
|
|
||||||
\
|
|
||||||
Write-Host 'Verifying install ("git version") ...'; \
|
|
||||||
git version; \
|
|
||||||
\
|
|
||||||
Write-Host 'Complete.';
|
|
||||||
|
|
||||||
ENV GOPATH=C:\\go
|
|
||||||
|
|
||||||
RUN $newPath = ('{0}\bin;C:\Program Files\Go\bin;{1}' -f $env:GOPATH, $env:PATH); \
|
|
||||||
Write-Host ('Updating PATH: {0}' -f $newPath); \
|
|
||||||
[Environment]::SetEnvironmentVariable('PATH', $newPath, [EnvironmentVariableTarget]::Machine);
|
|
||||||
|
|
||||||
# Keep go version in sync with Build GA job.
|
# Keep go version in sync with Build GA job.
|
||||||
ENV GOLANG_VERSION=1.19
|
FROM golang:1.22.0-windowsservercore-ltsc2022 as builder
|
||||||
|
|
||||||
RUN $url = 'https://go.dev/dl/go1.19.windows-amd64.zip'; \
|
|
||||||
Write-Host ('Downloading {0} ...' -f $url); \
|
|
||||||
[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12; \
|
|
||||||
Invoke-WebRequest -Uri $url -OutFile 'go.zip'; \
|
|
||||||
\
|
|
||||||
$sha256 = 'bcaaf966f91980d35ae93c37a8fe890e4ddfca19448c0d9f66c027d287e2823a'; \
|
|
||||||
Write-Host ('Verifying sha256 ({0}) ...' -f $sha256); \
|
|
||||||
if ((Get-FileHash go.zip -Algorithm sha256).Hash -ne $sha256) { \
|
|
||||||
Write-Host 'FAILED!'; \
|
|
||||||
exit 1; \
|
|
||||||
}; \
|
|
||||||
\
|
|
||||||
Write-Host 'Expanding ...'; \
|
|
||||||
Expand-Archive go.zip -DestinationPath C:\; \
|
|
||||||
\
|
|
||||||
Write-Host 'Moving ...'; \
|
|
||||||
Move-Item -Path C:\go -Destination 'C:\Program Files\Go'; \
|
|
||||||
\
|
|
||||||
Write-Host 'Removing ...'; \
|
|
||||||
Remove-Item go.zip -Force; \
|
|
||||||
\
|
|
||||||
Write-Host 'Verifying install ("go version") ...'; \
|
|
||||||
go version; \
|
|
||||||
\
|
|
||||||
Write-Host 'Complete.';
|
|
||||||
|
|
||||||
COPY . /neo-go
|
COPY . /neo-go
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
MIT License
|
MIT License
|
||||||
|
|
||||||
Copyright (c) 2018-2022 Anthony De Meulemeester (@anthdm), City of Zion
|
Copyright (c) 2018-2023 NeoSPCC (@nspcc-dev), Anthony De Meulemeester (@anthdm), City of Zion community (@CityOfZion)
|
||||||
community (@CityOfZion), NeoSPCC (@nspcc-dev)
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
|
2
Makefile
2
Makefile
|
@ -3,7 +3,7 @@ REPONAME = "neo-go"
|
||||||
NETMODE ?= "privnet"
|
NETMODE ?= "privnet"
|
||||||
BINARY=neo-go
|
BINARY=neo-go
|
||||||
BINARY_PATH=./bin/$(BINARY)$(shell go env GOEXE)
|
BINARY_PATH=./bin/$(BINARY)$(shell go env GOEXE)
|
||||||
GO_VERSION ?= 1.18
|
GO_VERSION ?= 1.20
|
||||||
DESTDIR = ""
|
DESTDIR = ""
|
||||||
SYSCONFIGDIR = "/etc"
|
SYSCONFIGDIR = "/etc"
|
||||||
BINDIR = "/usr/bin"
|
BINDIR = "/usr/bin"
|
||||||
|
|
|
@ -12,7 +12,7 @@
|
||||||
<hr />
|
<hr />
|
||||||
|
|
||||||
[![codecov](https://codecov.io/gh/nspcc-dev/neo-go/branch/master/graph/badge.svg)](https://codecov.io/gh/nspcc-dev/neo-go)
|
[![codecov](https://codecov.io/gh/nspcc-dev/neo-go/branch/master/graph/badge.svg)](https://codecov.io/gh/nspcc-dev/neo-go)
|
||||||
[![GithubWorkflows Tests](https://github.com/nspcc-dev/neo-go/actions/workflows/run_tests.yml/badge.svg)](https://github.com/nspcc-dev/neo-go/actions/workflows/run_tests.yml)
|
[![GithubWorkflows Tests](https://github.com/nspcc-dev/neo-go/actions/workflows/tests.yml/badge.svg)](https://github.com/nspcc-dev/neo-go/actions/workflows/tests.yml)
|
||||||
[![Report](https://goreportcard.com/badge/github.com/nspcc-dev/neo-go)](https://goreportcard.com/report/github.com/nspcc-dev/neo-go)
|
[![Report](https://goreportcard.com/badge/github.com/nspcc-dev/neo-go)](https://goreportcard.com/report/github.com/nspcc-dev/neo-go)
|
||||||
[![GoDoc](https://godoc.org/github.com/nspcc-dev/neo-go?status.svg)](https://godoc.org/github.com/nspcc-dev/neo-go)
|
[![GoDoc](https://godoc.org/github.com/nspcc-dev/neo-go?status.svg)](https://godoc.org/github.com/nspcc-dev/neo-go)
|
||||||
![GitHub release (latest SemVer)](https://img.shields.io/github/v/release/nspcc-dev/neo-go?sort=semver)
|
![GitHub release (latest SemVer)](https://img.shields.io/github/v/release/nspcc-dev/neo-go?sort=semver)
|
||||||
|
@ -51,7 +51,7 @@ NeoGo, `:latest` points to the latest release) or build yourself.
|
||||||
|
|
||||||
### Building
|
### Building
|
||||||
|
|
||||||
Building NeoGo requires Go 1.17+ and `make`:
|
Building NeoGo requires Go 1.20+ and `make`:
|
||||||
|
|
||||||
```
|
```
|
||||||
make
|
make
|
||||||
|
@ -180,9 +180,8 @@ describing the feature/topic you are going to implement.
|
||||||
|
|
||||||
# Contact
|
# Contact
|
||||||
|
|
||||||
- [@roman-khimov](https://github.com/roman-khimov) on GitHub
|
|
||||||
- [@AnnaShaleva](https://github.com/AnnaShaleva) on GitHub
|
- [@AnnaShaleva](https://github.com/AnnaShaleva) on GitHub
|
||||||
- [@fyrchik](https://github.com/fyrchik) on GitHub
|
- [@roman-khimov](https://github.com/roman-khimov) on GitHub
|
||||||
- Reach out to us on the [Neo Discord](https://discordapp.com/invite/R8v48YA) channel
|
- Reach out to us on the [Neo Discord](https://discordapp.com/invite/R8v48YA) channel
|
||||||
|
|
||||||
# License
|
# License
|
||||||
|
|
147
ROADMAP.md
147
ROADMAP.md
|
@ -7,14 +7,13 @@ functionality.
|
||||||
## Versions 0.7X.Y (as needed)
|
## Versions 0.7X.Y (as needed)
|
||||||
* Neo 2.0 support (bug fixes, minor functionality additions)
|
* Neo 2.0 support (bug fixes, minor functionality additions)
|
||||||
|
|
||||||
## Version 0.102.0 (~March 2022)
|
## Version 0.107.0 (~Jun-Jul 2024)
|
||||||
* 3.6.0 compatibility
|
* protocol updates
|
||||||
|
* bug fixes
|
||||||
|
* node resynchronisation from local DB
|
||||||
|
* CLI library upgrade
|
||||||
|
|
||||||
## Version 0.102.1 (~April 2022)
|
## Version 1.0 (2024, TBD)
|
||||||
* improved RPC error codes
|
|
||||||
* extended data types for iterators to be used by RPC wrapper generator
|
|
||||||
|
|
||||||
## Version 1.0 (2023, TBD)
|
|
||||||
* stable version
|
* stable version
|
||||||
|
|
||||||
# Deprecated functionality
|
# Deprecated functionality
|
||||||
|
@ -26,111 +25,47 @@ APIs/commands/configurations will be removed and here is a list of scheduled
|
||||||
breaking changes. Consider changing your code/scripts/configurations if you're
|
breaking changes. Consider changing your code/scripts/configurations if you're
|
||||||
using anything mentioned here.
|
using anything mentioned here.
|
||||||
|
|
||||||
## Old RPC client APIs
|
## GetPeers RPC server response type changes and RPC client support
|
||||||
|
|
||||||
A huge set of RPC client APIs was deprecated in versions 0.99.2 and 0.99.3
|
GetPeers RPC command returns a list of Peers where the port type has changed from
|
||||||
(August-September 2022), including very frequently used ones like
|
string to uint16 to match C#. The RPC client currently supports unmarshalling both
|
||||||
SignAndPushInvocationTx, AddNetworkFee, TransferNEP17. A new set of
|
formats.
|
||||||
invoker/actor/unwrap/nep17/etc packages was introduced decoupling these
|
|
||||||
functions from RPC client and simplifying typical backend code. Please refer
|
|
||||||
to rpcclient package documentation for specific replacements for each of these
|
|
||||||
APIs and convert your code to using them.
|
|
||||||
|
|
||||||
While a lot of the code is already converted to new APIs, old ones still can
|
Removal of Peer unmarshalling with string based ports is scheduled for Jun-Jul 2024
|
||||||
be used in some code not known to us. Therefore we will remove old APIs not
|
(~0.107.0 release).
|
||||||
earlier than May 2023, with 0.103.0 release.
|
|
||||||
|
|
||||||
## util.FromAddress smart contract helper
|
## `NEOBalance` from stack item
|
||||||
|
|
||||||
`util` smart contract library has a FromAddress function that is one of the
|
We check struct items count before convert LastGasPerVote to let RPC client be compatible with
|
||||||
oldest lines in the entire NeoGo code base, dating back to 2018. Version
|
old versions.
|
||||||
0.99.4 of NeoGo (October 2022) has introduced a new `address` package with
|
|
||||||
`ToHash160` function, it covers a bit more use cases but can be used as a
|
|
||||||
direct replacement of the old function, so please update your code.
|
|
||||||
|
|
||||||
util.FromAddress is expected to be removed around March 2023 (~0.102.0
|
Removal of this compatiblility code is scheduled for Jun-Jul 2024.
|
||||||
|
|
||||||
|
## `serv_node_version` Prometheus gauge metric
|
||||||
|
|
||||||
|
This metric is replaced by the new `neogo_version` and `server_id` Prometheus gauge
|
||||||
|
metrics with proper version formatting. `neogo_version` contains NeoGo version
|
||||||
|
hidden under `version` label and `server_id` contains network server ID hidden
|
||||||
|
under `server_id` label.
|
||||||
|
|
||||||
|
Removal of `serv_node_version` is scheduled for Jun-Jul 2024 (~0.107.0 release).
|
||||||
|
|
||||||
|
## RPC error codes returned by old versions and C#-nodes
|
||||||
|
|
||||||
|
NeoGo retains certain deprecated error codes: `neorpc.ErrCompatGeneric`,
|
||||||
|
`neorpc.ErrCompatNoOpenedWallet`. They returned by nodes not compliant with the
|
||||||
|
neo-project/proposals#156 (NeoGo pre-0.102.0 and all known C# versions).
|
||||||
|
|
||||||
|
Removal of the deprecated RPC error codes is planned for Jun-Jul 2024 (~0.107.0
|
||||||
release).
|
release).
|
||||||
|
|
||||||
## WSClient Notifications channel and SubscribeFor* APIs
|
## Block based web-socket waiter transaction awaiting
|
||||||
|
|
||||||
Version 0.99.5 of NeoGo introduces a new set of subscription APIs that gives
|
Web-socket RPC based `waiter.EventWaiter` uses `header_of_added_block` notifications
|
||||||
more control to the WSClient user that can pass specific channels to be used
|
subscription to manage transaction awaiting. To support old NeoGo RPC servers
|
||||||
for specific subscriptions now. Old APIs and generic Notifications channel are
|
(older than 0.105.0) that do not have block headers subscription ability,
|
||||||
still available, but will be removed, so please convert your code to using new
|
event-based waiter fallbacks to the old way of block monitoring with
|
||||||
Receive* APIs.
|
`block_added` notifications subscription.
|
||||||
|
|
||||||
Removal of these APIs is scheduled for May 2023 (~0.103.0 release).
|
Removal of stale RPC server compatibility code from `waiter.EventWaiter` is
|
||||||
|
scheduled for Jun-Jul 2024 (~0.107.0 release).
|
||||||
## Prometheus RPC counters
|
|
||||||
|
|
||||||
A number of neogo_${method}_called Prometheus counters are marked as
|
|
||||||
deprecated since version 0.99.5, neogo_rpc_${method}_time histograms can be
|
|
||||||
used instead (that also have a counter).
|
|
||||||
|
|
||||||
It's not a frequently used thing and it's easy to replace it, so removal of
|
|
||||||
old counters is scheduled for January-February 2023 (~0.100.X release).
|
|
||||||
|
|
||||||
## SecondsPerBlock protocol configuration
|
|
||||||
|
|
||||||
With 0.100.0 version SecondsPerBlock protocol configuration setting was
|
|
||||||
deprecated and replaced by a bit more generic and precise TimePerBlock
|
|
||||||
(allowing for subsecond time). An informational message is printed on node
|
|
||||||
startup to inform about this, it's very easy to deal with this configuration
|
|
||||||
change, just replace one line.
|
|
||||||
|
|
||||||
Removal of SecondsPerBlock is scheduled for May-June 2023 (~0.103.0 release).
|
|
||||||
|
|
||||||
## Services/node address and port configuration
|
|
||||||
|
|
||||||
Version 0.100.0 of NeoGo introduces a multiple binding addresses capability to
|
|
||||||
the node's services (RPC server, TLS RPC configuration, Prometheus, Pprof) and
|
|
||||||
the node itself. It allows to specify several listen addresses/ports using an
|
|
||||||
array of "address:port" pairs in the service's `Addresses` config section and
|
|
||||||
array of "address:port:announcedPort" tuples in the `ApplicationConfiguration`'s
|
|
||||||
`Addresses` node config section. Deprecated `Address` and `Port` sections of
|
|
||||||
`RPC`, `Prometheus`, `Pprof` subsections of the `ApplicationConfiguration`
|
|
||||||
as far as the one of RPC server's `TLSConfig` are still available, but will be
|
|
||||||
removed, so please convert your node configuration file to use new `P2P`-level
|
|
||||||
`Addresses` section for the node services. Deprecated `Address`, `NodePort` and
|
|
||||||
`AnnouncedPort` sections of `ApplicationConfiguration` will also be removed
|
|
||||||
eventually, so please update your node configuration file to use `Addresses`
|
|
||||||
section for the P2P addresses configuration.
|
|
||||||
|
|
||||||
Removal of these config sections is scheduled for May-June 2023 (~0.103.0 release).
|
|
||||||
|
|
||||||
## P2P application settings configuration
|
|
||||||
|
|
||||||
Version 0.100.0 of NeoGo marks the following P2P application settings as
|
|
||||||
deprecated: `AttemptConnPeers`, `BroadcastFactor`, `DialTimeout`,
|
|
||||||
`ExtensiblePoolSize`, `MaxPeers`, `MinPeers`, `PingInterval`, `PingTimeout`,
|
|
||||||
`ProtoTickInterval`. These settings are moved to a separate `P2P` section of
|
|
||||||
`ApplicationConfiguration`. The `DialTimeout`, `PingInterval`, `PingTimeout`,
|
|
||||||
`ProtoTickInterval` settings are converted to more precise `Duration` format
|
|
||||||
(allowing for subsecond time). Please, update your node configuration (all you
|
|
||||||
need is to move specified settings under the `P2P` section and convert
|
|
||||||
time-related settings to `Duration` format).
|
|
||||||
|
|
||||||
Removal of deprecated P2P related application settings is scheduled for May-June
|
|
||||||
2023 (~0.103.0 release).
|
|
||||||
|
|
||||||
## Direct UnlockWallet consensus configuration
|
|
||||||
|
|
||||||
Top-level UnlockWallet section in ApplicationConfiguration was used as an
|
|
||||||
implicit consensus service configuration, now this setting (with Enabled flag)
|
|
||||||
is moved into a section of its own (Consensus). Old configurations are still
|
|
||||||
supported, but this support will eventually be removed.
|
|
||||||
|
|
||||||
Removal of this compatibility code is scheduled for May-June 2023 (~0.103.0
|
|
||||||
release).
|
|
||||||
|
|
||||||
## Node-specific configuration moved from Protocol to Application
|
|
||||||
|
|
||||||
GarbageCollectionPeriod, KeepOnlyLatestState, RemoveUntraceableBlocks,
|
|
||||||
SaveStorageBatch and VerifyBlocks settings were moved from
|
|
||||||
ProtocolConfiguration to ApplicationConfiguration in version 0.100.0. Old
|
|
||||||
configurations are still supported, except for VerifyBlocks which is replaced
|
|
||||||
by SkipBlockVerification with inverted meaning (and hence an inverted default)
|
|
||||||
for security reasons.
|
|
||||||
|
|
||||||
Removal of these options from ProtocolConfiguration is scheduled for May-June
|
|
||||||
2023 (~0.103.0 release).
|
|
|
@ -4,11 +4,12 @@ import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/internal/testcli"
|
"github.com/nspcc-dev/neo-go/internal/testcli"
|
||||||
|
"github.com/nspcc-dev/neo-go/internal/versionutil"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/config"
|
"github.com/nspcc-dev/neo-go/pkg/config"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestCLIVersion(t *testing.T) {
|
func TestCLIVersion(t *testing.T) {
|
||||||
config.Version = "0.90.0-test" // Zero-length version string disables '--version' completely.
|
config.Version = versionutil.TestVersion // Zero-length version string disables '--version' completely.
|
||||||
e := testcli.NewExecutor(t, false)
|
e := testcli.NewExecutor(t, false)
|
||||||
e.Run(t, "neo-go", "--version")
|
e.Run(t, "neo-go", "--version")
|
||||||
e.CheckNextLine(t, "^NeoGo")
|
e.CheckNextLine(t, "^NeoGo")
|
||||||
|
|
|
@ -84,8 +84,8 @@ const (
|
||||||
* 'dead' is a byte array with a value of 'dead'
|
* 'dead' is a byte array with a value of 'dead'
|
||||||
* 'string:dead' is a string with a value of 'dead'
|
* 'string:dead' is a string with a value of 'dead'
|
||||||
* 'filebytes:my_data.txt' is bytes decoded from a content of my_data.txt
|
* 'filebytes:my_data.txt' is bytes decoded from a content of my_data.txt
|
||||||
* 'AK2nJJpJr6o664CWJKi1QRXjqeic2zRp8y' is a hash160 with a value
|
* 'NSiVJYZej4XsxG5CUpdwn7VRQk8iiiDMPM' is a hash160 with a value
|
||||||
of '23ba2703c53263e8d6e522dc32203339dcd8eee9'
|
of '682cca3ebdc66210e5847d7f8115846586079d4a'
|
||||||
* '\4\2' is an integer with a value of 42
|
* '\4\2' is an integer with a value of 42
|
||||||
* '\\4\2' is a string with a value of '\42'
|
* '\\4\2' is a string with a value of '\42'
|
||||||
* 'string:string' is a string with a value of 'string'
|
* 'string:string' is a string with a value of 'string'
|
||||||
|
@ -230,9 +230,9 @@ func parseCosigner(c string) (transaction.Signer, error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetDataFromContext returns data parameter from context args.
|
// GetDataFromContext returns data parameter from context args.
|
||||||
func GetDataFromContext(ctx *cli.Context) (int, interface{}, *cli.ExitError) {
|
func GetDataFromContext(ctx *cli.Context) (int, any, *cli.ExitError) {
|
||||||
var (
|
var (
|
||||||
data interface{}
|
data any
|
||||||
offset int
|
offset int
|
||||||
params []smartcontract.Parameter
|
params []smartcontract.Parameter
|
||||||
err error
|
err error
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
package cmdargs
|
package cmdargs
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/hex"
|
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
@ -45,7 +44,7 @@ func TestParseCosigner(t *testing.T) {
|
||||||
Scopes: transaction.CalledByEntry | transaction.CustomContracts,
|
Scopes: transaction.CalledByEntry | transaction.CustomContracts,
|
||||||
AllowedContracts: []util.Uint160{c1, c2},
|
AllowedContracts: []util.Uint160{c1, c2},
|
||||||
},
|
},
|
||||||
acc.StringLE() + ":CustomGroups:" + hex.EncodeToString(priv.PublicKey().Bytes()): {
|
acc.StringLE() + ":CustomGroups:" + priv.PublicKey().StringCompressed(): {
|
||||||
Account: acc,
|
Account: acc,
|
||||||
Scopes: transaction.CustomGroups,
|
Scopes: transaction.CustomGroups,
|
||||||
AllowedGroups: keys.PublicKeys{priv.PublicKey()},
|
AllowedGroups: keys.PublicKeys{priv.PublicKey()},
|
||||||
|
|
|
@ -14,6 +14,7 @@ import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/internal/testcli"
|
"github.com/nspcc-dev/neo-go/internal/testcli"
|
||||||
|
"github.com/nspcc-dev/neo-go/internal/versionutil"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/native/nativenames"
|
"github.com/nspcc-dev/neo-go/pkg/core/native/nativenames"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
||||||
|
@ -29,6 +30,9 @@ const (
|
||||||
nftOwnerAddr = "NbrUYaZgyhSkNoRo9ugRyEMdUZxrhkNaWB"
|
nftOwnerAddr = "NbrUYaZgyhSkNoRo9ugRyEMdUZxrhkNaWB"
|
||||||
nftOwnerWallet = "../../examples/my_wallet.json"
|
nftOwnerWallet = "../../examples/my_wallet.json"
|
||||||
nftOwnerPass = "qwerty"
|
nftOwnerPass = "qwerty"
|
||||||
|
|
||||||
|
// Keep contract NEFs consistent between runs.
|
||||||
|
_ = versionutil.TestVersion
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestNEP11Import(t *testing.T) {
|
func TestNEP11Import(t *testing.T) {
|
||||||
|
@ -324,6 +328,14 @@ func TestNEP11_ND_OwnerOf_BalanceOf_Transfer(t *testing.T) {
|
||||||
e.Run(t, append(cmdCheckBalance, "--token", h.StringLE())...)
|
e.Run(t, append(cmdCheckBalance, "--token", h.StringLE())...)
|
||||||
checkBalanceResult(t, nftOwnerAddr, tokenID1)
|
checkBalanceResult(t, nftOwnerAddr, tokenID1)
|
||||||
|
|
||||||
|
// check --await flag
|
||||||
|
tokenID2 := mint(t)
|
||||||
|
e.In.WriteString(nftOwnerPass + "\r")
|
||||||
|
e.Run(t, append(cmdTransfer, "--await", "--id", hex.EncodeToString(tokenID2))...)
|
||||||
|
e.CheckAwaitableTxPersisted(t)
|
||||||
|
e.Run(t, append(cmdCheckBalance, "--token", h.StringLE())...)
|
||||||
|
checkBalanceResult(t, nftOwnerAddr, tokenID1)
|
||||||
|
|
||||||
// transfer: good, to NEP-11-Payable contract, with data
|
// transfer: good, to NEP-11-Payable contract, with data
|
||||||
verifyH := deployVerifyContract(t, e)
|
verifyH := deployVerifyContract(t, e)
|
||||||
cmdTransfer = []string{
|
cmdTransfer = []string{
|
||||||
|
|
|
@ -19,19 +19,34 @@ import (
|
||||||
|
|
||||||
func TestNEP17Balance(t *testing.T) {
|
func TestNEP17Balance(t *testing.T) {
|
||||||
e := testcli.NewExecutor(t, true)
|
e := testcli.NewExecutor(t, true)
|
||||||
|
|
||||||
|
args := []string{
|
||||||
|
"neo-go", "wallet", "nep17", "multitransfer",
|
||||||
|
"--rpc-endpoint", "http://" + e.RPC.Addresses()[0],
|
||||||
|
"--wallet", testcli.ValidatorWallet,
|
||||||
|
"--from", testcli.ValidatorAddr,
|
||||||
|
"GAS:" + testcli.TestWalletMultiAccount1 + ":1",
|
||||||
|
"NEO:" + testcli.TestWalletMultiAccount1 + ":10",
|
||||||
|
"GAS:" + testcli.TestWalletMultiAccount3 + ":3",
|
||||||
|
"--force",
|
||||||
|
}
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.Run(t, args...)
|
||||||
|
e.CheckTxPersisted(t)
|
||||||
|
|
||||||
cmdbalance := []string{"neo-go", "wallet", "nep17", "balance"}
|
cmdbalance := []string{"neo-go", "wallet", "nep17", "balance"}
|
||||||
cmdbase := append(cmdbalance,
|
cmdbase := append(cmdbalance,
|
||||||
"--rpc-endpoint", "http://"+e.RPC.Addresses()[0],
|
"--rpc-endpoint", "http://"+e.RPC.Addresses()[0],
|
||||||
"--wallet", testcli.ValidatorWallet,
|
"--wallet", testcli.TestWalletMultiPath,
|
||||||
)
|
)
|
||||||
cmd := append(cmdbase, "--address", testcli.ValidatorAddr)
|
cmd := append(cmdbase, "--address", testcli.TestWalletMultiAccount1)
|
||||||
t.Run("excessive parameters", func(t *testing.T) {
|
t.Run("excessive parameters", func(t *testing.T) {
|
||||||
e.RunWithError(t, append(cmd, "--token", "NEO", "gas")...)
|
e.RunWithError(t, append(cmd, "--token", "NEO", "gas")...)
|
||||||
})
|
})
|
||||||
t.Run("NEO", func(t *testing.T) {
|
t.Run("NEO", func(t *testing.T) {
|
||||||
b, index := e.Chain.GetGoverningTokenBalance(testcli.ValidatorHash)
|
b, index := e.Chain.GetGoverningTokenBalance(testcli.TestWalletMultiAccount1Hash)
|
||||||
checkResult := func(t *testing.T) {
|
checkResult := func(t *testing.T) {
|
||||||
e.CheckNextLine(t, "^\\s*Account\\s+"+testcli.ValidatorAddr)
|
e.CheckNextLine(t, "^\\s*Account\\s+"+testcli.TestWalletMultiAccount1)
|
||||||
e.CheckNextLine(t, "^\\s*NEO:\\s+NeoToken \\("+e.Chain.GoverningTokenHash().StringLE()+"\\)")
|
e.CheckNextLine(t, "^\\s*NEO:\\s+NeoToken \\("+e.Chain.GoverningTokenHash().StringLE()+"\\)")
|
||||||
e.CheckNextLine(t, "^\\s*Amount\\s*:\\s*"+b.String()+"$")
|
e.CheckNextLine(t, "^\\s*Amount\\s*:\\s*"+b.String()+"$")
|
||||||
e.CheckNextLine(t, "^\\s*Updated\\s*:\\s*"+strconv.FormatUint(uint64(index), 10))
|
e.CheckNextLine(t, "^\\s*Updated\\s*:\\s*"+strconv.FormatUint(uint64(index), 10))
|
||||||
|
@ -48,65 +63,53 @@ func TestNEP17Balance(t *testing.T) {
|
||||||
})
|
})
|
||||||
t.Run("GAS", func(t *testing.T) {
|
t.Run("GAS", func(t *testing.T) {
|
||||||
e.Run(t, append(cmd, "--token", "GAS")...)
|
e.Run(t, append(cmd, "--token", "GAS")...)
|
||||||
e.CheckNextLine(t, "^\\s*Account\\s+"+testcli.ValidatorAddr)
|
e.CheckNextLine(t, "^\\s*Account\\s+"+testcli.TestWalletMultiAccount1)
|
||||||
e.CheckNextLine(t, "^\\s*GAS:\\s+GasToken \\("+e.Chain.UtilityTokenHash().StringLE()+"\\)")
|
e.CheckNextLine(t, "^\\s*GAS:\\s+GasToken \\("+e.Chain.UtilityTokenHash().StringLE()+"\\)")
|
||||||
b := e.Chain.GetUtilityTokenBalance(testcli.ValidatorHash)
|
b := e.Chain.GetUtilityTokenBalance(testcli.TestWalletMultiAccount1Hash)
|
||||||
e.CheckNextLine(t, "^\\s*Amount\\s*:\\s*"+fixedn.Fixed8(b.Int64()).String()+"$")
|
e.CheckNextLine(t, "^\\s*Amount\\s*:\\s*"+fixedn.Fixed8(b.Int64()).String()+"$")
|
||||||
})
|
})
|
||||||
t.Run("zero balance of known token", func(t *testing.T) {
|
t.Run("zero balance of known token", func(t *testing.T) {
|
||||||
e.Run(t, append(cmdbase, []string{"--token", "NEO"}...)...)
|
e.Run(t, append(cmdbase, []string{"--token", "NEO", "--address", testcli.TestWalletMultiAccount2}...)...)
|
||||||
addr1, err := address.StringToUint160("Nhfg3TbpwogLvDGVvAvqyThbsHgoSUKwtn")
|
e.CheckNextLine(t, "^Account "+testcli.TestWalletMultiAccount2)
|
||||||
require.NoError(t, err)
|
|
||||||
e.CheckNextLine(t, "^Account "+address.Uint160ToString(addr1))
|
|
||||||
e.CheckNextLine(t, "^\\s*NEO:\\s+NeoToken \\("+e.Chain.GoverningTokenHash().StringLE()+"\\)")
|
e.CheckNextLine(t, "^\\s*NEO:\\s+NeoToken \\("+e.Chain.GoverningTokenHash().StringLE()+"\\)")
|
||||||
e.CheckNextLine(t, "^\\s*Amount\\s*:\\s*"+fixedn.Fixed8(0).String()+"$")
|
e.CheckNextLine(t, "^\\s*Amount\\s*:\\s*"+fixedn.Fixed8(0).String()+"$")
|
||||||
e.CheckNextLine(t, "^\\s*Updated:")
|
e.CheckNextLine(t, "^\\s*Updated:")
|
||||||
e.CheckNextLine(t, "^\\s*$")
|
e.CheckEOF(t)
|
||||||
})
|
})
|
||||||
t.Run("all accounts", func(t *testing.T) {
|
t.Run("all accounts", func(t *testing.T) {
|
||||||
e.Run(t, cmdbase...)
|
e.Run(t, cmdbase...)
|
||||||
addr1, err := address.StringToUint160("Nhfg3TbpwogLvDGVvAvqyThbsHgoSUKwtn")
|
|
||||||
require.NoError(t, err)
|
|
||||||
e.CheckNextLine(t, "^Account "+address.Uint160ToString(addr1))
|
|
||||||
e.CheckNextLine(t, "^\\s*GAS:\\s+GasToken \\("+e.Chain.UtilityTokenHash().StringLE()+"\\)")
|
|
||||||
balance := e.Chain.GetUtilityTokenBalance(addr1)
|
|
||||||
e.CheckNextLine(t, "^\\s*Amount\\s*:\\s*"+fixedn.Fixed8(balance.Int64()).String()+"$")
|
|
||||||
e.CheckNextLine(t, "^\\s*Updated:")
|
|
||||||
e.CheckNextLine(t, "^\\s*$")
|
|
||||||
|
|
||||||
addr2, err := address.StringToUint160("NVTiAjNgagDkTr5HTzDmQP9kPwPHN5BgVq")
|
e.CheckNextLine(t, "^Account "+testcli.TestWalletMultiAccount1)
|
||||||
require.NoError(t, err)
|
|
||||||
e.CheckNextLine(t, "^Account "+address.Uint160ToString(addr2))
|
|
||||||
e.CheckNextLine(t, "^\\s*$")
|
|
||||||
|
|
||||||
addr3, err := address.StringToUint160("NfgHwwTi3wHAS8aFAN243C5vGbkYDpqLHP")
|
|
||||||
require.NoError(t, err)
|
|
||||||
e.CheckNextLine(t, "^Account "+address.Uint160ToString(addr3))
|
|
||||||
// The order of assets is undefined.
|
// The order of assets is undefined.
|
||||||
for i := 0; i < 2; i++ {
|
for i := 0; i < 2; i++ {
|
||||||
line := e.GetNextLine(t)
|
line := e.GetNextLine(t)
|
||||||
if strings.Contains(line, "GAS") {
|
if strings.Contains(line, "GAS") {
|
||||||
e.CheckLine(t, line, "^\\s*GAS:\\s+GasToken \\("+e.Chain.UtilityTokenHash().StringLE()+"\\)")
|
e.CheckLine(t, line, "^\\s*GAS:\\s+GasToken \\("+e.Chain.UtilityTokenHash().StringLE()+"\\)")
|
||||||
balance = e.Chain.GetUtilityTokenBalance(addr3)
|
balance := e.Chain.GetUtilityTokenBalance(testcli.TestWalletMultiAccount1Hash)
|
||||||
e.CheckNextLine(t, "^\\s*Amount\\s*:\\s*"+fixedn.Fixed8(balance.Int64()).String()+"$")
|
e.CheckNextLine(t, "^\\s*Amount\\s*:\\s*"+fixedn.Fixed8(balance.Int64()).String()+"$")
|
||||||
e.CheckNextLine(t, "^\\s*Updated:")
|
e.CheckNextLine(t, "^\\s*Updated:")
|
||||||
} else {
|
} else {
|
||||||
balance, index := e.Chain.GetGoverningTokenBalance(testcli.ValidatorHash)
|
balance, index := e.Chain.GetGoverningTokenBalance(testcli.TestWalletMultiAccount1Hash)
|
||||||
e.CheckLine(t, line, "^\\s*NEO:\\s+NeoToken \\("+e.Chain.GoverningTokenHash().StringLE()+"\\)")
|
e.CheckLine(t, line, "^\\s*NEO:\\s+NeoToken \\("+e.Chain.GoverningTokenHash().StringLE()+"\\)")
|
||||||
e.CheckNextLine(t, "^\\s*Amount\\s*:\\s*"+balance.String()+"$")
|
e.CheckNextLine(t, "^\\s*Amount\\s*:\\s*"+balance.String()+"$")
|
||||||
e.CheckNextLine(t, "^\\s*Updated\\s*:\\s*"+strconv.FormatUint(uint64(index), 10))
|
e.CheckNextLine(t, "^\\s*Updated\\s*:\\s*"+strconv.FormatUint(uint64(index), 10))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
e.CheckNextLine(t, "^\\s*$")
|
e.CheckNextLine(t, "^\\s*$")
|
||||||
addr4, err := address.StringToUint160("NiFxRcC5Anz9pmqQyMHh5vamBUZDbRRRzA") // deployed verify.go contract
|
|
||||||
require.NoError(t, err)
|
e.CheckNextLine(t, "^Account "+testcli.TestWalletMultiAccount2)
|
||||||
e.CheckNextLine(t, "^Account "+address.Uint160ToString(addr4))
|
e.CheckNextLine(t, "^\\s*$")
|
||||||
|
|
||||||
|
e.CheckNextLine(t, "^Account "+testcli.TestWalletMultiAccount3)
|
||||||
|
e.CheckNextLine(t, "^\\s*GAS:\\s+GasToken \\("+e.Chain.UtilityTokenHash().StringLE()+"\\)")
|
||||||
|
balance := e.Chain.GetUtilityTokenBalance(testcli.TestWalletMultiAccount3Hash)
|
||||||
|
e.CheckNextLine(t, "^\\s*Amount\\s*:\\s*"+fixedn.Fixed8(balance.Int64()).String()+"$")
|
||||||
|
e.CheckNextLine(t, "^\\s*Updated:")
|
||||||
e.CheckEOF(t)
|
e.CheckEOF(t)
|
||||||
})
|
})
|
||||||
t.Run("Bad token", func(t *testing.T) {
|
t.Run("Bad token", func(t *testing.T) {
|
||||||
e.Run(t, append(cmd, "--token", "kek")...)
|
e.Run(t, append(cmd, "--token", "kek")...)
|
||||||
e.CheckNextLine(t, "^\\s*Account\\s+"+testcli.ValidatorAddr)
|
e.CheckNextLine(t, "^\\s*Account\\s+"+testcli.TestWalletMultiAccount1)
|
||||||
e.CheckNextLine(t, `^\s*Can't find data for "kek" token\s*`)
|
e.CheckNextLine(t, `^\s*Can't find data for "kek" token\s*`)
|
||||||
e.CheckEOF(t)
|
e.CheckEOF(t)
|
||||||
})
|
})
|
||||||
|
@ -223,12 +226,19 @@ func TestNEP17Transfer(t *testing.T) {
|
||||||
"neo-go", "wallet", "nep17", "transfer",
|
"neo-go", "wallet", "nep17", "transfer",
|
||||||
"--rpc-endpoint", "http://" + e.RPC.Addresses()[0],
|
"--rpc-endpoint", "http://" + e.RPC.Addresses()[0],
|
||||||
"--wallet", testcli.ValidatorWallet,
|
"--wallet", testcli.ValidatorWallet,
|
||||||
"--to", address.Uint160ToString(e.Chain.GetNotaryContractScriptHash()),
|
|
||||||
"--token", "GAS",
|
"--token", "GAS",
|
||||||
"--amount", "1",
|
"--amount", "1",
|
||||||
"--from", testcli.ValidatorAddr,
|
|
||||||
"--force",
|
"--force",
|
||||||
"[", testcli.ValidatorAddr, strconv.Itoa(int(validTil)), "]"}
|
"--from", testcli.ValidatorAddr}
|
||||||
|
|
||||||
|
t.Run("with await", func(t *testing.T) {
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.Run(t, append(cmd, "--to", nftOwnerAddr, "--await")...)
|
||||||
|
e.CheckAwaitableTxPersisted(t)
|
||||||
|
})
|
||||||
|
|
||||||
|
cmd = append(cmd, "--to", address.Uint160ToString(e.Chain.GetNotaryContractScriptHash()),
|
||||||
|
"[", testcli.ValidatorAddr, strconv.Itoa(int(validTil)), "]")
|
||||||
|
|
||||||
t.Run("with data", func(t *testing.T) {
|
t.Run("with data", func(t *testing.T) {
|
||||||
e.In.WriteString("one\r")
|
e.In.WriteString("one\r")
|
||||||
|
|
|
@ -11,33 +11,56 @@ import (
|
||||||
"os"
|
"os"
|
||||||
"runtime"
|
"runtime"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/nspcc-dev/neo-go/cli/flags"
|
||||||
|
"github.com/nspcc-dev/neo-go/cli/input"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/config"
|
"github.com/nspcc-dev/neo-go/pkg/config"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/config/netmode"
|
"github.com/nspcc-dev/neo-go/pkg/config/netmode"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/io"
|
"github.com/nspcc-dev/neo-go/pkg/io"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/rpcclient"
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/actor"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/rpcclient/invoker"
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/invoker"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/util"
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/wallet"
|
||||||
"github.com/urfave/cli"
|
"github.com/urfave/cli"
|
||||||
"go.uber.org/zap"
|
"go.uber.org/zap"
|
||||||
"go.uber.org/zap/zapcore"
|
"go.uber.org/zap/zapcore"
|
||||||
|
"golang.org/x/term"
|
||||||
|
"gopkg.in/yaml.v3"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
// DefaultTimeout is the default timeout used for RPC requests.
|
// DefaultTimeout is the default timeout used for RPC requests.
|
||||||
const DefaultTimeout = 10 * time.Second
|
DefaultTimeout = 10 * time.Second
|
||||||
|
// DefaultAwaitableTimeout is the default timeout used for RPC requests that
|
||||||
|
// require transaction awaiting. It is set to the approximate time of three
|
||||||
|
// Neo N3 mainnet blocks accepting.
|
||||||
|
DefaultAwaitableTimeout = 3 * 15 * time.Second
|
||||||
|
)
|
||||||
|
|
||||||
// RPCEndpointFlag is a long flag name for an RPC endpoint. It can be used to
|
// RPCEndpointFlag is a long flag name for an RPC endpoint. It can be used to
|
||||||
// check for flag presence in the context.
|
// check for flag presence in the context.
|
||||||
const RPCEndpointFlag = "rpc-endpoint"
|
const RPCEndpointFlag = "rpc-endpoint"
|
||||||
|
|
||||||
|
// Wallet is a set of flags used for wallet operations.
|
||||||
|
var Wallet = []cli.Flag{cli.StringFlag{
|
||||||
|
Name: "wallet, w",
|
||||||
|
Usage: "wallet to use to get the key for transaction signing; conflicts with --wallet-config flag",
|
||||||
|
}, cli.StringFlag{
|
||||||
|
Name: "wallet-config",
|
||||||
|
Usage: "path to wallet config to use to get the key for transaction signing; conflicts with --wallet flag"},
|
||||||
|
}
|
||||||
|
|
||||||
// Network is a set of flags for choosing the network to operate on
|
// Network is a set of flags for choosing the network to operate on
|
||||||
// (privnet/mainnet/testnet).
|
// (privnet/mainnet/testnet).
|
||||||
var Network = []cli.Flag{
|
var Network = []cli.Flag{
|
||||||
cli.BoolFlag{Name: "privnet, p", Usage: "use private network configuration"},
|
cli.BoolFlag{Name: "privnet, p", Usage: "use private network configuration (if --config-file option is not specified)"},
|
||||||
cli.BoolFlag{Name: "mainnet, m", Usage: "use mainnet network configuration"},
|
cli.BoolFlag{Name: "mainnet, m", Usage: "use mainnet network configuration (if --config-file option is not specified)"},
|
||||||
cli.BoolFlag{Name: "testnet, t", Usage: "use testnet network configuration"},
|
cli.BoolFlag{Name: "testnet, t", Usage: "use testnet network configuration (if --config-file option is not specified)"},
|
||||||
cli.BoolFlag{Name: "unittest", Hidden: true},
|
cli.BoolFlag{Name: "unittest", Hidden: true},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -63,7 +86,21 @@ var Historic = cli.StringFlag{
|
||||||
// Config is a flag for commands that use node configuration.
|
// Config is a flag for commands that use node configuration.
|
||||||
var Config = cli.StringFlag{
|
var Config = cli.StringFlag{
|
||||||
Name: "config-path",
|
Name: "config-path",
|
||||||
Usage: "path to directory with configuration files",
|
Usage: "path to directory with per-network configuration files (may be overridden by --config-file option for the configuration file)",
|
||||||
|
}
|
||||||
|
|
||||||
|
// ConfigFile is a flag for commands that use node configuration and provide
|
||||||
|
// path to the specific config file instead of config path.
|
||||||
|
var ConfigFile = cli.StringFlag{
|
||||||
|
Name: "config-file",
|
||||||
|
Usage: "path to the node configuration file (overrides --config-path option)",
|
||||||
|
}
|
||||||
|
|
||||||
|
// RelativePath is a flag for commands that use node configuration and provide
|
||||||
|
// a prefix to all relative paths in config files.
|
||||||
|
var RelativePath = cli.StringFlag{
|
||||||
|
Name: "relative-path",
|
||||||
|
Usage: "a prefix to all relative paths in the node configuration file",
|
||||||
}
|
}
|
||||||
|
|
||||||
// Debug is a flag for commands that allow node in debug mode usage.
|
// Debug is a flag for commands that allow node in debug mode usage.
|
||||||
|
@ -74,6 +111,8 @@ var Debug = cli.BoolFlag{
|
||||||
|
|
||||||
var errNoEndpoint = errors.New("no RPC endpoint specified, use option '--" + RPCEndpointFlag + "' or '-r'")
|
var errNoEndpoint = errors.New("no RPC endpoint specified, use option '--" + RPCEndpointFlag + "' or '-r'")
|
||||||
var errInvalidHistoric = errors.New("invalid 'historic' parameter, neither a block number, nor a block/state hash")
|
var errInvalidHistoric = errors.New("invalid 'historic' parameter, neither a block number, nor a block/state hash")
|
||||||
|
var errNoWallet = errors.New("no wallet parameter found, specify it with the '--wallet' or '-w' flag or specify wallet config file with the '--wallet-config' flag")
|
||||||
|
var errConflictingWalletFlags = errors.New("--wallet flag conflicts with --wallet-config flag, please, provide one of them to specify wallet location")
|
||||||
|
|
||||||
// GetNetwork examines Context's flags and returns the appropriate network. It
|
// GetNetwork examines Context's flags and returns the appropriate network. It
|
||||||
// defaults to PrivNet if no flags are given.
|
// defaults to PrivNet if no flags are given.
|
||||||
|
@ -97,6 +136,9 @@ func GetTimeoutContext(ctx *cli.Context) (context.Context, func()) {
|
||||||
if dur == 0 {
|
if dur == 0 {
|
||||||
dur = DefaultTimeout
|
dur = DefaultTimeout
|
||||||
}
|
}
|
||||||
|
if !ctx.IsSet("timeout") && ctx.Bool("await") {
|
||||||
|
dur = DefaultAwaitableTimeout
|
||||||
|
}
|
||||||
return context.WithTimeout(context.Background(), dur)
|
return context.WithTimeout(context.Background(), dur)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -152,11 +194,18 @@ func GetRPCWithInvoker(gctx context.Context, ctx *cli.Context, signers []transac
|
||||||
// GetConfigFromContext looks at the path and the mode flags in the given config and
|
// GetConfigFromContext looks at the path and the mode flags in the given config and
|
||||||
// returns an appropriate config.
|
// returns an appropriate config.
|
||||||
func GetConfigFromContext(ctx *cli.Context) (config.Config, error) {
|
func GetConfigFromContext(ctx *cli.Context) (config.Config, error) {
|
||||||
configPath := "./config"
|
var (
|
||||||
|
configFile = ctx.String("config-file")
|
||||||
|
relativePath = ctx.String("relative-path")
|
||||||
|
)
|
||||||
|
if len(configFile) != 0 {
|
||||||
|
return config.LoadFile(configFile, relativePath)
|
||||||
|
}
|
||||||
|
var configPath = "./config"
|
||||||
if argCp := ctx.String("config-path"); argCp != "" {
|
if argCp := ctx.String("config-path"); argCp != "" {
|
||||||
configPath = argCp
|
configPath = argCp
|
||||||
}
|
}
|
||||||
return config.Load(configPath, GetNetwork(ctx))
|
return config.Load(configPath, GetNetwork(ctx), relativePath)
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
@ -171,6 +220,7 @@ var (
|
||||||
// If logPath is configured -- function creates a dir and a file for logging.
|
// If logPath is configured -- function creates a dir and a file for logging.
|
||||||
// If logPath is configured on Windows -- function returns closer to be
|
// If logPath is configured on Windows -- function returns closer to be
|
||||||
// able to close sink for the opened log output file.
|
// able to close sink for the opened log output file.
|
||||||
|
// If the program is run in TTY then logger adds timestamp to its entries.
|
||||||
func HandleLoggingParams(debug bool, cfg config.ApplicationConfiguration) (*zap.Logger, *zap.AtomicLevel, func() error, error) {
|
func HandleLoggingParams(debug bool, cfg config.ApplicationConfiguration) (*zap.Logger, *zap.AtomicLevel, func() error, error) {
|
||||||
var (
|
var (
|
||||||
level = zapcore.InfoLevel
|
level = zapcore.InfoLevel
|
||||||
|
@ -191,7 +241,11 @@ func HandleLoggingParams(debug bool, cfg config.ApplicationConfiguration) (*zap.
|
||||||
cc.DisableStacktrace = true
|
cc.DisableStacktrace = true
|
||||||
cc.EncoderConfig.EncodeDuration = zapcore.StringDurationEncoder
|
cc.EncoderConfig.EncodeDuration = zapcore.StringDurationEncoder
|
||||||
cc.EncoderConfig.EncodeLevel = zapcore.CapitalLevelEncoder
|
cc.EncoderConfig.EncodeLevel = zapcore.CapitalLevelEncoder
|
||||||
|
if term.IsTerminal(int(os.Stdout.Fd())) {
|
||||||
cc.EncoderConfig.EncodeTime = zapcore.ISO8601TimeEncoder
|
cc.EncoderConfig.EncodeTime = zapcore.ISO8601TimeEncoder
|
||||||
|
} else {
|
||||||
|
cc.EncoderConfig.EncodeTime = func(t time.Time, encoder zapcore.PrimitiveArrayEncoder) {}
|
||||||
|
}
|
||||||
cc.Encoding = "console"
|
cc.Encoding = "console"
|
||||||
cc.Level = zap.NewAtomicLevelAt(level)
|
cc.Level = zap.NewAtomicLevelAt(level)
|
||||||
cc.Sampling = nil
|
cc.Sampling = nil
|
||||||
|
@ -249,3 +303,108 @@ func HandleLoggingParams(debug bool, cfg config.ApplicationConfiguration) (*zap.
|
||||||
log, err := cc.Build()
|
log, err := cc.Build()
|
||||||
return log, &cc.Level, _winfileSinkCloser, err
|
return log, &cc.Level, _winfileSinkCloser, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GetRPCWithActor returns an RPC client instance and Actor instance for the given context.
|
||||||
|
func GetRPCWithActor(gctx context.Context, ctx *cli.Context, signers []actor.SignerAccount) (*rpcclient.Client, *actor.Actor, cli.ExitCoder) {
|
||||||
|
c, err := GetRPCClient(gctx, ctx)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
a, actorErr := actor.New(c, signers)
|
||||||
|
if actorErr != nil {
|
||||||
|
c.Close()
|
||||||
|
return nil, nil, cli.NewExitError(fmt.Errorf("failed to create Actor: %w", actorErr), 1)
|
||||||
|
}
|
||||||
|
return c, a, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetAccFromContext returns account and wallet from context. If address is not set, default address is used.
|
||||||
|
func GetAccFromContext(ctx *cli.Context) (*wallet.Account, *wallet.Wallet, error) {
|
||||||
|
var addr util.Uint160
|
||||||
|
|
||||||
|
wPath := ctx.String("wallet")
|
||||||
|
walletConfigPath := ctx.String("wallet-config")
|
||||||
|
if len(wPath) != 0 && len(walletConfigPath) != 0 {
|
||||||
|
return nil, nil, errConflictingWalletFlags
|
||||||
|
}
|
||||||
|
if len(wPath) == 0 && len(walletConfigPath) == 0 {
|
||||||
|
return nil, nil, errNoWallet
|
||||||
|
}
|
||||||
|
var pass *string
|
||||||
|
if len(walletConfigPath) != 0 {
|
||||||
|
cfg, err := ReadWalletConfig(walletConfigPath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
wPath = cfg.Path
|
||||||
|
pass = &cfg.Password
|
||||||
|
}
|
||||||
|
|
||||||
|
wall, err := wallet.NewWalletFromFile(wPath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
addrFlag := ctx.Generic("address").(*flags.Address)
|
||||||
|
if addrFlag.IsSet {
|
||||||
|
addr = addrFlag.Uint160()
|
||||||
|
} else {
|
||||||
|
addr = wall.GetChangeAddress()
|
||||||
|
if addr.Equals(util.Uint160{}) {
|
||||||
|
return nil, wall, errors.New("can't get default address")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
acc, err := GetUnlockedAccount(wall, addr, pass)
|
||||||
|
return acc, wall, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetUnlockedAccount returns account from wallet, address and uses pass to unlock specified account if given.
|
||||||
|
// If the password is not given, then it is requested from user.
|
||||||
|
func GetUnlockedAccount(wall *wallet.Wallet, addr util.Uint160, pass *string) (*wallet.Account, error) {
|
||||||
|
acc := wall.GetAccount(addr)
|
||||||
|
if acc == nil {
|
||||||
|
return nil, fmt.Errorf("wallet contains no account for '%s'", address.Uint160ToString(addr))
|
||||||
|
}
|
||||||
|
|
||||||
|
if acc.CanSign() || acc.EncryptedWIF == "" {
|
||||||
|
return acc, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if pass == nil {
|
||||||
|
rawPass, err := input.ReadPassword(
|
||||||
|
fmt.Sprintf("Enter account %s password > ", address.Uint160ToString(addr)))
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("Error reading password: %w", err)
|
||||||
|
}
|
||||||
|
trimmed := strings.TrimRight(string(rawPass), "\n")
|
||||||
|
pass = &trimmed
|
||||||
|
}
|
||||||
|
err := acc.Decrypt(*pass, wall.Scrypt)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return acc, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ReadWalletConfig reads wallet config from the given path.
|
||||||
|
func ReadWalletConfig(configPath string) (*config.Wallet, error) {
|
||||||
|
file, err := os.Open(configPath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
configData, err := os.ReadFile(configPath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("unable to read wallet config: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cfg := &config.Wallet{}
|
||||||
|
|
||||||
|
err = yaml.Unmarshal(configData, &cfg)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to unmarshal wallet config YAML: %w", err)
|
||||||
|
}
|
||||||
|
return cfg, nil
|
||||||
|
}
|
||||||
|
|
|
@ -3,7 +3,6 @@ package query
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/base64"
|
"encoding/base64"
|
||||||
"encoding/hex"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"sort"
|
"sort"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
@ -111,7 +110,10 @@ func queryTx(ctx *cli.Context) error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
DumpApplicationLog(ctx, res, &txOut.Transaction, &txOut.TransactionMetadata, ctx.Bool("verbose"))
|
err = DumpApplicationLog(ctx, res, &txOut.Transaction, &txOut.TransactionMetadata, ctx.Bool("verbose"))
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(err, 1)
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -120,47 +122,49 @@ func DumpApplicationLog(
|
||||||
res *result.ApplicationLog,
|
res *result.ApplicationLog,
|
||||||
tx *transaction.Transaction,
|
tx *transaction.Transaction,
|
||||||
txMeta *result.TransactionMetadata,
|
txMeta *result.TransactionMetadata,
|
||||||
verbose bool) {
|
verbose bool) error {
|
||||||
buf := bytes.NewBuffer(nil)
|
var buf []byte
|
||||||
|
|
||||||
// Ignore the errors below because `Write` to buffer doesn't return error.
|
buf = fmt.Appendf(buf, "Hash:\t%s\n", tx.Hash().StringLE())
|
||||||
tw := tabwriter.NewWriter(buf, 0, 4, 4, '\t', 0)
|
buf = fmt.Appendf(buf, "OnChain:\t%t\n", res != nil)
|
||||||
_, _ = tw.Write([]byte("Hash:\t" + tx.Hash().StringLE() + "\n"))
|
|
||||||
_, _ = tw.Write([]byte(fmt.Sprintf("OnChain:\t%t\n", res != nil)))
|
|
||||||
if res == nil {
|
if res == nil {
|
||||||
_, _ = tw.Write([]byte("ValidUntil:\t" + strconv.FormatUint(uint64(tx.ValidUntilBlock), 10) + "\n"))
|
buf = fmt.Appendf(buf, "ValidUntil:\t%s\n", strconv.FormatUint(uint64(tx.ValidUntilBlock), 10))
|
||||||
} else {
|
} else {
|
||||||
if txMeta != nil {
|
if txMeta != nil {
|
||||||
_, _ = tw.Write([]byte("BlockHash:\t" + txMeta.Blockhash.StringLE() + "\n"))
|
buf = fmt.Appendf(buf, "BlockHash:\t%s\n", txMeta.Blockhash.StringLE())
|
||||||
}
|
}
|
||||||
if len(res.Executions) != 1 {
|
if len(res.Executions) != 1 {
|
||||||
_, _ = tw.Write([]byte("Success:\tunknown (no execution data)\n"))
|
buf = fmt.Appendf(buf, "Success:\tunknown (no execution data)\n")
|
||||||
} else {
|
} else {
|
||||||
_, _ = tw.Write([]byte(fmt.Sprintf("Success:\t%t\n", res.Executions[0].VMState == vmstate.Halt)))
|
buf = fmt.Appendf(buf, "Success:\t%t\n", res.Executions[0].VMState == vmstate.Halt)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if verbose {
|
if verbose {
|
||||||
for _, sig := range tx.Signers {
|
for _, sig := range tx.Signers {
|
||||||
_, _ = tw.Write([]byte(fmt.Sprintf("Signer:\t%s (%s)",
|
buf = fmt.Appendf(buf, "Signer:\t%s (%s)\n", address.Uint160ToString(sig.Account), sig.Scopes)
|
||||||
address.Uint160ToString(sig.Account),
|
|
||||||
sig.Scopes) + "\n"))
|
|
||||||
}
|
}
|
||||||
_, _ = tw.Write([]byte("SystemFee:\t" + fixedn.Fixed8(tx.SystemFee).String() + " GAS\n"))
|
buf = fmt.Appendf(buf, "SystemFee:\t%s GAS\n", fixedn.Fixed8(tx.SystemFee).String())
|
||||||
_, _ = tw.Write([]byte("NetworkFee:\t" + fixedn.Fixed8(tx.NetworkFee).String() + " GAS\n"))
|
buf = fmt.Appendf(buf, "NetworkFee:\t%s GAS\n", fixedn.Fixed8(tx.NetworkFee).String())
|
||||||
_, _ = tw.Write([]byte("Script:\t" + base64.StdEncoding.EncodeToString(tx.Script) + "\n"))
|
buf = fmt.Appendf(buf, "Script:\t%s\n", base64.StdEncoding.EncodeToString(tx.Script))
|
||||||
v := vm.New()
|
v := vm.New()
|
||||||
v.Load(tx.Script)
|
v.Load(tx.Script)
|
||||||
v.PrintOps(tw)
|
opts := bytes.NewBuffer(nil)
|
||||||
|
v.PrintOps(opts)
|
||||||
|
buf = append(buf, opts.Bytes()...)
|
||||||
if res != nil {
|
if res != nil {
|
||||||
for _, e := range res.Executions {
|
for _, e := range res.Executions {
|
||||||
if e.VMState != vmstate.Halt {
|
if e.VMState != vmstate.Halt {
|
||||||
_, _ = tw.Write([]byte("Exception:\t" + e.FaultException + "\n"))
|
buf = fmt.Appendf(buf, "Exception:\t%s\n", e.FaultException)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
_ = tw.Flush()
|
tw := tabwriter.NewWriter(ctx.App.Writer, 0, 4, 4, '\t', 0)
|
||||||
fmt.Fprint(ctx.App.Writer, buf.String())
|
_, err := tw.Write(buf)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return tw.Flush()
|
||||||
}
|
}
|
||||||
|
|
||||||
func queryCandidates(ctx *cli.Context) error {
|
func queryCandidates(ctx *cli.Context) error {
|
||||||
|
@ -196,15 +200,17 @@ func queryCandidates(ctx *cli.Context) error {
|
||||||
}
|
}
|
||||||
return vals[i].PublicKey.Cmp(&vals[j].PublicKey) == -1
|
return vals[i].PublicKey.Cmp(&vals[j].PublicKey) == -1
|
||||||
})
|
})
|
||||||
buf := bytes.NewBuffer(nil)
|
var res []byte
|
||||||
tw := tabwriter.NewWriter(buf, 0, 2, 2, ' ', 0)
|
res = fmt.Appendf(res, "Key\tVotes\tCommittee\tConsensus\n")
|
||||||
_, _ = tw.Write([]byte("Key\tVotes\tCommittee\tConsensus\n"))
|
|
||||||
for _, val := range vals {
|
for _, val := range vals {
|
||||||
_, _ = tw.Write([]byte(fmt.Sprintf("%s\t%d\t%t\t%t\n", hex.EncodeToString(val.PublicKey.Bytes()), val.Votes, comm.Contains(&val.PublicKey), val.Active)))
|
res = fmt.Appendf(res, "%s\t%d\t%t\t%t\n", val.PublicKey.StringCompressed(), val.Votes, comm.Contains(&val.PublicKey), val.Active)
|
||||||
}
|
}
|
||||||
_ = tw.Flush()
|
tw := tabwriter.NewWriter(ctx.App.Writer, 0, 2, 2, ' ', 0)
|
||||||
fmt.Fprint(ctx.App.Writer, buf.String())
|
_, err = tw.Write(res)
|
||||||
return nil
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return tw.Flush()
|
||||||
}
|
}
|
||||||
|
|
||||||
func queryCommittee(ctx *cli.Context) error {
|
func queryCommittee(ctx *cli.Context) error {
|
||||||
|
@ -228,7 +234,7 @@ func queryCommittee(ctx *cli.Context) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, k := range comm {
|
for _, k := range comm {
|
||||||
fmt.Fprintln(ctx.App.Writer, hex.EncodeToString(k.Bytes()))
|
fmt.Fprintln(ctx.App.Writer, k.StringCompressed())
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -299,7 +305,7 @@ func queryVoter(ctx *cli.Context) error {
|
||||||
}
|
}
|
||||||
voted := "null"
|
voted := "null"
|
||||||
if st.VoteTo != nil {
|
if st.VoteTo != nil {
|
||||||
voted = fmt.Sprintf("%s (%s)", hex.EncodeToString(st.VoteTo.Bytes()), address.Uint160ToString(st.VoteTo.GetScriptHash()))
|
voted = fmt.Sprintf("%s (%s)", st.VoteTo.StringCompressed(), address.Uint160ToString(st.VoteTo.GetScriptHash()))
|
||||||
}
|
}
|
||||||
fmt.Fprintf(ctx.App.Writer, "\tVoted: %s\n", voted)
|
fmt.Fprintf(ctx.App.Writer, "\tVoted: %s\n", voted)
|
||||||
fmt.Fprintf(ctx.App.Writer, "\tAmount : %s\n", fixedn.ToString(&st.Balance, int(dec)))
|
fmt.Fprintf(ctx.App.Writer, "\tAmount : %s\n", fixedn.ToString(&st.Balance, int(dec)))
|
||||||
|
|
|
@ -13,6 +13,9 @@ import (
|
||||||
"gopkg.in/yaml.v3"
|
"gopkg.in/yaml.v3"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// generated via `go run ./scripts/gendump/main.go --out ./cli/server/testdata/chain50x2.acc --blocks 50 --txs 2`.
|
||||||
|
const inDump = "./testdata/chain50x2.acc"
|
||||||
|
|
||||||
func TestDBRestoreDump(t *testing.T) {
|
func TestDBRestoreDump(t *testing.T) {
|
||||||
tmpDir := t.TempDir()
|
tmpDir := t.TempDir()
|
||||||
|
|
||||||
|
@ -32,8 +35,6 @@ func TestDBRestoreDump(t *testing.T) {
|
||||||
cfgPath := filepath.Join(tmpDir, "protocol.unit_testnet.yml")
|
cfgPath := filepath.Join(tmpDir, "protocol.unit_testnet.yml")
|
||||||
require.NoError(t, os.WriteFile(cfgPath, out, os.ModePerm))
|
require.NoError(t, os.WriteFile(cfgPath, out, os.ModePerm))
|
||||||
|
|
||||||
// generated via `go run ./scripts/gendump/main.go --out ./cli/server/testdata/chain50x2.acc --blocks 50 --txs 2`
|
|
||||||
const inDump = "./testdata/chain50x2.acc"
|
|
||||||
e := testcli.NewExecutor(t, false)
|
e := testcli.NewExecutor(t, false)
|
||||||
|
|
||||||
stateDump := filepath.Join(tmpDir, "neogo.teststate")
|
stateDump := filepath.Join(tmpDir, "neogo.teststate")
|
||||||
|
@ -111,3 +112,47 @@ func TestDBRestoreDump(t *testing.T) {
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.Equal(t, d1, d2, "dumps differ")
|
require.Equal(t, d1, d2, "dumps differ")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestDBDumpRestoreIncremental(t *testing.T) {
|
||||||
|
tmpDir := t.TempDir()
|
||||||
|
chainPath := filepath.Join(tmpDir, "neogotestchain")
|
||||||
|
nonincDump := filepath.Join(tmpDir, "nonincDump.acc")
|
||||||
|
incDump := filepath.Join(tmpDir, "incDump.acc")
|
||||||
|
|
||||||
|
cfg, err := config.LoadFile(filepath.Join("..", "..", "config", "protocol.unit_testnet.yml"))
|
||||||
|
require.NoError(t, err, "could not load config")
|
||||||
|
cfg.ApplicationConfiguration.DBConfiguration.Type = dbconfig.LevelDB
|
||||||
|
cfg.ApplicationConfiguration.DBConfiguration.LevelDBOptions.DataDirectoryPath = chainPath
|
||||||
|
out, err := yaml.Marshal(cfg)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
cfgPath := filepath.Join(tmpDir, "protocol.unit_testnet.yml")
|
||||||
|
require.NoError(t, os.WriteFile(cfgPath, out, os.ModePerm))
|
||||||
|
|
||||||
|
e := testcli.NewExecutor(t, false)
|
||||||
|
|
||||||
|
// Create DB from dump.
|
||||||
|
e.Run(t, "neo-go", "db", "restore", "--unittest", "--config-path", tmpDir, "--in", inDump)
|
||||||
|
|
||||||
|
// Create two dumps: non-incremental and incremental.
|
||||||
|
dumpBaseArgs := []string{"neo-go", "db", "dump", "--unittest",
|
||||||
|
"--config-path", tmpDir}
|
||||||
|
|
||||||
|
// Dump first 15 blocks to a non-incremental dump.
|
||||||
|
e.Run(t, append(dumpBaseArgs, "--out", nonincDump, "--count", "15")...)
|
||||||
|
|
||||||
|
// Dump second 15 blocks to an incremental dump.
|
||||||
|
e.Run(t, append(dumpBaseArgs, "--out", incDump, "--start", "15", "--count", "15")...)
|
||||||
|
|
||||||
|
// Clean the DB.
|
||||||
|
require.NoError(t, os.RemoveAll(chainPath))
|
||||||
|
|
||||||
|
// Restore chain from two dumps.
|
||||||
|
restoreBaseArgs := []string{"neo-go", "db", "restore", "--unittest", "--config-path", tmpDir}
|
||||||
|
|
||||||
|
// Restore first 15 blocks from non-incremental dump.
|
||||||
|
e.Run(t, append(restoreBaseArgs, "--in", nonincDump)...)
|
||||||
|
|
||||||
|
// Restore second 15 blocks from incremental dump.
|
||||||
|
e.Run(t, append(restoreBaseArgs, "--in", incDump, "-n", "--count", "15")...)
|
||||||
|
}
|
||||||
|
|
|
@ -77,7 +77,8 @@ func TestServerStart(t *testing.T) {
|
||||||
})
|
})
|
||||||
t.Run("invalid consensus config", func(t *testing.T) {
|
t.Run("invalid consensus config", func(t *testing.T) {
|
||||||
saveCfg(t, func(cfg *config.Config) {
|
saveCfg(t, func(cfg *config.Config) {
|
||||||
cfg.ApplicationConfiguration.UnlockWallet.Path = "bad_consensus_wallet.json"
|
cfg.ApplicationConfiguration.Consensus.Enabled = true
|
||||||
|
cfg.ApplicationConfiguration.Consensus.UnlockWallet.Path = "bad_consensus_wallet.json"
|
||||||
})
|
})
|
||||||
e.RunWithError(t, baseCmd...)
|
e.RunWithError(t, baseCmd...)
|
||||||
})
|
})
|
||||||
|
|
|
@ -34,7 +34,7 @@ import (
|
||||||
|
|
||||||
// NewCommands returns 'node' command.
|
// NewCommands returns 'node' command.
|
||||||
func NewCommands() []cli.Command {
|
func NewCommands() []cli.Command {
|
||||||
cfgFlags := []cli.Flag{options.Config}
|
cfgFlags := []cli.Flag{options.Config, options.ConfigFile, options.RelativePath}
|
||||||
cfgFlags = append(cfgFlags, options.Network...)
|
cfgFlags = append(cfgFlags, options.Network...)
|
||||||
var cfgWithCountFlags = make([]cli.Flag, len(cfgFlags))
|
var cfgWithCountFlags = make([]cli.Flag, len(cfgFlags))
|
||||||
copy(cfgWithCountFlags, cfgFlags)
|
copy(cfgWithCountFlags, cfgFlags)
|
||||||
|
@ -85,7 +85,7 @@ func NewCommands() []cli.Command {
|
||||||
{
|
{
|
||||||
Name: "node",
|
Name: "node",
|
||||||
Usage: "start a NeoGo node",
|
Usage: "start a NeoGo node",
|
||||||
UsageText: "neo-go node [--config-path path] [-d] [-p/-m/-t]",
|
UsageText: "neo-go node [--config-path path] [-d] [-p/-m/-t] [--config-file file]",
|
||||||
Action: startServer,
|
Action: startServer,
|
||||||
Flags: cfgFlags,
|
Flags: cfgFlags,
|
||||||
},
|
},
|
||||||
|
@ -96,21 +96,21 @@ func NewCommands() []cli.Command {
|
||||||
{
|
{
|
||||||
Name: "dump",
|
Name: "dump",
|
||||||
Usage: "dump blocks (starting with block #1) to the file",
|
Usage: "dump blocks (starting with block #1) to the file",
|
||||||
UsageText: "neo-go db dump -o file [-s start] [-c count] [--config-path path] [-p/-m/-t]",
|
UsageText: "neo-go db dump -o file [-s start] [-c count] [--config-path path] [-p/-m/-t] [--config-file file]",
|
||||||
Action: dumpDB,
|
Action: dumpDB,
|
||||||
Flags: cfgCountOutFlags,
|
Flags: cfgCountOutFlags,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "restore",
|
Name: "restore",
|
||||||
Usage: "restore blocks from the file",
|
Usage: "restore blocks from the file",
|
||||||
UsageText: "neo-go db restore -i file [--dump] [-n] [-c count] [--config-path path] [-p/-m/-t]",
|
UsageText: "neo-go db restore -i file [--dump] [-n] [-c count] [--config-path path] [-p/-m/-t] [--config-file file]",
|
||||||
Action: restoreDB,
|
Action: restoreDB,
|
||||||
Flags: cfgCountInFlags,
|
Flags: cfgCountInFlags,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "reset",
|
Name: "reset",
|
||||||
Usage: "reset database to the previous state",
|
Usage: "reset database to the previous state",
|
||||||
UsageText: "neo-go db reset --height height [--config-path path] [-p/-m/-t]",
|
UsageText: "neo-go db reset --height height [--config-path path] [-p/-m/-t] [--config-file file]",
|
||||||
Action: resetDB,
|
Action: resetDB,
|
||||||
Flags: cfgHeightFlags,
|
Flags: cfgHeightFlags,
|
||||||
},
|
},
|
||||||
|
@ -136,7 +136,6 @@ func initBCWithMetrics(cfg config.Config, log *zap.Logger) (*core.Blockchain, *m
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, nil, cli.NewExitError(err, 1)
|
return nil, nil, nil, cli.NewExitError(err, 1)
|
||||||
}
|
}
|
||||||
configureAddresses(&cfg.ApplicationConfiguration)
|
|
||||||
prometheus := metrics.NewPrometheusService(cfg.ApplicationConfiguration.Prometheus, log)
|
prometheus := metrics.NewPrometheusService(cfg.ApplicationConfiguration.Prometheus, log)
|
||||||
pprof := metrics.NewPprofService(cfg.ApplicationConfiguration.Pprof, log)
|
pprof := metrics.NewPprofService(cfg.ApplicationConfiguration.Pprof, log)
|
||||||
|
|
||||||
|
@ -198,6 +197,9 @@ func dumpDB(ctx *cli.Context) error {
|
||||||
if count == 0 {
|
if count == 0 {
|
||||||
count = chainCount - start
|
count = chainCount - start
|
||||||
}
|
}
|
||||||
|
if start != 0 {
|
||||||
|
writer.WriteU32LE(start)
|
||||||
|
}
|
||||||
writer.WriteU32LE(count)
|
writer.WriteU32LE(count)
|
||||||
err = chaindump.Dump(chain, writer, start, count)
|
err = chaindump.Dump(chain, writer, start, count)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -356,7 +358,14 @@ func resetDB(ctx *cli.Context) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func mkOracle(config config.OracleConfiguration, magic netmode.Magic, chain *core.Blockchain, serv *network.Server, log *zap.Logger) (*oracle.Oracle, error) {
|
// oracleService is an interface representing Oracle service with network.Service
|
||||||
|
// capabilities and ability to submit oracle responses.
|
||||||
|
type oracleService interface {
|
||||||
|
rpcsrv.OracleHandler
|
||||||
|
network.Service
|
||||||
|
}
|
||||||
|
|
||||||
|
func mkOracle(config config.OracleConfiguration, magic netmode.Magic, chain *core.Blockchain, serv *network.Server, log *zap.Logger) (oracleService, error) {
|
||||||
if !config.Enabled {
|
if !config.Enabled {
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
@ -413,7 +422,7 @@ func mkP2PNotary(config config.P2PNotary, chain *core.Blockchain, serv *network.
|
||||||
}
|
}
|
||||||
n, err := notary.NewNotary(cfg, serv.Net, serv.GetNotaryPool(), func(tx *transaction.Transaction) error {
|
n, err := notary.NewNotary(cfg, serv.Net, serv.GetNotaryPool(), func(tx *transaction.Transaction) error {
|
||||||
err := serv.RelayTxn(tx)
|
err := serv.RelayTxn(tx)
|
||||||
if err != nil && !errors.Is(err, core.ErrAlreadyExists) {
|
if err != nil && !errors.Is(err, core.ErrAlreadyExists) && !errors.Is(err, core.ErrAlreadyInPool) {
|
||||||
return fmt.Errorf("can't relay completed notary transaction: hash %s, error: %w", tx.Hash().StringLE(), err)
|
return fmt.Errorf("can't relay completed notary transaction: hash %s, error: %w", tx.Hash().StringLE(), err)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
@ -489,9 +498,12 @@ func startServer(ctx *cli.Context) error {
|
||||||
rpcServer := rpcsrv.New(chain, cfg.ApplicationConfiguration.RPC, serv, oracleSrv, log, errChan)
|
rpcServer := rpcsrv.New(chain, cfg.ApplicationConfiguration.RPC, serv, oracleSrv, log, errChan)
|
||||||
serv.AddService(&rpcServer)
|
serv.AddService(&rpcServer)
|
||||||
|
|
||||||
go serv.Start(errChan)
|
serv.Start()
|
||||||
if !cfg.ApplicationConfiguration.RPC.StartWhenSynchronized {
|
if !cfg.ApplicationConfiguration.RPC.StartWhenSynchronized {
|
||||||
rpcServer.Start()
|
// Run RPC server in a separate routine. This is necessary to avoid a potential
|
||||||
|
// deadlock: Start() can write errors to errChan which is not yet read in the
|
||||||
|
// current execution context (see for-loop below).
|
||||||
|
go rpcServer.Start()
|
||||||
}
|
}
|
||||||
|
|
||||||
sigCh := make(chan os.Signal, 1)
|
sigCh := make(chan os.Signal, 1)
|
||||||
|
@ -534,7 +546,6 @@ Main:
|
||||||
break // Continue working.
|
break // Continue working.
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
configureAddresses(&cfgnew.ApplicationConfiguration)
|
|
||||||
switch sig {
|
switch sig {
|
||||||
case sighup:
|
case sighup:
|
||||||
if newLogLevel != zapcore.InvalidLevel {
|
if newLogLevel != zapcore.InvalidLevel {
|
||||||
|
@ -546,7 +557,8 @@ Main:
|
||||||
rpcServer = rpcsrv.New(chain, cfgnew.ApplicationConfiguration.RPC, serv, oracleSrv, log, errChan)
|
rpcServer = rpcsrv.New(chain, cfgnew.ApplicationConfiguration.RPC, serv, oracleSrv, log, errChan)
|
||||||
serv.AddService(&rpcServer)
|
serv.AddService(&rpcServer)
|
||||||
if !cfgnew.ApplicationConfiguration.RPC.StartWhenSynchronized || serv.IsInSync() {
|
if !cfgnew.ApplicationConfiguration.RPC.StartWhenSynchronized || serv.IsInSync() {
|
||||||
rpcServer.Start()
|
// Here similar to the initial run (see above for-loop), so async.
|
||||||
|
go rpcServer.Start()
|
||||||
}
|
}
|
||||||
pprof.ShutDown()
|
pprof.ShutDown()
|
||||||
pprof = metrics.NewPprofService(cfgnew.ApplicationConfiguration.Pprof, log)
|
pprof = metrics.NewPprofService(cfgnew.ApplicationConfiguration.Pprof, log)
|
||||||
|
@ -634,24 +646,6 @@ Main:
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// configureAddresses sets up addresses for RPC, Prometheus and Pprof depending from the provided config.
|
|
||||||
// In case RPC or Prometheus or Pprof Address provided each of them will use it.
|
|
||||||
// In case global Address (of the node) provided and RPC/Prometheus/Pprof don't have configured addresses they will
|
|
||||||
// use global one. So Node and RPC and Prometheus and Pprof will run on one address.
|
|
||||||
func configureAddresses(cfg *config.ApplicationConfiguration) {
|
|
||||||
if cfg.Address != nil && *cfg.Address != "" { //nolint:staticcheck // SA1019: cfg.Address is deprecated
|
|
||||||
if cfg.RPC.Address == nil || *cfg.RPC.Address == "" { //nolint:staticcheck // SA1019: cfg.RPC.Address is deprecated
|
|
||||||
cfg.RPC.Address = cfg.Address //nolint:staticcheck // SA1019: cfg.RPC.Address is deprecated
|
|
||||||
}
|
|
||||||
if cfg.Prometheus.Address == nil || *cfg.Prometheus.Address == "" { //nolint:staticcheck // SA1019: cfg.Prometheus.Address is deprecated
|
|
||||||
cfg.Prometheus.Address = cfg.Address //nolint:staticcheck // SA1019: cfg.Prometheus.Address is deprecated
|
|
||||||
}
|
|
||||||
if cfg.Pprof.Address == nil || *cfg.Pprof.Address == "" { //nolint:staticcheck // SA1019: cfg.Pprof.Address is deprecated
|
|
||||||
cfg.Pprof.Address = cfg.Address //nolint:staticcheck // SA1019: cfg.Pprof.Address is deprecated
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// initBlockChain initializes BlockChain with preselected DB.
|
// initBlockChain initializes BlockChain with preselected DB.
|
||||||
func initBlockChain(cfg config.Config, log *zap.Logger) (*core.Blockchain, storage.Store, error) {
|
func initBlockChain(cfg config.Config, log *zap.Logger) (*core.Blockchain, storage.Store, error) {
|
||||||
store, err := storage.NewStore(cfg.ApplicationConfiguration.DBConfiguration)
|
store, err := storage.NewStore(cfg.ApplicationConfiguration.DBConfiguration)
|
||||||
|
@ -662,7 +656,7 @@ func initBlockChain(cfg config.Config, log *zap.Logger) (*core.Blockchain, stora
|
||||||
chain, err := core.NewBlockchain(store, cfg.Blockchain(), log)
|
chain, err := core.NewBlockchain(store, cfg.Blockchain(), log)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
errText := "could not initialize blockchain: %w"
|
errText := "could not initialize blockchain: %w"
|
||||||
errArgs := []interface{}{err}
|
errArgs := []any{err}
|
||||||
closeErr := store.Close()
|
closeErr := store.Close()
|
||||||
if closeErr != nil {
|
if closeErr != nil {
|
||||||
errText += "; failed to close the DB: %w"
|
errText += "; failed to close the DB: %w"
|
||||||
|
|
|
@ -5,6 +5,7 @@ import (
|
||||||
"flag"
|
"flag"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
"runtime"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"go.uber.org/zap/zapcore"
|
"go.uber.org/zap/zapcore"
|
||||||
|
@ -30,6 +31,7 @@ func init() {
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestGetConfigFromContext(t *testing.T) {
|
func TestGetConfigFromContext(t *testing.T) {
|
||||||
|
t.Run("config-path", func(t *testing.T) {
|
||||||
set := flag.NewFlagSet("flagSet", flag.ExitOnError)
|
set := flag.NewFlagSet("flagSet", flag.ExitOnError)
|
||||||
set.String("config-path", "../../config", "")
|
set.String("config-path", "../../config", "")
|
||||||
set.Bool("testnet", true, "")
|
set.Bool("testnet", true, "")
|
||||||
|
@ -37,6 +39,50 @@ func TestGetConfigFromContext(t *testing.T) {
|
||||||
cfg, err := options.GetConfigFromContext(ctx)
|
cfg, err := options.GetConfigFromContext(ctx)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.Equal(t, netmode.TestNet, cfg.ProtocolConfiguration.Magic)
|
require.Equal(t, netmode.TestNet, cfg.ProtocolConfiguration.Magic)
|
||||||
|
})
|
||||||
|
t.Run("config-file", func(t *testing.T) {
|
||||||
|
set := flag.NewFlagSet("flagSet", flag.ExitOnError)
|
||||||
|
set.String("config-path", "../../config", "")
|
||||||
|
set.Bool("testnet", true, "")
|
||||||
|
set.String("config-file", "../../config/protocol.testnet.yml", "")
|
||||||
|
ctx := cli.NewContext(cli.NewApp(), set, nil)
|
||||||
|
cfg, err := options.GetConfigFromContext(ctx)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, netmode.TestNet, cfg.ProtocolConfiguration.Magic)
|
||||||
|
})
|
||||||
|
t.Run("relative-path windows", func(t *testing.T) {
|
||||||
|
if runtime.GOOS != "windows" {
|
||||||
|
t.Skip("skipping Windows specific test")
|
||||||
|
}
|
||||||
|
|
||||||
|
set := flag.NewFlagSet("flagSet", flag.ExitOnError)
|
||||||
|
set.String("relative-path", "..\\..\\config", "")
|
||||||
|
set.Bool("testnet", true, "")
|
||||||
|
set.String("config-file", ".\\testdata\\protocol.testnet.windows.yml", "")
|
||||||
|
ctx := cli.NewContext(cli.NewApp(), set, nil)
|
||||||
|
cfg, err := options.GetConfigFromContext(ctx)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, filepath.Join("..", "..", "config", "chains", "testnet"), cfg.ApplicationConfiguration.DBConfiguration.LevelDBOptions.DataDirectoryPath)
|
||||||
|
require.Equal(t, "C:\\someFolder\\cn_wallet.json", cfg.ApplicationConfiguration.Consensus.UnlockWallet.Path)
|
||||||
|
require.Equal(t, "C:\\someFolder\\notary_wallet.json", cfg.ApplicationConfiguration.P2PNotary.UnlockWallet.Path)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("relative-path non-windows", func(t *testing.T) {
|
||||||
|
if runtime.GOOS == "windows" {
|
||||||
|
t.Skip("skipping non-Windows specific test")
|
||||||
|
}
|
||||||
|
|
||||||
|
set := flag.NewFlagSet("flagSet", flag.ExitOnError)
|
||||||
|
set.String("relative-path", "../../config", "")
|
||||||
|
set.Bool("testnet", true, "")
|
||||||
|
set.String("config-file", "../../config/protocol.testnet.yml", "")
|
||||||
|
ctx := cli.NewContext(cli.NewApp(), set, nil)
|
||||||
|
cfg, err := options.GetConfigFromContext(ctx)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, filepath.Join("..", "..", "config", "chains", "testnet"), cfg.ApplicationConfiguration.DBConfiguration.LevelDBOptions.DataDirectoryPath)
|
||||||
|
require.Equal(t, "/cn_wallet.json", cfg.ApplicationConfiguration.Consensus.UnlockWallet.Path)
|
||||||
|
require.Equal(t, "/notary_wallet.json", cfg.ApplicationConfiguration.P2PNotary.UnlockWallet.Path)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestHandleLoggingParams(t *testing.T) {
|
func TestHandleLoggingParams(t *testing.T) {
|
||||||
|
@ -312,64 +358,6 @@ func TestRestoreDB(t *testing.T) {
|
||||||
require.NoError(t, restoreDB(ctx))
|
require.NoError(t, restoreDB(ctx))
|
||||||
}
|
}
|
||||||
|
|
||||||
// TestConfigureAddresses checks deprecated code compatibility, it should be removed
|
|
||||||
// along with deprecated `Address` field removal.
|
|
||||||
func TestConfigureAddresses(t *testing.T) {
|
|
||||||
defaultAddress := "http://localhost:10333"
|
|
||||||
customAddress := "http://localhost:10334"
|
|
||||||
|
|
||||||
t.Run("default addresses", func(t *testing.T) {
|
|
||||||
cfg := &config.ApplicationConfiguration{
|
|
||||||
Address: &defaultAddress, //nolint:staticcheck // SA1019: Address is deprecated
|
|
||||||
}
|
|
||||||
configureAddresses(cfg)
|
|
||||||
require.Equal(t, defaultAddress, *cfg.RPC.Address) //nolint:staticcheck // SA1019: cfg.RPC.Address is deprecated
|
|
||||||
require.Equal(t, defaultAddress, *cfg.Prometheus.Address) //nolint:staticcheck // SA1019: cfg.Prometheus.Address is deprecated
|
|
||||||
require.Equal(t, defaultAddress, *cfg.Pprof.Address) //nolint:staticcheck // SA1019: cfg.Pprof.Address is deprecated
|
|
||||||
})
|
|
||||||
|
|
||||||
t.Run("custom RPC address", func(t *testing.T) {
|
|
||||||
cfg := &config.ApplicationConfiguration{
|
|
||||||
Address: &defaultAddress, //nolint:staticcheck // SA1019: Address is deprecated
|
|
||||||
RPC: config.RPC{
|
|
||||||
BasicService: config.BasicService{
|
|
||||||
Address: &customAddress, //nolint:staticcheck // SA1019: Address is deprecated
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
configureAddresses(cfg)
|
|
||||||
require.Equal(t, *cfg.RPC.Address, customAddress) //nolint:staticcheck // SA1019: cfg.RPC.Address is deprecated
|
|
||||||
require.Equal(t, *cfg.Prometheus.Address, defaultAddress) //nolint:staticcheck // SA1019: cfg.Prometheus.Address is deprecated
|
|
||||||
require.Equal(t, *cfg.Pprof.Address, defaultAddress) //nolint:staticcheck // SA1019: cfg.Pprof.Address is deprecated
|
|
||||||
})
|
|
||||||
|
|
||||||
t.Run("custom Pprof address", func(t *testing.T) {
|
|
||||||
cfg := &config.ApplicationConfiguration{
|
|
||||||
Address: &defaultAddress, //nolint:staticcheck // SA1019: Address is deprecated
|
|
||||||
Pprof: config.BasicService{
|
|
||||||
Address: &customAddress, //nolint:staticcheck // SA1019: Address is deprecated
|
|
||||||
},
|
|
||||||
}
|
|
||||||
configureAddresses(cfg)
|
|
||||||
require.Equal(t, *cfg.RPC.Address, defaultAddress) //nolint:staticcheck // SA1019: cfg.RPC.Address is deprecated
|
|
||||||
require.Equal(t, *cfg.Prometheus.Address, defaultAddress) //nolint:staticcheck // SA1019: cfg.Prometheus.Address is deprecated
|
|
||||||
require.Equal(t, *cfg.Pprof.Address, customAddress) //nolint:staticcheck // SA1019: cfg.Pprof.Address is deprecated
|
|
||||||
})
|
|
||||||
|
|
||||||
t.Run("custom Prometheus address", func(t *testing.T) {
|
|
||||||
cfg := &config.ApplicationConfiguration{
|
|
||||||
Address: &defaultAddress, //nolint:staticcheck // SA1019: Address is deprecated
|
|
||||||
Prometheus: config.BasicService{
|
|
||||||
Address: &customAddress, //nolint:staticcheck // SA1019: Address is deprecated
|
|
||||||
},
|
|
||||||
}
|
|
||||||
configureAddresses(cfg)
|
|
||||||
require.Equal(t, *cfg.RPC.Address, defaultAddress) //nolint:staticcheck // SA1019: cfg.RPC.Address is deprecated
|
|
||||||
require.Equal(t, *cfg.Prometheus.Address, customAddress) //nolint:staticcheck // SA1019: cfg.Prometheus.Address is deprecated
|
|
||||||
require.Equal(t, *cfg.Pprof.Address, defaultAddress) //nolint:staticcheck // SA1019: cfg.Pprof.Address is deprecated
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestInitBlockChain(t *testing.T) {
|
func TestInitBlockChain(t *testing.T) {
|
||||||
t.Run("bad storage", func(t *testing.T) {
|
t.Run("bad storage", func(t *testing.T) {
|
||||||
_, _, err := initBlockChain(config.Config{}, nil)
|
_, _, err := initBlockChain(config.Config{}, nil)
|
||||||
|
|
96
cli/server/testdata/protocol.testnet.windows.yml
vendored
Normal file
96
cli/server/testdata/protocol.testnet.windows.yml
vendored
Normal file
|
@ -0,0 +1,96 @@
|
||||||
|
ProtocolConfiguration:
|
||||||
|
Magic: 894710606
|
||||||
|
MaxBlockSize: 2097152
|
||||||
|
MaxBlockSystemFee: 150000000000
|
||||||
|
MaxTraceableBlocks: 2102400
|
||||||
|
MaxTransactionsPerBlock: 5000
|
||||||
|
InitialGASSupply: 52000000
|
||||||
|
TimePerBlock: 15s
|
||||||
|
MemPoolSize: 50000
|
||||||
|
StandbyCommittee:
|
||||||
|
- 023e9b32ea89b94d066e649b124fd50e396ee91369e8e2a6ae1b11c170d022256d
|
||||||
|
- 03009b7540e10f2562e5fd8fac9eaec25166a58b26e412348ff5a86927bfac22a2
|
||||||
|
- 02ba2c70f5996f357a43198705859fae2cfea13e1172962800772b3d588a9d4abd
|
||||||
|
- 03408dcd416396f64783ac587ea1e1593c57d9fea880c8a6a1920e92a259477806
|
||||||
|
- 02a7834be9b32e2981d157cb5bbd3acb42cfd11ea5c3b10224d7a44e98c5910f1b
|
||||||
|
- 0214baf0ceea3a66f17e7e1e839ea25fd8bed6cd82e6bb6e68250189065f44ff01
|
||||||
|
- 030205e9cefaea5a1dfc580af20c8d5aa2468bb0148f1a5e4605fc622c80e604ba
|
||||||
|
- 025831cee3708e87d78211bec0d1bfee9f4c85ae784762f042e7f31c0d40c329b8
|
||||||
|
- 02cf9dc6e85d581480d91e88e8cbeaa0c153a046e89ded08b4cefd851e1d7325b5
|
||||||
|
- 03840415b0a0fcf066bcc3dc92d8349ebd33a6ab1402ef649bae00e5d9f5840828
|
||||||
|
- 026328aae34f149853430f526ecaa9cf9c8d78a4ea82d08bdf63dd03c4d0693be6
|
||||||
|
- 02c69a8d084ee7319cfecf5161ff257aa2d1f53e79bf6c6f164cff5d94675c38b3
|
||||||
|
- 0207da870cedb777fceff948641021714ec815110ca111ccc7a54c168e065bda70
|
||||||
|
- 035056669864feea401d8c31e447fb82dd29f342a9476cfd449584ce2a6165e4d7
|
||||||
|
- 0370c75c54445565df62cfe2e76fbec4ba00d1298867972213530cae6d418da636
|
||||||
|
- 03957af9e77282ae3263544b7b2458903624adc3f5dee303957cb6570524a5f254
|
||||||
|
- 03d84d22b8753cf225d263a3a782a4e16ca72ef323cfde04977c74f14873ab1e4c
|
||||||
|
- 02147c1b1d5728e1954958daff2f88ee2fa50a06890a8a9db3fa9e972b66ae559f
|
||||||
|
- 03c609bea5a4825908027e4ab217e7efc06e311f19ecad9d417089f14927a173d5
|
||||||
|
- 0231edee3978d46c335e851c76059166eb8878516f459e085c0dd092f0f1d51c21
|
||||||
|
- 03184b018d6b2bc093e535519732b3fd3f7551c8cffaf4621dd5a0b89482ca66c9
|
||||||
|
ValidatorsCount: 7
|
||||||
|
SeedList:
|
||||||
|
- seed1t5.neo.org:20333
|
||||||
|
- seed2t5.neo.org:20333
|
||||||
|
- seed3t5.neo.org:20333
|
||||||
|
- seed4t5.neo.org:20333
|
||||||
|
- seed5t5.neo.org:20333
|
||||||
|
VerifyTransactions: false
|
||||||
|
P2PSigExtensions: false
|
||||||
|
Hardforks:
|
||||||
|
Aspidochelone: 210000
|
||||||
|
Basilisk: 2680000
|
||||||
|
|
||||||
|
ApplicationConfiguration:
|
||||||
|
SkipBlockVerification: false
|
||||||
|
DBConfiguration:
|
||||||
|
Type: "leveldb" #other options: 'inmemory','boltdb'
|
||||||
|
# DB type options. Uncomment those you need in case you want to switch DB type.
|
||||||
|
LevelDBOptions:
|
||||||
|
DataDirectoryPath: ".\\chains\\testnet"
|
||||||
|
P2P:
|
||||||
|
Addresses:
|
||||||
|
- ":20333" # in form of "[host]:[port][:announcedPort]"
|
||||||
|
DialTimeout: 3s
|
||||||
|
ProtoTickInterval: 2s
|
||||||
|
PingInterval: 30s
|
||||||
|
PingTimeout: 90s
|
||||||
|
MaxPeers: 100
|
||||||
|
AttemptConnPeers: 20
|
||||||
|
MinPeers: 10
|
||||||
|
Relay: true
|
||||||
|
Consensus:
|
||||||
|
Enabled: false
|
||||||
|
UnlockWallet:
|
||||||
|
Path: "C:\\someFolder\\cn_wallet.json"
|
||||||
|
Password: "pass"
|
||||||
|
Oracle:
|
||||||
|
Enabled: false
|
||||||
|
AllowedContentTypes:
|
||||||
|
- application/json
|
||||||
|
P2PNotary:
|
||||||
|
Enabled: false
|
||||||
|
UnlockWallet:
|
||||||
|
Path: "C:\\someFolder\\notary_wallet.json"
|
||||||
|
Password: "pass"
|
||||||
|
RPC:
|
||||||
|
Enabled: true
|
||||||
|
Addresses:
|
||||||
|
- ":20332"
|
||||||
|
MaxGasInvoke: 15
|
||||||
|
EnableCORSWorkaround: false
|
||||||
|
TLSConfig:
|
||||||
|
Enabled: false
|
||||||
|
Addresses:
|
||||||
|
- ":20331"
|
||||||
|
CertFile: serv.crt
|
||||||
|
KeyFile: serv.key
|
||||||
|
Prometheus:
|
||||||
|
Enabled: true
|
||||||
|
Addresses:
|
||||||
|
- ":2112"
|
||||||
|
Pprof:
|
||||||
|
Enabled: false
|
||||||
|
Addresses:
|
||||||
|
- ":2113"
|
|
@ -4,6 +4,7 @@ import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
@ -13,6 +14,7 @@ import (
|
||||||
"github.com/nspcc-dev/neo-go/cli/smartcontract"
|
"github.com/nspcc-dev/neo-go/cli/smartcontract"
|
||||||
"github.com/nspcc-dev/neo-go/internal/random"
|
"github.com/nspcc-dev/neo-go/internal/random"
|
||||||
"github.com/nspcc-dev/neo-go/internal/testcli"
|
"github.com/nspcc-dev/neo-go/internal/testcli"
|
||||||
|
"github.com/nspcc-dev/neo-go/internal/versionutil"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/config"
|
"github.com/nspcc-dev/neo-go/pkg/config"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/interop/storage"
|
"github.com/nspcc-dev/neo-go/pkg/core/interop/storage"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
||||||
|
@ -23,6 +25,7 @@ import (
|
||||||
"github.com/nspcc-dev/neo-go/pkg/neorpc/result"
|
"github.com/nspcc-dev/neo-go/pkg/neorpc/result"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract/nef"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/nef"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/trigger"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/util"
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/vm/vmstate"
|
"github.com/nspcc-dev/neo-go/pkg/vm/vmstate"
|
||||||
|
@ -31,6 +34,9 @@ import (
|
||||||
"gopkg.in/yaml.v3"
|
"gopkg.in/yaml.v3"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// Keep contract NEFs consistent between runs.
|
||||||
|
const _ = versionutil.TestVersion
|
||||||
|
|
||||||
func TestCalcHash(t *testing.T) {
|
func TestCalcHash(t *testing.T) {
|
||||||
tmpDir := t.TempDir()
|
tmpDir := t.TempDir()
|
||||||
e := testcli.NewExecutor(t, false)
|
e := testcli.NewExecutor(t, false)
|
||||||
|
@ -97,9 +103,6 @@ func TestCalcHash(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestContractBindings(t *testing.T) {
|
func TestContractBindings(t *testing.T) {
|
||||||
// For proper nef generation.
|
|
||||||
config.Version = "v0.98.1-test"
|
|
||||||
|
|
||||||
// For proper contract init. The actual version as it will be replaced.
|
// For proper contract init. The actual version as it will be replaced.
|
||||||
smartcontract.ModVersion = "v0.0.0"
|
smartcontract.ModVersion = "v0.0.0"
|
||||||
|
|
||||||
|
@ -143,18 +146,7 @@ func Blocks() []*alias.Block {
|
||||||
cmd = append(cmd, "--in", ctrPath, "--bindings", bindingsPath)
|
cmd = append(cmd, "--in", ctrPath, "--bindings", bindingsPath)
|
||||||
|
|
||||||
// Replace `pkg/interop` in go.mod to avoid getting an actual module version.
|
// Replace `pkg/interop` in go.mod to avoid getting an actual module version.
|
||||||
goMod := filepath.Join(ctrPath, "go.mod")
|
require.NoError(t, updateGoMod(ctrPath, "myimport.com/testcontract", "../../pkg/interop"))
|
||||||
data, err := os.ReadFile(goMod)
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
i := bytes.IndexByte(data, '\n')
|
|
||||||
data = append([]byte("module myimport.com/testcontract"), data[i:]...)
|
|
||||||
|
|
||||||
wd, err := os.Getwd()
|
|
||||||
require.NoError(t, err)
|
|
||||||
data = append(data, "\nreplace github.com/nspcc-dev/neo-go/pkg/interop => "...)
|
|
||||||
data = append(data, filepath.Join(wd, "../../pkg/interop")...)
|
|
||||||
require.NoError(t, os.WriteFile(goMod, data, os.ModePerm))
|
|
||||||
|
|
||||||
cmd = append(cmd, "--config", cfgPath,
|
cmd = append(cmd, "--config", cfgPath,
|
||||||
"--out", filepath.Join(tmpDir, "out.nef"),
|
"--out", filepath.Join(tmpDir, "out.nef"),
|
||||||
|
@ -174,7 +166,9 @@ func Blocks() []*alias.Block {
|
||||||
|
|
||||||
bs, err := os.ReadFile(outPath)
|
bs, err := os.ReadFile(outPath)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.Equal(t, `// Package testcontract contains wrappers for testcontract contract.
|
require.Equal(t, `// Code generated by neo-go contract generate-wrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
|
// Package testcontract contains wrappers for testcontract contract.
|
||||||
package testcontract
|
package testcontract
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
@ -209,10 +203,85 @@ func ToMap(a []testcontract.MyPair) map[int]string {
|
||||||
`, string(bs))
|
`, string(bs))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestContractInitAndCompile(t *testing.T) {
|
// updateGoMod updates the go.mod file located in the specified directory.
|
||||||
// For proper nef generation.
|
// It sets the module name and replaces the neo-go interop package path with
|
||||||
config.Version = "v0.98.1-test"
|
// the provided one to avoid getting an actual module version.
|
||||||
|
func updateGoMod(dir, moduleName, neoGoPath string) error {
|
||||||
|
goModPath := filepath.Join(dir, "go.mod")
|
||||||
|
data, err := os.ReadFile(goModPath)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to read go.mod: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
i := bytes.IndexByte(data, '\n')
|
||||||
|
if i == -1 {
|
||||||
|
return fmt.Errorf("unexpected go.mod format")
|
||||||
|
}
|
||||||
|
|
||||||
|
updatedData := append([]byte("module "+moduleName), data[i:]...)
|
||||||
|
wd, err := os.Getwd()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to get working directory: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
replacementPath := filepath.Join(wd, neoGoPath)
|
||||||
|
updatedData = append(updatedData, "\nreplace github.com/nspcc-dev/neo-go/pkg/interop => "+replacementPath+" \n"...)
|
||||||
|
|
||||||
|
if err := os.WriteFile(goModPath, updatedData, os.ModePerm); err != nil {
|
||||||
|
return fmt.Errorf("failed to write updated go.mod: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestDynamicWrapper(t *testing.T) {
|
||||||
|
// For proper contract init. The actual version as it will be replaced.
|
||||||
|
smartcontract.ModVersion = "v0.0.0"
|
||||||
|
|
||||||
|
tmpDir := t.TempDir()
|
||||||
|
e := testcli.NewExecutor(t, true)
|
||||||
|
|
||||||
|
ctrPath := "../smartcontract/testdata"
|
||||||
|
|
||||||
|
verifyHash := testcli.DeployContract(t, e, filepath.Join(ctrPath, "verify.go"), filepath.Join(ctrPath, "verify.yml"), testcli.ValidatorWallet, testcli.ValidatorAddr, testcli.ValidatorPass)
|
||||||
|
|
||||||
|
helperContract := `package testcontract
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/interop"
|
||||||
|
verify "myimport.com/testcontract/bindings"
|
||||||
|
)
|
||||||
|
|
||||||
|
func CallVerifyContract(h interop.Hash160) bool{
|
||||||
|
contractInstance := verify.NewContract(h)
|
||||||
|
return contractInstance.Verify()
|
||||||
|
}`
|
||||||
|
|
||||||
|
helperDir := filepath.Join(tmpDir, "helper")
|
||||||
|
e.Run(t, "neo-go", "contract", "init", "--name", helperDir)
|
||||||
|
|
||||||
|
require.NoError(t, updateGoMod(helperDir, "myimport.com/testcontract", "../../pkg/interop"))
|
||||||
|
require.NoError(t, os.WriteFile(filepath.Join(helperDir, "main.go"), []byte(helperContract), os.ModePerm))
|
||||||
|
require.NoError(t, os.Mkdir(filepath.Join(helperDir, "bindings"), os.ModePerm))
|
||||||
|
|
||||||
|
e.Run(t, "neo-go", "contract", "generate-wrapper",
|
||||||
|
"--config", filepath.Join(ctrPath, "verify.bindings.yml"), "--manifest", filepath.Join(ctrPath, "verify.manifest.json"),
|
||||||
|
"--out", filepath.Join(helperDir, "bindings", "testdata.go"))
|
||||||
|
e.Run(t, "neo-go", "contract", "compile", "--in", filepath.Join(helperDir, "main.go"), "--config", filepath.Join(helperDir, "neo-go.yml"))
|
||||||
|
helperHash := testcli.DeployContract(t, e, filepath.Join(helperDir, "main.go"), filepath.Join(helperDir, "neo-go.yml"), testcli.ValidatorWallet, testcli.ValidatorAddr, testcli.ValidatorPass)
|
||||||
|
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.Run(t, "neo-go", "contract", "invokefunction",
|
||||||
|
"--rpc-endpoint", "http://"+e.RPC.Addresses()[0],
|
||||||
|
"--wallet", testcli.ValidatorWallet, "--address", testcli.ValidatorAddr, "--force", "--await", helperHash.StringLE(), "callVerifyContract", verifyHash.StringLE())
|
||||||
|
|
||||||
|
tx, _ := e.CheckTxPersisted(t, "Sent invocation transaction ")
|
||||||
|
aer, err := e.Chain.GetAppExecResults(tx.Hash(), trigger.Application)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, aer[0].Stack[0].Value().(bool), true)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestContractInitAndCompile(t *testing.T) {
|
||||||
// For proper contract init. The actual version as it will be replaced.
|
// For proper contract init. The actual version as it will be replaced.
|
||||||
smartcontract.ModVersion = "v0.0.0"
|
smartcontract.ModVersion = "v0.0.0"
|
||||||
|
|
||||||
|
@ -237,7 +306,8 @@ func TestContractInitAndCompile(t *testing.T) {
|
||||||
e.RunWithError(t, "neo-go", "contract", "init", "--name", ctrPath)
|
e.RunWithError(t, "neo-go", "contract", "init", "--name", ctrPath)
|
||||||
})
|
})
|
||||||
|
|
||||||
srcPath := filepath.Join(ctrPath, "main.go")
|
ctrRootPath := filepath.Join(ctrPath, "main")
|
||||||
|
srcPath := ctrRootPath + ".go"
|
||||||
cfgPath := filepath.Join(ctrPath, "neo-go.yml")
|
cfgPath := filepath.Join(ctrPath, "neo-go.yml")
|
||||||
nefPath := filepath.Join(tmpDir, "testcontract.nef")
|
nefPath := filepath.Join(tmpDir, "testcontract.nef")
|
||||||
manifestPath := filepath.Join(tmpDir, "testcontract.manifest.json")
|
manifestPath := filepath.Join(tmpDir, "testcontract.manifest.json")
|
||||||
|
@ -263,15 +333,7 @@ func TestContractInitAndCompile(t *testing.T) {
|
||||||
})
|
})
|
||||||
|
|
||||||
// Replace `pkg/interop` in go.mod to avoid getting an actual module version.
|
// Replace `pkg/interop` in go.mod to avoid getting an actual module version.
|
||||||
goMod := filepath.Join(ctrPath, "go.mod")
|
require.NoError(t, updateGoMod(ctrPath, "myimport.com/testcontract", "../../pkg/interop"))
|
||||||
data, err := os.ReadFile(goMod)
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
wd, err := os.Getwd()
|
|
||||||
require.NoError(t, err)
|
|
||||||
data = append(data, "\nreplace github.com/nspcc-dev/neo-go/pkg/interop => "...)
|
|
||||||
data = append(data, filepath.Join(wd, "../../pkg/interop")...)
|
|
||||||
require.NoError(t, os.WriteFile(goMod, data, os.ModePerm))
|
|
||||||
|
|
||||||
cmd = append(cmd, "--config", cfgPath)
|
cmd = append(cmd, "--config", cfgPath)
|
||||||
|
|
||||||
|
@ -288,6 +350,19 @@ func TestContractInitAndCompile(t *testing.T) {
|
||||||
e.Run(t, append(cmd, "--verbose")...)
|
e.Run(t, append(cmd, "--verbose")...)
|
||||||
e.CheckNextLine(t, "^[0-9a-hA-H]+$")
|
e.CheckNextLine(t, "^[0-9a-hA-H]+$")
|
||||||
})
|
})
|
||||||
|
|
||||||
|
t.Run("autocomplete outputs", func(t *testing.T) {
|
||||||
|
cfg, err := os.ReadFile(cfgPath)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.NoError(t, os.WriteFile(filepath.Join(ctrPath, "main.yml"), cfg, os.ModePerm))
|
||||||
|
e.Run(t, "neo-go", "contract", "compile", "--in", srcPath)
|
||||||
|
defaultNefPath := ctrRootPath + ".nef"
|
||||||
|
defaultManifestPath := ctrRootPath + ".manifest.json"
|
||||||
|
defaultBindingsPath := ctrRootPath + ".bindings.yml"
|
||||||
|
require.FileExists(t, defaultNefPath)
|
||||||
|
require.FileExists(t, defaultManifestPath)
|
||||||
|
require.FileExists(t, defaultBindingsPath)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// Checks that error is returned if GAS available for test-invoke exceeds
|
// Checks that error is returned if GAS available for test-invoke exceeds
|
||||||
|
@ -296,9 +371,6 @@ func TestDeployBigContract(t *testing.T) {
|
||||||
e := testcli.NewExecutorWithConfig(t, true, true, func(c *config.Config) {
|
e := testcli.NewExecutorWithConfig(t, true, true, func(c *config.Config) {
|
||||||
c.ApplicationConfiguration.RPC.MaxGasInvoke = fixedn.Fixed8(1)
|
c.ApplicationConfiguration.RPC.MaxGasInvoke = fixedn.Fixed8(1)
|
||||||
})
|
})
|
||||||
|
|
||||||
// For proper nef generation.
|
|
||||||
config.Version = "0.90.0-test"
|
|
||||||
tmpDir := t.TempDir()
|
tmpDir := t.TempDir()
|
||||||
|
|
||||||
nefName := filepath.Join(tmpDir, "deploy.nef")
|
nefName := filepath.Join(tmpDir, "deploy.nef")
|
||||||
|
@ -317,9 +389,6 @@ func TestDeployBigContract(t *testing.T) {
|
||||||
|
|
||||||
func TestContractDeployWithData(t *testing.T) {
|
func TestContractDeployWithData(t *testing.T) {
|
||||||
eCompile := testcli.NewExecutor(t, false)
|
eCompile := testcli.NewExecutor(t, false)
|
||||||
|
|
||||||
// For proper nef generation.
|
|
||||||
config.Version = "0.90.0-test"
|
|
||||||
tmpDir := t.TempDir()
|
tmpDir := t.TempDir()
|
||||||
|
|
||||||
nefName := filepath.Join(tmpDir, "deploy.nef")
|
nefName := filepath.Join(tmpDir, "deploy.nef")
|
||||||
|
@ -329,7 +398,7 @@ func TestContractDeployWithData(t *testing.T) {
|
||||||
"--config", "testdata/deploy/neo-go.yml",
|
"--config", "testdata/deploy/neo-go.yml",
|
||||||
"--out", nefName, "--manifest", manifestName)
|
"--out", nefName, "--manifest", manifestName)
|
||||||
|
|
||||||
deployContract := func(t *testing.T, haveData bool, scope string) {
|
deployContract := func(t *testing.T, haveData bool, scope string, await bool) {
|
||||||
e := testcli.NewExecutor(t, true)
|
e := testcli.NewExecutor(t, true)
|
||||||
cmd := []string{
|
cmd := []string{
|
||||||
"neo-go", "contract", "deploy",
|
"neo-go", "contract", "deploy",
|
||||||
|
@ -339,6 +408,9 @@ func TestContractDeployWithData(t *testing.T) {
|
||||||
"--force",
|
"--force",
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if await {
|
||||||
|
cmd = append(cmd, "--await")
|
||||||
|
}
|
||||||
if haveData {
|
if haveData {
|
||||||
cmd = append(cmd, "[", "key1", "12", "key2", "take_me_to_church", "]")
|
cmd = append(cmd, "[", "key1", "12", "key2", "take_me_to_church", "]")
|
||||||
}
|
}
|
||||||
|
@ -349,8 +421,13 @@ func TestContractDeployWithData(t *testing.T) {
|
||||||
}
|
}
|
||||||
e.In.WriteString(testcli.ValidatorPass + "\r")
|
e.In.WriteString(testcli.ValidatorPass + "\r")
|
||||||
e.Run(t, cmd...)
|
e.Run(t, cmd...)
|
||||||
|
var tx *transaction.Transaction
|
||||||
|
if await {
|
||||||
|
tx, _ = e.CheckAwaitableTxPersisted(t)
|
||||||
|
} else {
|
||||||
|
tx, _ = e.CheckTxPersisted(t)
|
||||||
|
}
|
||||||
|
|
||||||
tx, _ := e.CheckTxPersisted(t, "Sent invocation transaction ")
|
|
||||||
require.Equal(t, scope, tx.Signers[0].Scopes.String())
|
require.Equal(t, scope, tx.Signers[0].Scopes.String())
|
||||||
if !haveData {
|
if !haveData {
|
||||||
return
|
return
|
||||||
|
@ -387,16 +464,16 @@ func TestContractDeployWithData(t *testing.T) {
|
||||||
require.Equal(t, []byte("take_me_to_church"), res.Stack[0].Value())
|
require.Equal(t, []byte("take_me_to_church"), res.Stack[0].Value())
|
||||||
}
|
}
|
||||||
|
|
||||||
deployContract(t, true, "")
|
deployContract(t, true, "", false)
|
||||||
deployContract(t, false, "Global")
|
deployContract(t, false, "Global", false)
|
||||||
deployContract(t, true, "Global")
|
deployContract(t, true, "Global", false)
|
||||||
|
deployContract(t, false, "", true)
|
||||||
|
deployContract(t, true, "Global", true)
|
||||||
|
deployContract(t, true, "", true)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDeployWithSigners(t *testing.T) {
|
func TestDeployWithSigners(t *testing.T) {
|
||||||
e := testcli.NewExecutor(t, true)
|
e := testcli.NewExecutor(t, true)
|
||||||
|
|
||||||
// For proper nef generation.
|
|
||||||
config.Version = "0.90.0-test"
|
|
||||||
tmpDir := t.TempDir()
|
tmpDir := t.TempDir()
|
||||||
|
|
||||||
nefName := filepath.Join(tmpDir, "deploy.nef")
|
nefName := filepath.Join(tmpDir, "deploy.nef")
|
||||||
|
@ -458,9 +535,6 @@ func TestDeployWithSigners(t *testing.T) {
|
||||||
|
|
||||||
func TestContractManifestGroups(t *testing.T) {
|
func TestContractManifestGroups(t *testing.T) {
|
||||||
e := testcli.NewExecutor(t, true)
|
e := testcli.NewExecutor(t, true)
|
||||||
|
|
||||||
// For proper nef generation.
|
|
||||||
config.Version = "0.90.0-test"
|
|
||||||
tmpDir := t.TempDir()
|
tmpDir := t.TempDir()
|
||||||
|
|
||||||
_, err := wallet.NewWalletFromFile(testcli.TestWalletPath)
|
_, err := wallet.NewWalletFromFile(testcli.TestWalletPath)
|
||||||
|
@ -617,9 +691,6 @@ func TestContract_TestInvokeScript(t *testing.T) {
|
||||||
|
|
||||||
func TestComlileAndInvokeFunction(t *testing.T) {
|
func TestComlileAndInvokeFunction(t *testing.T) {
|
||||||
e := testcli.NewExecutor(t, true)
|
e := testcli.NewExecutor(t, true)
|
||||||
|
|
||||||
// For proper nef generation.
|
|
||||||
config.Version = "0.90.0-test"
|
|
||||||
tmpDir := t.TempDir()
|
tmpDir := t.TempDir()
|
||||||
|
|
||||||
nefName := filepath.Join(tmpDir, "deploy.nef")
|
nefName := filepath.Join(tmpDir, "deploy.nef")
|
||||||
|
@ -772,6 +843,12 @@ func TestComlileAndInvokeFunction(t *testing.T) {
|
||||||
e.Run(t, append(cmd, h.StringLE(), "getValue",
|
e.Run(t, append(cmd, h.StringLE(), "getValue",
|
||||||
"--", testcli.ValidatorAddr, hVerify.StringLE())...)
|
"--", testcli.ValidatorAddr, hVerify.StringLE())...)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
t.Run("with await", func(t *testing.T) {
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.Run(t, append(cmd, "--force", "--await", h.StringLE(), "getValue")...)
|
||||||
|
e.CheckAwaitableTxPersisted(t)
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
t.Run("real invoke and save tx", func(t *testing.T) {
|
t.Run("real invoke and save tx", func(t *testing.T) {
|
||||||
|
@ -949,9 +1026,6 @@ func TestComlileAndInvokeFunction(t *testing.T) {
|
||||||
|
|
||||||
func TestContractInspect(t *testing.T) {
|
func TestContractInspect(t *testing.T) {
|
||||||
e := testcli.NewExecutor(t, false)
|
e := testcli.NewExecutor(t, false)
|
||||||
|
|
||||||
// For proper nef generation.
|
|
||||||
config.Version = "0.90.0-test"
|
|
||||||
const srcPath = "testdata/deploy/main.go"
|
const srcPath = "testdata/deploy/main.go"
|
||||||
tmpDir := t.TempDir()
|
tmpDir := t.TempDir()
|
||||||
|
|
||||||
|
@ -986,9 +1060,6 @@ func TestCompileExamples(t *testing.T) {
|
||||||
infos, err := os.ReadDir(examplePath)
|
infos, err := os.ReadDir(examplePath)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
// For proper nef generation.
|
|
||||||
config.Version = "0.90.0-test"
|
|
||||||
|
|
||||||
e := testcli.NewExecutor(t, false)
|
e := testcli.NewExecutor(t, false)
|
||||||
|
|
||||||
for _, info := range infos {
|
for _, info := range infos {
|
||||||
|
@ -997,6 +1068,10 @@ func TestCompileExamples(t *testing.T) {
|
||||||
// there are also a couple of files inside the `/examples` which doesn't need to be compiled
|
// there are also a couple of files inside the `/examples` which doesn't need to be compiled
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
if info.Name() == "zkp" {
|
||||||
|
// A set of special ZKP-related examples, they have their own tests.
|
||||||
|
continue
|
||||||
|
}
|
||||||
t.Run(info.Name(), func(t *testing.T) {
|
t.Run(info.Name(), func(t *testing.T) {
|
||||||
infos, err := os.ReadDir(filepath.Join(examplePath, info.Name()))
|
infos, err := os.ReadDir(filepath.Join(examplePath, info.Name()))
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
@ -1073,3 +1148,27 @@ func filterFilename(infos []os.DirEntry, ext string) string {
|
||||||
}
|
}
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestContractCompile_NEFSizeCheck(t *testing.T) {
|
||||||
|
tmpDir := t.TempDir()
|
||||||
|
e := testcli.NewExecutor(t, false)
|
||||||
|
|
||||||
|
src := `package nefconstraints
|
||||||
|
var data = "%s"
|
||||||
|
|
||||||
|
func Main() string {
|
||||||
|
return data
|
||||||
|
}`
|
||||||
|
data := make([]byte, stackitem.MaxSize-10)
|
||||||
|
for i := range data {
|
||||||
|
data[i] = byte('a')
|
||||||
|
}
|
||||||
|
|
||||||
|
in := filepath.Join(tmpDir, "main.go")
|
||||||
|
cfg := filepath.Join(tmpDir, "main.yml")
|
||||||
|
require.NoError(t, os.WriteFile(cfg, []byte("name: main"), os.ModePerm))
|
||||||
|
require.NoError(t, os.WriteFile(in, []byte(fmt.Sprintf(src, data)), os.ModePerm))
|
||||||
|
|
||||||
|
e.RunWithError(t, "neo-go", "contract", "compile", "--in", in)
|
||||||
|
require.NoFileExists(t, filepath.Join(tmpDir, "main.nef"))
|
||||||
|
}
|
||||||
|
|
|
@ -30,16 +30,20 @@ var generatorFlags = []cli.Flag{
|
||||||
},
|
},
|
||||||
cli.StringFlag{
|
cli.StringFlag{
|
||||||
Name: "hash",
|
Name: "hash",
|
||||||
Required: true,
|
Usage: "Smart-contract hash. If not passed, the wrapper will be designed for dynamic hash usage",
|
||||||
Usage: "Smart-contract hash",
|
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
var generateWrapperCmd = cli.Command{
|
var generateWrapperCmd = cli.Command{
|
||||||
Name: "generate-wrapper",
|
Name: "generate-wrapper",
|
||||||
Usage: "generate wrapper to use in other contracts",
|
Usage: "generate wrapper to use in other contracts",
|
||||||
UsageText: "neo-go contract generate-wrapper --manifest <file.json> --out <file.go> --hash <hash> [--config <config>]",
|
UsageText: "neo-go contract generate-wrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]",
|
||||||
Description: ``,
|
Description: `Generates a Go wrapper to use it in other smart contracts. If the
|
||||||
|
--hash flag is provided, CALLT instruction is used for the target contract
|
||||||
|
invocation as an optimization of the wrapper contract code. If omitted, the
|
||||||
|
generated wrapper will be designed for dynamic hash usage, allowing
|
||||||
|
the hash to be specified at runtime.
|
||||||
|
`,
|
||||||
Action: contractGenerateWrapper,
|
Action: contractGenerateWrapper,
|
||||||
Flags: generatorFlags,
|
Flags: generatorFlags,
|
||||||
}
|
}
|
||||||
|
@ -47,7 +51,7 @@ var generateWrapperCmd = cli.Command{
|
||||||
var generateRPCWrapperCmd = cli.Command{
|
var generateRPCWrapperCmd = cli.Command{
|
||||||
Name: "generate-rpcwrapper",
|
Name: "generate-rpcwrapper",
|
||||||
Usage: "generate RPC wrapper to use for data reads",
|
Usage: "generate RPC wrapper to use for data reads",
|
||||||
UsageText: "neo-go contract generate-rpcwrapper --manifest <file.json> --out <file.go> --hash <hash> [--config <config>]",
|
UsageText: "neo-go contract generate-rpcwrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]",
|
||||||
Action: contractGenerateRPCWrapper,
|
Action: contractGenerateRPCWrapper,
|
||||||
Flags: generatorFlags,
|
Flags: generatorFlags,
|
||||||
}
|
}
|
||||||
|
@ -65,10 +69,16 @@ func contractGenerateSomething(ctx *cli.Context, cb func(binding.Config) error)
|
||||||
if err := cmdargs.EnsureNone(ctx); err != nil {
|
if err := cmdargs.EnsureNone(ctx); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
h, err := util.Uint160DecodeStringLE(strings.TrimPrefix(ctx.String("hash"), "0x"))
|
var (
|
||||||
|
h util.Uint160
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
if hStr := ctx.String("hash"); len(hStr) != 0 {
|
||||||
|
h, err = util.Uint160DecodeStringLE(strings.TrimPrefix(hStr, "0x"))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(fmt.Errorf("invalid contract hash: %w", err), 1)
|
return cli.NewExitError(fmt.Errorf("invalid contract hash: %w", err), 1)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
m, _, err := readManifest(ctx.String("manifest"), h)
|
m, _, err := readManifest(ctx.String("manifest"), h)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(fmt.Errorf("can't read contract manifest: %w", err), 1)
|
return cli.NewExitError(fmt.Errorf("can't read contract manifest: %w", err), 1)
|
||||||
|
|
|
@ -3,6 +3,7 @@ package smartcontract
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
|
@ -150,7 +151,9 @@ callflags:
|
||||||
"--hash", h.StringLE(),
|
"--hash", h.StringLE(),
|
||||||
}))
|
}))
|
||||||
|
|
||||||
const expected = `// Package wrapper contains wrappers for MyContract contract.
|
const expected = `// Code generated by neo-go contract generate-wrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
|
// Package wrapper contains wrappers for MyContract contract.
|
||||||
package wrapper
|
package wrapper
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
@ -171,8 +174,8 @@ func Sum(first int, second int) int {
|
||||||
return neogointernal.CallWithToken(Hash, "sum", int(contract.All), first, second).(int)
|
return neogointernal.CallWithToken(Hash, "sum", int(contract.All), first, second).(int)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Sum_3 invokes ` + "`sum`" + ` method of contract.
|
// Sum2 invokes ` + "`sum`" + ` method of contract.
|
||||||
func Sum_3(first int, second int, third int) int {
|
func Sum2(first int, second int, third int) int {
|
||||||
return neogointernal.CallWithToken(Hash, "sum", int(contract.All), first, second, third).(int)
|
return neogointernal.CallWithToken(Hash, "sum", int(contract.All), first, second, third).(int)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -187,7 +190,7 @@ func Zum(typev int, typev_ int, funcv int) int {
|
||||||
}
|
}
|
||||||
|
|
||||||
// JustExecute invokes ` + "`justExecute`" + ` method of contract.
|
// JustExecute invokes ` + "`justExecute`" + ` method of contract.
|
||||||
func JustExecute(arr []interface{}) {
|
func JustExecute(arr []any) {
|
||||||
neogointernal.CallWithTokenNoRet(Hash, "justExecute", int(contract.All), arr)
|
neogointernal.CallWithTokenNoRet(Hash, "justExecute", int(contract.All), arr)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -197,7 +200,7 @@ func GetPublicKey() interop.PublicKey {
|
||||||
}
|
}
|
||||||
|
|
||||||
// OtherTypes invokes ` + "`otherTypes`" + ` method of contract.
|
// OtherTypes invokes ` + "`otherTypes`" + ` method of contract.
|
||||||
func OtherTypes(ctr interop.Hash160, tx interop.Hash256, sig interop.Signature, data interface{}) bool {
|
func OtherTypes(ctr interop.Hash160, tx interop.Hash256, sig interop.Signature, data any) bool {
|
||||||
return neogointernal.CallWithToken(Hash, "otherTypes", int(contract.All), ctr, tx, sig, data).(bool)
|
return neogointernal.CallWithToken(Hash, "otherTypes", int(contract.All), ctr, tx, sig, data).(bool)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -212,8 +215,8 @@ func GetFromMap(intMap map[string]int, indices []string) []int {
|
||||||
}
|
}
|
||||||
|
|
||||||
// DoSomething invokes ` + "`doSomething`" + ` method of contract.
|
// DoSomething invokes ` + "`doSomething`" + ` method of contract.
|
||||||
func DoSomething(bytes []byte, str string) interface{} {
|
func DoSomething(bytes []byte, str string) any {
|
||||||
return neogointernal.CallWithToken(Hash, "doSomething", int(contract.ReadStates), bytes, str).(interface{})
|
return neogointernal.CallWithToken(Hash, "doSomething", int(contract.ReadStates), bytes, str).(any)
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetBlockWrapper invokes ` + "`getBlockWrapper`" + ` method of contract.
|
// GetBlockWrapper invokes ` + "`getBlockWrapper`" + ` method of contract.
|
||||||
|
@ -230,6 +233,99 @@ func MyFunc(in map[int]mycontract.Input) []mycontract.Output {
|
||||||
data, err := os.ReadFile(outFile)
|
data, err := os.ReadFile(outFile)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.Equal(t, expected, string(data))
|
require.Equal(t, expected, string(data))
|
||||||
|
|
||||||
|
require.NoError(t, app.Run([]string{"", "generate-wrapper",
|
||||||
|
"--manifest", manifestFile,
|
||||||
|
"--config", cfgPath,
|
||||||
|
"--out", outFile,
|
||||||
|
}))
|
||||||
|
expectedWithDynamicHash := `// Code generated by neo-go contract generate-wrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
|
// Package wrapper contains wrappers for MyContract contract.
|
||||||
|
package wrapper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/heyitsme/mycontract"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/interop"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/interop/contract"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/interop/iterator"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/interop/native/ledger"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/interop/storage"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Contract represents the MyContract smart contract.
|
||||||
|
type Contract struct {
|
||||||
|
Hash interop.Hash160
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewContract returns a new Contract instance with the specified hash.
|
||||||
|
func NewContract(hash interop.Hash160) Contract {
|
||||||
|
return Contract{Hash: hash}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sum invokes ` + "`sum`" + ` method of contract.
|
||||||
|
func (c Contract) Sum(first int, second int) int {
|
||||||
|
return contract.Call(c.Hash, "sum", contract.All, first, second).(int)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sum2 invokes ` + "`sum`" + ` method of contract.
|
||||||
|
func (c Contract) Sum2(first int, second int, third int) int {
|
||||||
|
return contract.Call(c.Hash, "sum", contract.All, first, second, third).(int)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sum3 invokes ` + "`sum3`" + ` method of contract.
|
||||||
|
func (c Contract) Sum3() int {
|
||||||
|
return contract.Call(c.Hash, "sum3", contract.ReadOnly).(int)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Zum invokes ` + "`zum`" + ` method of contract.
|
||||||
|
func (c Contract) Zum(typev int, typev_ int, funcv int) int {
|
||||||
|
return contract.Call(c.Hash, "zum", contract.All, typev, typev_, funcv).(int)
|
||||||
|
}
|
||||||
|
|
||||||
|
// JustExecute invokes ` + "`justExecute`" + ` method of contract.
|
||||||
|
func (c Contract) JustExecute(arr []any) {
|
||||||
|
contract.Call(c.Hash, "justExecute", contract.All, arr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetPublicKey invokes ` + "`getPublicKey`" + ` method of contract.
|
||||||
|
func (c Contract) GetPublicKey() interop.PublicKey {
|
||||||
|
return contract.Call(c.Hash, "getPublicKey", contract.All).(interop.PublicKey)
|
||||||
|
}
|
||||||
|
|
||||||
|
// OtherTypes invokes ` + "`otherTypes`" + ` method of contract.
|
||||||
|
func (c Contract) OtherTypes(ctr interop.Hash160, tx interop.Hash256, sig interop.Signature, data any) bool {
|
||||||
|
return contract.Call(c.Hash, "otherTypes", contract.All, ctr, tx, sig, data).(bool)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SearchStorage invokes ` + "`searchStorage`" + ` method of contract.
|
||||||
|
func (c Contract) SearchStorage(ctx storage.Context) iterator.Iterator {
|
||||||
|
return contract.Call(c.Hash, "searchStorage", contract.All, ctx).(iterator.Iterator)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetFromMap invokes ` + "`getFromMap`" + ` method of contract.
|
||||||
|
func (c Contract) GetFromMap(intMap map[string]int, indices []string) []int {
|
||||||
|
return contract.Call(c.Hash, "getFromMap", contract.All, intMap, indices).([]int)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DoSomething invokes ` + "`doSomething`" + ` method of contract.
|
||||||
|
func (c Contract) DoSomething(bytes []byte, str string) any {
|
||||||
|
return contract.Call(c.Hash, "doSomething", contract.ReadStates, bytes, str).(any)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetBlockWrapper invokes ` + "`getBlockWrapper`" + ` method of contract.
|
||||||
|
func (c Contract) GetBlockWrapper() ledger.Block {
|
||||||
|
return contract.Call(c.Hash, "getBlockWrapper", contract.All).(ledger.Block)
|
||||||
|
}
|
||||||
|
|
||||||
|
// MyFunc invokes ` + "`myFunc`" + ` method of contract.
|
||||||
|
func (c Contract) MyFunc(in map[int]mycontract.Input) []mycontract.Output {
|
||||||
|
return contract.Call(c.Hash, "myFunc", contract.All, in).([]mycontract.Output)
|
||||||
|
}
|
||||||
|
`
|
||||||
|
data, err = os.ReadFile(outFile)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, expectedWithDynamicHash, string(data))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestGenerateValidPackageName(t *testing.T) {
|
func TestGenerateValidPackageName(t *testing.T) {
|
||||||
|
@ -264,7 +360,9 @@ func TestGenerateValidPackageName(t *testing.T) {
|
||||||
|
|
||||||
data, err := os.ReadFile(outFile)
|
data, err := os.ReadFile(outFile)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.Equal(t, `// Package myspacecontract contains wrappers for My space contract contract.
|
require.Equal(t, `// Code generated by neo-go contract generate-wrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
|
// Package myspacecontract contains wrappers for My space contract contract.
|
||||||
package myspacecontract
|
package myspacecontract
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
@ -288,7 +386,9 @@ func Get() int {
|
||||||
|
|
||||||
data, err = os.ReadFile(outFile)
|
data, err = os.ReadFile(outFile)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.Equal(t, `// Package myspacecontract contains RPC wrappers for My space contract contract.
|
require.Equal(t, `// Code generated by neo-go contract generate-rpcwrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
|
// Package myspacecontract contains RPC wrappers for My space contract contract.
|
||||||
package myspacecontract
|
package myspacecontract
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
@ -303,27 +403,32 @@ var Hash = util.Uint160{0x4, 0x8, 0x15, 0x16, 0x23, 0x42, 0x43, 0x44, 0x0, 0x1,
|
||||||
|
|
||||||
// Invoker is used by ContractReader to call various safe methods.
|
// Invoker is used by ContractReader to call various safe methods.
|
||||||
type Invoker interface {
|
type Invoker interface {
|
||||||
Call(contract util.Uint160, operation string, params ...interface{}) (*result.Invoke, error)
|
Call(contract util.Uint160, operation string, params ...any) (*result.Invoke, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ContractReader implements safe contract methods.
|
// ContractReader implements safe contract methods.
|
||||||
type ContractReader struct {
|
type ContractReader struct {
|
||||||
invoker Invoker
|
invoker Invoker
|
||||||
|
hash util.Uint160
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewReader creates an instance of ContractReader using Hash and the given Invoker.
|
// NewReader creates an instance of ContractReader using Hash and the given Invoker.
|
||||||
func NewReader(invoker Invoker) *ContractReader {
|
func NewReader(invoker Invoker) *ContractReader {
|
||||||
return &ContractReader{invoker}
|
var hash = Hash
|
||||||
|
return &ContractReader{invoker, hash}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Get invokes `+"`get`"+` method of contract.
|
// Get invokes `+"`get`"+` method of contract.
|
||||||
func (c *ContractReader) Get() (*big.Int, error) {
|
func (c *ContractReader) Get() (*big.Int, error) {
|
||||||
return unwrap.BigInt(c.invoker.Call(Hash, "get"))
|
return unwrap.BigInt(c.invoker.Call(c.hash, "get"))
|
||||||
}
|
}
|
||||||
`, string(data))
|
`, string(data))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// rewriteExpectedOutputs denotes whether expected output files should be rewritten
|
||||||
|
// for TestGenerateRPCBindings and TestAssistedRPCBindings.
|
||||||
|
const rewriteExpectedOutputs = false
|
||||||
|
|
||||||
func TestGenerateRPCBindings(t *testing.T) {
|
func TestGenerateRPCBindings(t *testing.T) {
|
||||||
tmpDir := t.TempDir()
|
tmpDir := t.TempDir()
|
||||||
app := cli.NewApp()
|
app := cli.NewApp()
|
||||||
|
@ -341,10 +446,14 @@ func TestGenerateRPCBindings(t *testing.T) {
|
||||||
data, err := os.ReadFile(outFile)
|
data, err := os.ReadFile(outFile)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
data = bytes.ReplaceAll(data, []byte("\r"), []byte{}) // Windows.
|
data = bytes.ReplaceAll(data, []byte("\r"), []byte{}) // Windows.
|
||||||
|
if rewriteExpectedOutputs {
|
||||||
|
require.NoError(t, os.WriteFile(good, data, os.ModePerm))
|
||||||
|
} else {
|
||||||
expected, err := os.ReadFile(good)
|
expected, err := os.ReadFile(good)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
expected = bytes.ReplaceAll(expected, []byte("\r"), []byte{}) // Windows.
|
expected = bytes.ReplaceAll(expected, []byte("\r"), []byte{}) // Windows.
|
||||||
require.Equal(t, string(expected), string(data))
|
require.Equal(t, string(expected), string(data))
|
||||||
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -363,6 +472,8 @@ func TestGenerateRPCBindings(t *testing.T) {
|
||||||
checkBinding(filepath.Join("testdata", "nonepiter", "iter.manifest.json"),
|
checkBinding(filepath.Join("testdata", "nonepiter", "iter.manifest.json"),
|
||||||
"0x00112233445566778899aabbccddeeff00112233",
|
"0x00112233445566778899aabbccddeeff00112233",
|
||||||
filepath.Join("testdata", "nonepiter", "iter.go"))
|
filepath.Join("testdata", "nonepiter", "iter.go"))
|
||||||
|
|
||||||
|
require.False(t, rewriteExpectedOutputs)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestAssistedRPCBindings(t *testing.T) {
|
func TestAssistedRPCBindings(t *testing.T) {
|
||||||
|
@ -370,38 +481,70 @@ func TestAssistedRPCBindings(t *testing.T) {
|
||||||
app := cli.NewApp()
|
app := cli.NewApp()
|
||||||
app.Commands = NewCommands()
|
app.Commands = NewCommands()
|
||||||
|
|
||||||
var checkBinding = func(source string) {
|
var checkBinding = func(source string, hasDefinedHash bool, guessEventTypes bool, suffix ...string) {
|
||||||
t.Run(source, func(t *testing.T) {
|
testName := source
|
||||||
|
if len(suffix) != 0 {
|
||||||
|
testName += suffix[0]
|
||||||
|
}
|
||||||
|
testName += fmt.Sprintf(", predefined hash: %t", hasDefinedHash)
|
||||||
|
t.Run(testName, func(t *testing.T) {
|
||||||
|
outFile := filepath.Join(tmpDir, "out.go")
|
||||||
|
configFile := filepath.Join(source, "config.yml")
|
||||||
|
expectedFile := filepath.Join(source, "rpcbindings.out")
|
||||||
|
if len(suffix) != 0 {
|
||||||
|
configFile = filepath.Join(source, "config"+suffix[0]+".yml")
|
||||||
|
expectedFile = filepath.Join(source, "rpcbindings"+suffix[0]+".out")
|
||||||
|
} else if !hasDefinedHash {
|
||||||
|
expectedFile = filepath.Join(source, "rpcbindings_dynamic_hash.out")
|
||||||
|
}
|
||||||
manifestF := filepath.Join(tmpDir, "manifest.json")
|
manifestF := filepath.Join(tmpDir, "manifest.json")
|
||||||
bindingF := filepath.Join(tmpDir, "binding.yml")
|
bindingF := filepath.Join(tmpDir, "binding.yml")
|
||||||
nefF := filepath.Join(tmpDir, "out.nef")
|
nefF := filepath.Join(tmpDir, "out.nef")
|
||||||
require.NoError(t, app.Run([]string{"", "contract", "compile",
|
cmd := []string{"", "contract", "compile",
|
||||||
"--in", source,
|
"--in", source,
|
||||||
"--config", filepath.Join(source, "config.yml"),
|
"--config", configFile,
|
||||||
"--manifest", manifestF,
|
"--manifest", manifestF,
|
||||||
"--bindings", bindingF,
|
"--bindings", bindingF,
|
||||||
"--out", nefF,
|
"--out", nefF,
|
||||||
}))
|
}
|
||||||
outFile := filepath.Join(tmpDir, "out.go")
|
if guessEventTypes {
|
||||||
require.NoError(t, app.Run([]string{"", "contract", "generate-rpcwrapper",
|
cmd = append(cmd, "--guess-eventtypes")
|
||||||
|
}
|
||||||
|
require.NoError(t, app.Run(cmd))
|
||||||
|
|
||||||
|
cmds := []string{"", "contract", "generate-rpcwrapper",
|
||||||
"--config", bindingF,
|
"--config", bindingF,
|
||||||
"--manifest", manifestF,
|
"--manifest", manifestF,
|
||||||
"--out", outFile,
|
"--out", outFile,
|
||||||
"--hash", "0x00112233445566778899aabbccddeeff00112233",
|
}
|
||||||
}))
|
if hasDefinedHash {
|
||||||
|
cmds = append(cmds, "--hash", "0x00112233445566778899aabbccddeeff00112233")
|
||||||
|
}
|
||||||
|
require.NoError(t, app.Run(cmds))
|
||||||
|
|
||||||
data, err := os.ReadFile(outFile)
|
data, err := os.ReadFile(outFile)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
data = bytes.ReplaceAll(data, []byte("\r"), []byte{}) // Windows.
|
data = bytes.ReplaceAll(data, []byte("\r"), []byte{}) // Windows.
|
||||||
expected, err := os.ReadFile(filepath.Join(source, "rpcbindings.out"))
|
if rewriteExpectedOutputs {
|
||||||
|
require.NoError(t, os.WriteFile(expectedFile, data, os.ModePerm))
|
||||||
|
} else {
|
||||||
|
expected, err := os.ReadFile(expectedFile)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
expected = bytes.ReplaceAll(expected, []byte("\r"), []byte{}) // Windows.
|
expected = bytes.ReplaceAll(expected, []byte("\r"), []byte{}) // Windows.
|
||||||
require.Equal(t, string(expected), string(data))
|
require.Equal(t, string(expected), string(data))
|
||||||
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
checkBinding(filepath.Join("testdata", "types"))
|
for _, hasDefinedHash := range []bool{true, false} {
|
||||||
checkBinding(filepath.Join("testdata", "structs"))
|
checkBinding(filepath.Join("testdata", "rpcbindings", "types"), hasDefinedHash, false)
|
||||||
|
checkBinding(filepath.Join("testdata", "rpcbindings", "structs"), hasDefinedHash, false)
|
||||||
|
}
|
||||||
|
checkBinding(filepath.Join("testdata", "rpcbindings", "notifications"), true, false)
|
||||||
|
checkBinding(filepath.Join("testdata", "rpcbindings", "notifications"), true, false, "_extended")
|
||||||
|
checkBinding(filepath.Join("testdata", "rpcbindings", "notifications"), true, true, "_guessed")
|
||||||
|
|
||||||
|
require.False(t, rewriteExpectedOutputs)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestGenerate_Errors(t *testing.T) {
|
func TestGenerate_Errors(t *testing.T) {
|
||||||
|
@ -467,3 +610,103 @@ callflags:
|
||||||
"--config", cfgPath, "--out", "zzz")
|
"--config", cfgPath, "--out", "zzz")
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestCompile_GuessEventTypes(t *testing.T) {
|
||||||
|
app := cli.NewApp()
|
||||||
|
app.Commands = NewCommands()
|
||||||
|
app.ExitErrHandler = func(*cli.Context, error) {}
|
||||||
|
|
||||||
|
checkError := func(t *testing.T, msg string, args ...string) {
|
||||||
|
// cli.ExitError doesn't implement wraping properly, so we check for an error message.
|
||||||
|
err := app.Run(args)
|
||||||
|
require.Error(t, err)
|
||||||
|
require.True(t, strings.Contains(err.Error(), msg), "got: %v", err)
|
||||||
|
}
|
||||||
|
check := func(t *testing.T, source string, expectedErrText string) {
|
||||||
|
tmpDir := t.TempDir()
|
||||||
|
configFile := filepath.Join(source, "invalid.yml")
|
||||||
|
manifestF := filepath.Join(tmpDir, "invalid.manifest.json")
|
||||||
|
bindingF := filepath.Join(tmpDir, "invalid.binding.yml")
|
||||||
|
nefF := filepath.Join(tmpDir, "invalid.out.nef")
|
||||||
|
cmd := []string{"", "contract", "compile",
|
||||||
|
"--in", source,
|
||||||
|
"--config", configFile,
|
||||||
|
"--manifest", manifestF,
|
||||||
|
"--bindings", bindingF,
|
||||||
|
"--out", nefF,
|
||||||
|
"--guess-eventtypes",
|
||||||
|
}
|
||||||
|
checkError(t, expectedErrText, cmd...)
|
||||||
|
}
|
||||||
|
|
||||||
|
t.Run("not declared in manifest", func(t *testing.T) {
|
||||||
|
check(t, filepath.Join("testdata", "rpcbindings", "invalid1"), "inconsistent usages of event `Non declared event`: not declared in the contract config")
|
||||||
|
})
|
||||||
|
t.Run("invalid number of params", func(t *testing.T) {
|
||||||
|
check(t, filepath.Join("testdata", "rpcbindings", "invalid2"), "inconsistent usages of event `SomeEvent` against config: number of params mismatch: 2 vs 1")
|
||||||
|
})
|
||||||
|
/*
|
||||||
|
// TODO: this on is a controversial one. If event information is provided in the config file, then conversion code
|
||||||
|
// will be emitted by the compiler according to the parameter type provided via config. Thus, we can be sure that
|
||||||
|
// either event parameter has the type specified in the config file or the execution of the contract will fail.
|
||||||
|
// Thus, this testcase is always failing (no compilation error occures).
|
||||||
|
// Question: do we want to compare `RealType` of the emitted parameter with the one expected in the manifest?
|
||||||
|
t.Run("SC parameter type mismatch", func(t *testing.T) {
|
||||||
|
check(t, filepath.Join("testdata", "rpcbindings", "invalid3"), "inconsistent usages of event `SomeEvent` against config: number of params mismatch: 2 vs 1")
|
||||||
|
})
|
||||||
|
*/
|
||||||
|
t.Run("extended types mismatch", func(t *testing.T) {
|
||||||
|
check(t, filepath.Join("testdata", "rpcbindings", "invalid4"), "inconsistent usages of event `SomeEvent`: extended type of param #0 mismatch")
|
||||||
|
})
|
||||||
|
t.Run("named types redeclare", func(t *testing.T) {
|
||||||
|
check(t, filepath.Join("testdata", "rpcbindings", "invalid5"), "configured declared named type intersects with the contract's one: `invalid5.NamedStruct`")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGenerateRPCBindings_Errors(t *testing.T) {
|
||||||
|
app := cli.NewApp()
|
||||||
|
app.Commands = NewCommands()
|
||||||
|
app.ExitErrHandler = func(*cli.Context, error) {}
|
||||||
|
|
||||||
|
t.Run("duplicating resulting fields", func(t *testing.T) {
|
||||||
|
check := func(t *testing.T, packageName string, autogen bool, expectedError string) {
|
||||||
|
tmpDir := t.TempDir()
|
||||||
|
source := filepath.Join("testdata", "rpcbindings", packageName)
|
||||||
|
configFile := filepath.Join(source, "invalid.yml")
|
||||||
|
out := filepath.Join(tmpDir, "rpcbindings.out")
|
||||||
|
manifestF := filepath.Join(tmpDir, "manifest.json")
|
||||||
|
bindingF := filepath.Join(tmpDir, "binding.yml")
|
||||||
|
nefF := filepath.Join(tmpDir, "out.nef")
|
||||||
|
cmd := []string{"", "contract", "compile",
|
||||||
|
"--in", source,
|
||||||
|
"--config", configFile,
|
||||||
|
"--manifest", manifestF,
|
||||||
|
"--bindings", bindingF,
|
||||||
|
"--out", nefF,
|
||||||
|
}
|
||||||
|
if autogen {
|
||||||
|
cmd = append(cmd, "--guess-eventtypes")
|
||||||
|
}
|
||||||
|
require.NoError(t, app.Run(cmd))
|
||||||
|
|
||||||
|
cmds := []string{"", "contract", "generate-rpcwrapper",
|
||||||
|
"--config", bindingF,
|
||||||
|
"--manifest", manifestF,
|
||||||
|
"--out", out,
|
||||||
|
}
|
||||||
|
err := app.Run(cmds)
|
||||||
|
require.Error(t, err)
|
||||||
|
require.True(t, strings.Contains(err.Error(), expectedError), err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
t.Run("event", func(t *testing.T) {
|
||||||
|
check(t, "invalid6", false, "error during generation: named type `SomeStruct` has two fields with identical resulting binding name `Field`")
|
||||||
|
})
|
||||||
|
t.Run("autogen event", func(t *testing.T) {
|
||||||
|
check(t, "invalid7", true, "error during generation: named type `invalid7.SomeStruct` has two fields with identical resulting binding name `Field`")
|
||||||
|
})
|
||||||
|
t.Run("struct", func(t *testing.T) {
|
||||||
|
check(t, "invalid8", false, "error during generation: named type `invalid8.SomeStruct` has two fields with identical resulting binding name `Field`")
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
|
@ -8,6 +8,7 @@ import (
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/cli/cmdargs"
|
"github.com/nspcc-dev/neo-go/cli/cmdargs"
|
||||||
"github.com/nspcc-dev/neo-go/cli/flags"
|
"github.com/nspcc-dev/neo-go/cli/flags"
|
||||||
|
"github.com/nspcc-dev/neo-go/cli/options"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract/nef"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/nef"
|
||||||
|
@ -37,7 +38,7 @@ func manifestAddGroup(ctx *cli.Context) error {
|
||||||
|
|
||||||
h := state.CreateContractHash(sender, nf.Checksum, m.Name)
|
h := state.CreateContractHash(sender, nf.Checksum, m.Name)
|
||||||
|
|
||||||
gAcc, w, err := getAccFromContext(ctx)
|
gAcc, w, err := options.GetAccFromContext(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(fmt.Errorf("can't get account to sign group with: %w", err), 1)
|
return cli.NewExitError(fmt.Errorf("can't get account to sign group with: %w", err), 1)
|
||||||
}
|
}
|
||||||
|
@ -109,7 +110,7 @@ func readManifest(filename string, hash util.Uint160) (*manifest.Manifest, []byt
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, err
|
return nil, nil, err
|
||||||
}
|
}
|
||||||
if err := m.IsValid(hash); err != nil {
|
if err := m.IsValid(hash, true); err != nil {
|
||||||
return nil, nil, fmt.Errorf("manifest is invalid: %w", err)
|
return nil, nil, fmt.Errorf("manifest is invalid: %w", err)
|
||||||
}
|
}
|
||||||
return m, manifestBytes, nil
|
return m, manifestBytes, nil
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
package smartcontract
|
package smartcontract
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/hex"
|
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
|
@ -19,7 +18,7 @@ const (
|
||||||
permMethodKey = "methods"
|
permMethodKey = "methods"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (p permission) MarshalYAML() (interface{}, error) {
|
func (p permission) MarshalYAML() (any, error) {
|
||||||
m := yaml.Node{Kind: yaml.MappingNode}
|
m := yaml.Node{Kind: yaml.MappingNode}
|
||||||
switch p.Contract.Type {
|
switch p.Contract.Type {
|
||||||
case manifest.PermissionWildcard:
|
case manifest.PermissionWildcard:
|
||||||
|
@ -28,15 +27,14 @@ func (p permission) MarshalYAML() (interface{}, error) {
|
||||||
&yaml.Node{Kind: yaml.ScalarNode, Value: permHashKey},
|
&yaml.Node{Kind: yaml.ScalarNode, Value: permHashKey},
|
||||||
&yaml.Node{Kind: yaml.ScalarNode, Value: p.Contract.Value.(util.Uint160).StringLE()})
|
&yaml.Node{Kind: yaml.ScalarNode, Value: p.Contract.Value.(util.Uint160).StringLE()})
|
||||||
case manifest.PermissionGroup:
|
case manifest.PermissionGroup:
|
||||||
bs := p.Contract.Value.(*keys.PublicKey).Bytes()
|
|
||||||
m.Content = append(m.Content,
|
m.Content = append(m.Content,
|
||||||
&yaml.Node{Kind: yaml.ScalarNode, Value: permGroupKey},
|
&yaml.Node{Kind: yaml.ScalarNode, Value: permGroupKey},
|
||||||
&yaml.Node{Kind: yaml.ScalarNode, Value: hex.EncodeToString(bs)})
|
&yaml.Node{Kind: yaml.ScalarNode, Value: p.Contract.Value.(*keys.PublicKey).StringCompressed()})
|
||||||
default:
|
default:
|
||||||
return nil, fmt.Errorf("invalid permission type: %d", p.Contract.Type)
|
return nil, fmt.Errorf("invalid permission type: %d", p.Contract.Type)
|
||||||
}
|
}
|
||||||
|
|
||||||
var val interface{} = "*"
|
var val any = "*"
|
||||||
if !p.Methods.IsWildcard() {
|
if !p.Methods.IsWildcard() {
|
||||||
val = p.Methods.Value
|
val = p.Methods.Value
|
||||||
}
|
}
|
||||||
|
@ -53,8 +51,8 @@ func (p permission) MarshalYAML() (interface{}, error) {
|
||||||
return m, nil
|
return m, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *permission) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
func (p *permission) UnmarshalYAML(unmarshal func(any) error) error {
|
||||||
var m map[string]interface{}
|
var m map[string]any
|
||||||
if err := unmarshal(&m); err != nil {
|
if err := unmarshal(&m); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -66,7 +64,7 @@ func (p *permission) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
||||||
return p.fillMethods(m)
|
return p.fillMethods(m)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *permission) fillType(m map[string]interface{}) error {
|
func (p *permission) fillType(m map[string]any) error {
|
||||||
vh, ok1 := m[permHashKey]
|
vh, ok1 := m[permHashKey]
|
||||||
vg, ok2 := m[permGroupKey]
|
vg, ok2 := m[permGroupKey]
|
||||||
switch {
|
switch {
|
||||||
|
@ -104,7 +102,7 @@ func (p *permission) fillType(m map[string]interface{}) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *permission) fillMethods(m map[string]interface{}) error {
|
func (p *permission) fillMethods(m map[string]any) error {
|
||||||
methods, ok := m[permMethodKey]
|
methods, ok := m[permMethodKey]
|
||||||
if !ok {
|
if !ok {
|
||||||
return errors.New("'methods' field is missing from permission")
|
return errors.New("'methods' field is missing from permission")
|
||||||
|
@ -116,7 +114,7 @@ func (p *permission) fillMethods(m map[string]interface{}) error {
|
||||||
p.Methods.Value = nil
|
p.Methods.Value = nil
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
case []interface{}:
|
case []any:
|
||||||
ms := make([]string, len(mt))
|
ms := make([]string, len(mt))
|
||||||
for i := range mt {
|
for i := range mt {
|
||||||
ms[i], ok = mt[i].(string)
|
ms[i], ok = mt[i].(string)
|
||||||
|
|
|
@ -11,19 +11,17 @@ import (
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/cli/cmdargs"
|
"github.com/nspcc-dev/neo-go/cli/cmdargs"
|
||||||
"github.com/nspcc-dev/neo-go/cli/flags"
|
"github.com/nspcc-dev/neo-go/cli/flags"
|
||||||
"github.com/nspcc-dev/neo-go/cli/input"
|
|
||||||
"github.com/nspcc-dev/neo-go/cli/options"
|
"github.com/nspcc-dev/neo-go/cli/options"
|
||||||
"github.com/nspcc-dev/neo-go/cli/txctx"
|
"github.com/nspcc-dev/neo-go/cli/txctx"
|
||||||
cliwallet "github.com/nspcc-dev/neo-go/cli/wallet"
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/compiler"
|
"github.com/nspcc-dev/neo-go/pkg/compiler"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/neorpc/result"
|
"github.com/nspcc-dev/neo-go/pkg/neorpc/result"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/rpcclient/actor"
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/actor"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/rpcclient/invoker"
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/invoker"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/rpcclient/management"
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/management"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/binding"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract/nef"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/nef"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/util"
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
|
@ -42,20 +40,9 @@ var (
|
||||||
errNoConfFile = errors.New("no config file was found, specify a config file with the '--config' or '-c' flag")
|
errNoConfFile = errors.New("no config file was found, specify a config file with the '--config' or '-c' flag")
|
||||||
errNoManifestFile = errors.New("no manifest file was found, specify manifest file with '--manifest' or '-m' flag")
|
errNoManifestFile = errors.New("no manifest file was found, specify manifest file with '--manifest' or '-m' flag")
|
||||||
errNoMethod = errors.New("no method specified for function invocation command")
|
errNoMethod = errors.New("no method specified for function invocation command")
|
||||||
errNoWallet = errors.New("no wallet parameter found, specify it with the '--wallet' or '-w' flag or specify wallet config file with the '--wallet-config' flag")
|
|
||||||
errConflictingWalletFlags = errors.New("--wallet flag conflicts with --wallet-config flag, please, provide one of them to specify wallet location")
|
|
||||||
errNoScriptHash = errors.New("no smart contract hash was provided, specify one as the first argument")
|
errNoScriptHash = errors.New("no smart contract hash was provided, specify one as the first argument")
|
||||||
errNoSmartContractName = errors.New("no name was provided, specify the '--name or -n' flag")
|
errNoSmartContractName = errors.New("no name was provided, specify the '--name or -n' flag")
|
||||||
errFileExist = errors.New("A file with given smart-contract name already exists")
|
errFileExist = errors.New("A file with given smart-contract name already exists")
|
||||||
|
|
||||||
walletFlag = cli.StringFlag{
|
|
||||||
Name: "wallet, w",
|
|
||||||
Usage: "wallet to use to get the key for transaction signing; conflicts with --wallet-config flag",
|
|
||||||
}
|
|
||||||
walletConfigFlag = cli.StringFlag{
|
|
||||||
Name: "wallet-config",
|
|
||||||
Usage: "path to wallet config to use to get the key for transaction signing; conflicts with --wallet flag",
|
|
||||||
}
|
|
||||||
addressFlag = flags.AddressFlag{
|
addressFlag = flags.AddressFlag{
|
||||||
Name: addressFlagName,
|
Name: addressFlagName,
|
||||||
Usage: "address to use as transaction signee (and gas source)",
|
Usage: "address to use as transaction signee (and gas source)",
|
||||||
|
@ -81,7 +68,7 @@ func init() {
|
||||||
}
|
}
|
||||||
|
|
||||||
// RuntimeNotify sends runtime notification with "Hello world!" name
|
// RuntimeNotify sends runtime notification with "Hello world!" name
|
||||||
func RuntimeNotify(args []interface{}) {
|
func RuntimeNotify(args []any) {
|
||||||
runtime.Notify(notificationName, args)
|
runtime.Notify(notificationName, args)
|
||||||
}`
|
}`
|
||||||
)
|
)
|
||||||
|
@ -99,14 +86,14 @@ func NewCommands() []cli.Command {
|
||||||
testInvokeFunctionFlags := []cli.Flag{options.Historic}
|
testInvokeFunctionFlags := []cli.Flag{options.Historic}
|
||||||
testInvokeFunctionFlags = append(testInvokeFunctionFlags, options.RPC...)
|
testInvokeFunctionFlags = append(testInvokeFunctionFlags, options.RPC...)
|
||||||
invokeFunctionFlags := []cli.Flag{
|
invokeFunctionFlags := []cli.Flag{
|
||||||
walletFlag,
|
|
||||||
walletConfigFlag,
|
|
||||||
addressFlag,
|
addressFlag,
|
||||||
txctx.GasFlag,
|
txctx.GasFlag,
|
||||||
txctx.SysGasFlag,
|
txctx.SysGasFlag,
|
||||||
txctx.OutFlag,
|
txctx.OutFlag,
|
||||||
txctx.ForceFlag,
|
txctx.ForceFlag,
|
||||||
|
txctx.AwaitFlag,
|
||||||
}
|
}
|
||||||
|
invokeFunctionFlags = append(invokeFunctionFlags, options.Wallet...)
|
||||||
invokeFunctionFlags = append(invokeFunctionFlags, options.RPC...)
|
invokeFunctionFlags = append(invokeFunctionFlags, options.RPC...)
|
||||||
deployFlags := append(invokeFunctionFlags, []cli.Flag{
|
deployFlags := append(invokeFunctionFlags, []cli.Flag{
|
||||||
cli.StringFlag{
|
cli.StringFlag{
|
||||||
|
@ -118,6 +105,24 @@ func NewCommands() []cli.Command {
|
||||||
Usage: "Manifest input file (*.manifest.json)",
|
Usage: "Manifest input file (*.manifest.json)",
|
||||||
},
|
},
|
||||||
}...)
|
}...)
|
||||||
|
manifestAddGroupFlags := append([]cli.Flag{
|
||||||
|
cli.StringFlag{
|
||||||
|
Name: "sender, s",
|
||||||
|
Usage: "deploy transaction sender",
|
||||||
|
},
|
||||||
|
flags.AddressFlag{
|
||||||
|
Name: addressFlagName, // use the same name for handler code unification.
|
||||||
|
Usage: "account to sign group with",
|
||||||
|
},
|
||||||
|
cli.StringFlag{
|
||||||
|
Name: "nef, n",
|
||||||
|
Usage: "path to the NEF file",
|
||||||
|
},
|
||||||
|
cli.StringFlag{
|
||||||
|
Name: "manifest, m",
|
||||||
|
Usage: "path to the manifest",
|
||||||
|
},
|
||||||
|
}, options.Wallet...)
|
||||||
return []cli.Command{{
|
return []cli.Command{{
|
||||||
Name: "contract",
|
Name: "contract",
|
||||||
Usage: "compile - debug - deploy smart contracts",
|
Usage: "compile - debug - deploy smart contracts",
|
||||||
|
@ -125,12 +130,20 @@ func NewCommands() []cli.Command {
|
||||||
{
|
{
|
||||||
Name: "compile",
|
Name: "compile",
|
||||||
Usage: "compile a smart contract to a .nef file",
|
Usage: "compile a smart contract to a .nef file",
|
||||||
UsageText: "neo-go contract compile -i path [-o nef] [-v] [-d] [-m manifest] [-c yaml] [--bindings file] [--no-standards] [--no-events] [--no-permissions]",
|
UsageText: "neo-go contract compile -i path [-o nef] [-v] [-d] [-m manifest] [-c yaml] [--bindings file] [--no-standards] [--no-events] [--no-permissions] [--guess-eventtypes]",
|
||||||
|
Description: `Compiles given smart contract to a .nef file and emits other associated
|
||||||
|
information (manifest, bindings configuration, debug information files) if
|
||||||
|
asked to. If none of --out, --manifest, --config, --bindings flags are specified,
|
||||||
|
then the output filenames for these flags will be guessed using the contract
|
||||||
|
name or path provided via --in option by trimming/adding corresponding suffixes
|
||||||
|
to the common part of the path. In the latter case the configuration filepath
|
||||||
|
will be guessed from the --in option using the same rule.
|
||||||
|
`,
|
||||||
Action: contractCompile,
|
Action: contractCompile,
|
||||||
Flags: []cli.Flag{
|
Flags: []cli.Flag{
|
||||||
cli.StringFlag{
|
cli.StringFlag{
|
||||||
Name: "in, i",
|
Name: "in, i",
|
||||||
Usage: "Input file for the smart contract to be compiled",
|
Usage: "Input file for the smart contract to be compiled (*.go file or directory)",
|
||||||
},
|
},
|
||||||
cli.StringFlag{
|
cli.StringFlag{
|
||||||
Name: "out, o",
|
Name: "out, o",
|
||||||
|
@ -164,6 +177,10 @@ func NewCommands() []cli.Command {
|
||||||
Name: "no-permissions",
|
Name: "no-permissions",
|
||||||
Usage: "do not check if invoked contracts are allowed in manifest",
|
Usage: "do not check if invoked contracts are allowed in manifest",
|
||||||
},
|
},
|
||||||
|
cli.BoolFlag{
|
||||||
|
Name: "guess-eventtypes",
|
||||||
|
Usage: "guess event types for smart-contract bindings configuration from the code usages",
|
||||||
|
},
|
||||||
cli.StringFlag{
|
cli.StringFlag{
|
||||||
Name: "bindings",
|
Name: "bindings",
|
||||||
Usage: "output file for smart-contract bindings configuration",
|
Usage: "output file for smart-contract bindings configuration",
|
||||||
|
@ -173,10 +190,12 @@ func NewCommands() []cli.Command {
|
||||||
{
|
{
|
||||||
Name: "deploy",
|
Name: "deploy",
|
||||||
Usage: "deploy a smart contract (.nef with description)",
|
Usage: "deploy a smart contract (.nef with description)",
|
||||||
UsageText: "neo-go contract deploy -r endpoint -w wallet [-a address] [-g gas] [-e sysgas] --in contract.nef --manifest contract.manifest.json [--out file] [--force] [data]",
|
UsageText: "neo-go contract deploy -r endpoint -w wallet [-a address] [-g gas] [-e sysgas] --in contract.nef --manifest contract.manifest.json [--out file] [--force] [--await] [data]",
|
||||||
Description: `Deploys given contract into the chain. The gas parameter is for additional
|
Description: `Deploys given contract into the chain. The gas parameter is for additional
|
||||||
gas to be added as a network fee to prioritize the transaction. The data
|
gas to be added as a network fee to prioritize the transaction. The data
|
||||||
parameter is an optional parameter to be passed to '_deploy' method.
|
parameter is an optional parameter to be passed to '_deploy' method. When
|
||||||
|
--await flag is specified, it waits for the transaction to be included
|
||||||
|
in a block.
|
||||||
`,
|
`,
|
||||||
Action: contractDeploy,
|
Action: contractDeploy,
|
||||||
Flags: deployFlags,
|
Flags: deployFlags,
|
||||||
|
@ -186,13 +205,14 @@ func NewCommands() []cli.Command {
|
||||||
{
|
{
|
||||||
Name: "invokefunction",
|
Name: "invokefunction",
|
||||||
Usage: "invoke deployed contract on the blockchain",
|
Usage: "invoke deployed contract on the blockchain",
|
||||||
UsageText: "neo-go contract invokefunction -r endpoint -w wallet [-a address] [-g gas] [-e sysgas] [--out file] [--force] scripthash [method] [arguments...] [--] [signers...]",
|
UsageText: "neo-go contract invokefunction -r endpoint -w wallet [-a address] [-g gas] [-e sysgas] [--out file] [--force] [--await] scripthash [method] [arguments...] [--] [signers...]",
|
||||||
Description: `Executes given (as a script hash) deployed script with the given method,
|
Description: `Executes given (as a script hash) deployed script with the given method,
|
||||||
arguments and signers. Sender is included in the list of signers by default
|
arguments and signers. Sender is included in the list of signers by default
|
||||||
with None witness scope. If you'd like to change default sender's scope,
|
with None witness scope. If you'd like to change default sender's scope,
|
||||||
specify it via signers parameter. See testinvokefunction documentation for
|
specify it via signers parameter. See testinvokefunction documentation for
|
||||||
the details about parameters. It differs from testinvokefunction in that this
|
the details about parameters. It differs from testinvokefunction in that this
|
||||||
command sends an invocation transaction to the network.
|
command sends an invocation transaction to the network. When --await flag is
|
||||||
|
specified, it waits for the transaction to be included in a block.
|
||||||
`,
|
`,
|
||||||
Action: invokeFunction,
|
Action: invokeFunction,
|
||||||
Flags: invokeFunctionFlags,
|
Flags: invokeFunctionFlags,
|
||||||
|
@ -289,26 +309,7 @@ func NewCommands() []cli.Command {
|
||||||
Usage: "adds group to the manifest",
|
Usage: "adds group to the manifest",
|
||||||
UsageText: "neo-go contract manifest add-group -w wallet [--wallet-config path] -n nef -m manifest -a address -s address",
|
UsageText: "neo-go contract manifest add-group -w wallet [--wallet-config path] -n nef -m manifest -a address -s address",
|
||||||
Action: manifestAddGroup,
|
Action: manifestAddGroup,
|
||||||
Flags: []cli.Flag{
|
Flags: manifestAddGroupFlags,
|
||||||
walletFlag,
|
|
||||||
walletConfigFlag,
|
|
||||||
cli.StringFlag{
|
|
||||||
Name: "sender, s",
|
|
||||||
Usage: "deploy transaction sender",
|
|
||||||
},
|
|
||||||
flags.AddressFlag{
|
|
||||||
Name: addressFlagName, // use the same name for handler code unification.
|
|
||||||
Usage: "account to sign group with",
|
|
||||||
},
|
|
||||||
cli.StringFlag{
|
|
||||||
Name: "nef, n",
|
|
||||||
Usage: "path to the NEF file",
|
|
||||||
},
|
|
||||||
cli.StringFlag{
|
|
||||||
Name: "manifest, m",
|
|
||||||
Usage: "path to the manifest",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -345,17 +346,19 @@ func initSmartContract(ctx *cli.Context) error {
|
||||||
SourceURL: "http://example.com/",
|
SourceURL: "http://example.com/",
|
||||||
SupportedStandards: []string{},
|
SupportedStandards: []string{},
|
||||||
SafeMethods: []string{},
|
SafeMethods: []string{},
|
||||||
Events: []manifest.Event{
|
Events: []compiler.HybridEvent{
|
||||||
{
|
{
|
||||||
Name: "Hello world!",
|
Name: "Hello world!",
|
||||||
Parameters: []manifest.Parameter{
|
Parameters: []compiler.HybridParameter{
|
||||||
{
|
{
|
||||||
|
Parameter: manifest.Parameter{
|
||||||
Name: "args",
|
Name: "args",
|
||||||
Type: smartcontract.ArrayType,
|
Type: smartcontract.ArrayType,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
},
|
||||||
Permissions: []permission{permission(*manifest.NewPermission(manifest.PermissionWildcard))},
|
Permissions: []permission{permission(*manifest.NewPermission(manifest.PermissionWildcard))},
|
||||||
}
|
}
|
||||||
b, err := yaml.Marshal(m)
|
b, err := yaml.Marshal(m)
|
||||||
|
@ -372,6 +375,9 @@ func initSmartContract(ctx *cli.Context) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
gm := []byte("module " + contractName + `
|
gm := []byte("module " + contractName + `
|
||||||
|
|
||||||
|
go 1.20
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/nspcc-dev/neo-go/pkg/interop ` + ver + `
|
github.com/nspcc-dev/neo-go/pkg/interop ` + ver + `
|
||||||
)`)
|
)`)
|
||||||
|
@ -400,20 +406,48 @@ func contractCompile(ctx *cli.Context) error {
|
||||||
manifestFile := ctx.String("manifest")
|
manifestFile := ctx.String("manifest")
|
||||||
confFile := ctx.String("config")
|
confFile := ctx.String("config")
|
||||||
debugFile := ctx.String("debug")
|
debugFile := ctx.String("debug")
|
||||||
if len(confFile) == 0 && (len(manifestFile) != 0 || len(debugFile) != 0) {
|
out := ctx.String("out")
|
||||||
|
bindings := ctx.String("bindings")
|
||||||
|
if len(confFile) == 0 && (len(manifestFile) != 0 || len(debugFile) != 0 || len(bindings) != 0) {
|
||||||
return cli.NewExitError(errNoConfFile, 1)
|
return cli.NewExitError(errNoConfFile, 1)
|
||||||
}
|
}
|
||||||
|
autocomplete := len(manifestFile) == 0 &&
|
||||||
|
len(confFile) == 0 &&
|
||||||
|
len(out) == 0 &&
|
||||||
|
len(bindings) == 0
|
||||||
|
if autocomplete {
|
||||||
|
var root string
|
||||||
|
fileInfo, err := os.Stat(src)
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(fmt.Errorf("failed to stat source file or directory: %w", err), 1)
|
||||||
|
}
|
||||||
|
if fileInfo.IsDir() {
|
||||||
|
base := filepath.Base(fileInfo.Name())
|
||||||
|
if base == string(filepath.Separator) {
|
||||||
|
base = "contract"
|
||||||
|
}
|
||||||
|
root = filepath.Join(src, base)
|
||||||
|
} else {
|
||||||
|
root = strings.TrimSuffix(src, ".go")
|
||||||
|
}
|
||||||
|
manifestFile = root + ".manifest.json"
|
||||||
|
confFile = root + ".yml"
|
||||||
|
out = root + ".nef"
|
||||||
|
bindings = root + ".bindings.yml"
|
||||||
|
}
|
||||||
|
|
||||||
o := &compiler.Options{
|
o := &compiler.Options{
|
||||||
Outfile: ctx.String("out"),
|
Outfile: out,
|
||||||
|
|
||||||
DebugInfo: debugFile,
|
DebugInfo: debugFile,
|
||||||
ManifestFile: manifestFile,
|
ManifestFile: manifestFile,
|
||||||
BindingsFile: ctx.String("bindings"),
|
BindingsFile: bindings,
|
||||||
|
|
||||||
NoStandardCheck: ctx.Bool("no-standards"),
|
NoStandardCheck: ctx.Bool("no-standards"),
|
||||||
NoEventsCheck: ctx.Bool("no-events"),
|
NoEventsCheck: ctx.Bool("no-events"),
|
||||||
NoPermissionsCheck: ctx.Bool("no-permissions"),
|
NoPermissionsCheck: ctx.Bool("no-permissions"),
|
||||||
|
|
||||||
|
GuessEventTypes: ctx.Bool("guess-eventtypes"),
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(confFile) != 0 {
|
if len(confFile) != 0 {
|
||||||
|
@ -424,6 +458,7 @@ func contractCompile(ctx *cli.Context) error {
|
||||||
o.Name = conf.Name
|
o.Name = conf.Name
|
||||||
o.SourceURL = conf.SourceURL
|
o.SourceURL = conf.SourceURL
|
||||||
o.ContractEvents = conf.Events
|
o.ContractEvents = conf.Events
|
||||||
|
o.DeclaredNamedTypes = conf.NamedTypes
|
||||||
o.ContractSupportedStandards = conf.SupportedStandards
|
o.ContractSupportedStandards = conf.SupportedStandards
|
||||||
o.Permissions = make([]manifest.Permission, len(conf.Permissions))
|
o.Permissions = make([]manifest.Permission, len(conf.Permissions))
|
||||||
for i := range conf.Permissions {
|
for i := range conf.Permissions {
|
||||||
|
@ -495,7 +530,7 @@ func invokeInternal(ctx *cli.Context, signAndPush bool) error {
|
||||||
err error
|
err error
|
||||||
exitErr *cli.ExitError
|
exitErr *cli.ExitError
|
||||||
operation string
|
operation string
|
||||||
params []interface{}
|
params []any
|
||||||
paramsStart = 1
|
paramsStart = 1
|
||||||
scParams []smartcontract.Parameter
|
scParams []smartcontract.Parameter
|
||||||
cosigners []transaction.Signer
|
cosigners []transaction.Signer
|
||||||
|
@ -521,7 +556,7 @@ func invokeInternal(ctx *cli.Context, signAndPush bool) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(err, 1)
|
return cli.NewExitError(err, 1)
|
||||||
}
|
}
|
||||||
params = make([]interface{}, len(scParams))
|
params = make([]any, len(scParams))
|
||||||
for i := range scParams {
|
for i := range scParams {
|
||||||
params[i] = scParams[i]
|
params[i] = scParams[i]
|
||||||
}
|
}
|
||||||
|
@ -538,7 +573,7 @@ func invokeInternal(ctx *cli.Context, signAndPush bool) error {
|
||||||
w *wallet.Wallet
|
w *wallet.Wallet
|
||||||
)
|
)
|
||||||
if signAndPush {
|
if signAndPush {
|
||||||
acc, w, err = getAccFromContext(ctx)
|
acc, w, err = options.GetAccFromContext(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(err, 1)
|
return cli.NewExitError(err, 1)
|
||||||
}
|
}
|
||||||
|
@ -548,7 +583,7 @@ func invokeInternal(ctx *cli.Context, signAndPush bool) error {
|
||||||
return invokeWithArgs(ctx, acc, w, script, operation, params, cosigners)
|
return invokeWithArgs(ctx, acc, w, script, operation, params, cosigners)
|
||||||
}
|
}
|
||||||
|
|
||||||
func invokeWithArgs(ctx *cli.Context, acc *wallet.Account, wall *wallet.Wallet, script util.Uint160, operation string, params []interface{}, cosigners []transaction.Signer) error {
|
func invokeWithArgs(ctx *cli.Context, acc *wallet.Account, wall *wallet.Wallet, script util.Uint160, operation string, params []any, cosigners []transaction.Signer) error {
|
||||||
var (
|
var (
|
||||||
err error
|
err error
|
||||||
signersAccounts []actor.SignerAccount
|
signersAccounts []actor.SignerAccount
|
||||||
|
@ -565,19 +600,14 @@ func invokeWithArgs(ctx *cli.Context, acc *wallet.Account, wall *wallet.Wallet,
|
||||||
}
|
}
|
||||||
gctx, cancel := options.GetTimeoutContext(ctx)
|
gctx, cancel := options.GetTimeoutContext(ctx)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
if signAndPush {
|
||||||
c, err := options.GetRPCClient(gctx, ctx)
|
_, act, err = options.GetRPCWithActor(gctx, ctx, signersAccounts)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if signAndPush {
|
|
||||||
act, err = actor.New(c, signersAccounts)
|
|
||||||
if err != nil {
|
|
||||||
return cli.NewExitError(fmt.Errorf("failed to create RPC actor: %w", err), 1)
|
|
||||||
}
|
|
||||||
inv = &act.Invoker
|
inv = &act.Invoker
|
||||||
} else {
|
} else {
|
||||||
inv, err = options.GetInvoker(c, ctx, cosigners)
|
_, inv, err = options.GetRPCWithInvoker(gctx, ctx, cosigners)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -672,9 +702,10 @@ type ProjectConfig struct {
|
||||||
SourceURL string
|
SourceURL string
|
||||||
SafeMethods []string
|
SafeMethods []string
|
||||||
SupportedStandards []string
|
SupportedStandards []string
|
||||||
Events []manifest.Event
|
Events []compiler.HybridEvent
|
||||||
Permissions []permission
|
Permissions []permission
|
||||||
Overloads map[string]string `yaml:"overloads,omitempty"`
|
Overloads map[string]string `yaml:"overloads,omitempty"`
|
||||||
|
NamedTypes map[string]binding.ExtendedType `yaml:"namedtypes,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func inspect(ctx *cli.Context) error {
|
func inspect(ctx *cli.Context) error {
|
||||||
|
@ -713,68 +744,6 @@ func inspect(ctx *cli.Context) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func getAccFromContext(ctx *cli.Context) (*wallet.Account, *wallet.Wallet, error) {
|
|
||||||
var addr util.Uint160
|
|
||||||
|
|
||||||
wPath := ctx.String("wallet")
|
|
||||||
walletConfigPath := ctx.String("wallet-config")
|
|
||||||
if len(wPath) != 0 && len(walletConfigPath) != 0 {
|
|
||||||
return nil, nil, errConflictingWalletFlags
|
|
||||||
}
|
|
||||||
if len(wPath) == 0 && len(walletConfigPath) == 0 {
|
|
||||||
return nil, nil, errNoWallet
|
|
||||||
}
|
|
||||||
var pass *string
|
|
||||||
if len(walletConfigPath) != 0 {
|
|
||||||
cfg, err := cliwallet.ReadWalletConfig(walletConfigPath)
|
|
||||||
if err != nil {
|
|
||||||
return nil, nil, err
|
|
||||||
}
|
|
||||||
wPath = cfg.Path
|
|
||||||
pass = &cfg.Password
|
|
||||||
}
|
|
||||||
|
|
||||||
wall, err := wallet.NewWalletFromFile(wPath)
|
|
||||||
if err != nil {
|
|
||||||
return nil, nil, err
|
|
||||||
}
|
|
||||||
addrFlag := ctx.Generic("address").(*flags.Address)
|
|
||||||
if addrFlag.IsSet {
|
|
||||||
addr = addrFlag.Uint160()
|
|
||||||
} else {
|
|
||||||
addr = wall.GetChangeAddress()
|
|
||||||
}
|
|
||||||
|
|
||||||
acc, err := getUnlockedAccount(wall, addr, pass)
|
|
||||||
return acc, wall, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func getUnlockedAccount(wall *wallet.Wallet, addr util.Uint160, pass *string) (*wallet.Account, error) {
|
|
||||||
acc := wall.GetAccount(addr)
|
|
||||||
if acc == nil {
|
|
||||||
return nil, fmt.Errorf("wallet contains no account for '%s'", address.Uint160ToString(addr))
|
|
||||||
}
|
|
||||||
|
|
||||||
if acc.CanSign() {
|
|
||||||
return acc, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if pass == nil {
|
|
||||||
rawPass, err := input.ReadPassword(
|
|
||||||
fmt.Sprintf("Enter account %s password > ", address.Uint160ToString(addr)))
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("Error reading password: %w", err)
|
|
||||||
}
|
|
||||||
trimmed := strings.TrimRight(string(rawPass), "\n")
|
|
||||||
pass = &trimmed
|
|
||||||
}
|
|
||||||
err := acc.Decrypt(*pass, wall.Scrypt)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return acc, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// contractDeploy deploys contract.
|
// contractDeploy deploys contract.
|
||||||
func contractDeploy(ctx *cli.Context) error {
|
func contractDeploy(ctx *cli.Context) error {
|
||||||
nefFile, f, err := readNEFFile(ctx.String("in"))
|
nefFile, f, err := readNEFFile(ctx.String("in"))
|
||||||
|
@ -787,7 +756,7 @@ func contractDeploy(ctx *cli.Context) error {
|
||||||
return cli.NewExitError(fmt.Errorf("failed to read manifest file: %w", err), 1)
|
return cli.NewExitError(fmt.Errorf("failed to read manifest file: %w", err), 1)
|
||||||
}
|
}
|
||||||
|
|
||||||
var appCallParams = []interface{}{f, manifestBytes}
|
var appCallParams = []any{f, manifestBytes}
|
||||||
|
|
||||||
signOffset, data, err := cmdargs.ParseParams(ctx.Args(), true)
|
signOffset, data, err := cmdargs.ParseParams(ctx.Args(), true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -800,7 +769,7 @@ func contractDeploy(ctx *cli.Context) error {
|
||||||
appCallParams = append(appCallParams, data[0])
|
appCallParams = append(appCallParams, data[0])
|
||||||
}
|
}
|
||||||
|
|
||||||
acc, w, err := getAccFromContext(ctx)
|
acc, w, err := options.GetAccFromContext(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(fmt.Errorf("can't get sender address: %w", err), 1)
|
return cli.NewExitError(fmt.Errorf("can't get sender address: %w", err), 1)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
package smartcontract
|
package smartcontract
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/hex"
|
|
||||||
"flag"
|
"flag"
|
||||||
"os"
|
"os"
|
||||||
"testing"
|
"testing"
|
||||||
|
@ -51,7 +50,7 @@ func init() {
|
||||||
}
|
}
|
||||||
|
|
||||||
// RuntimeNotify sends runtime notification with "Hello world!" name
|
// RuntimeNotify sends runtime notification with "Hello world!" name
|
||||||
func RuntimeNotify(args []interface{}) {
|
func RuntimeNotify(args []any) {
|
||||||
runtime.Notify(notificationName, args)
|
runtime.Notify(notificationName, args)
|
||||||
}`, string(main))
|
}`, string(main))
|
||||||
|
|
||||||
|
@ -109,7 +108,7 @@ func TestPermissionMarshal(t *testing.T) {
|
||||||
p.Methods.Add("abc")
|
p.Methods.Add("abc")
|
||||||
p.Methods.Add("lamao")
|
p.Methods.Add("lamao")
|
||||||
testPermissionMarshal(t, p,
|
testPermissionMarshal(t, p,
|
||||||
"group: "+hex.EncodeToString(priv.PublicKey().Bytes())+"\n"+
|
"group: "+priv.PublicKey().StringCompressed()+"\n"+
|
||||||
"methods:\n - abc\n - lamao\n")
|
"methods:\n - abc\n - lamao\n")
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -118,7 +117,7 @@ func TestPermissionUnmarshalInvalid(t *testing.T) {
|
||||||
priv, err := keys.NewPrivateKey()
|
priv, err := keys.NewPrivateKey()
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
pub := hex.EncodeToString(priv.PublicKey().Bytes())
|
pub := priv.PublicKey().StringCompressed()
|
||||||
u160 := random.Uint160().StringLE()
|
u160 := random.Uint160().StringLE()
|
||||||
testCases := []string{
|
testCases := []string{
|
||||||
"hash: []\nmethods: '*'\n", // invalid hash type
|
"hash: []\nmethods: '*'\n", // invalid hash type
|
||||||
|
|
8
cli/smartcontract/testdata/deploy/main.go
vendored
8
cli/smartcontract/testdata/deploy/main.go
vendored
|
@ -13,7 +13,7 @@ var key = "key"
|
||||||
|
|
||||||
const mgmtKey = "mgmt"
|
const mgmtKey = "mgmt"
|
||||||
|
|
||||||
func _deploy(data interface{}, isUpdate bool) {
|
func _deploy(data any, isUpdate bool) {
|
||||||
var value string
|
var value string
|
||||||
|
|
||||||
ctx := storage.GetContext()
|
ctx := storage.GetContext()
|
||||||
|
@ -25,7 +25,7 @@ func _deploy(data interface{}, isUpdate bool) {
|
||||||
storage.Put(ctx, mgmtKey, sh)
|
storage.Put(ctx, mgmtKey, sh)
|
||||||
|
|
||||||
if data != nil {
|
if data != nil {
|
||||||
arr := data.([]interface{})
|
arr := data.([]any)
|
||||||
for i := 0; i < len(arr)-1; i += 2 {
|
for i := 0; i < len(arr)-1; i += 2 {
|
||||||
storage.Put(ctx, arr[i], arr[i+1])
|
storage.Put(ctx, arr[i], arr[i+1])
|
||||||
}
|
}
|
||||||
|
@ -70,12 +70,12 @@ func GetValueWithKey(key string) string {
|
||||||
}
|
}
|
||||||
|
|
||||||
// TestFind finds items with the specified prefix.
|
// TestFind finds items with the specified prefix.
|
||||||
func TestFind(f storage.FindFlags) []interface{} {
|
func TestFind(f storage.FindFlags) []any {
|
||||||
ctx := storage.GetContext()
|
ctx := storage.GetContext()
|
||||||
storage.Put(ctx, "findkey1", "value1")
|
storage.Put(ctx, "findkey1", "value1")
|
||||||
storage.Put(ctx, "findkey2", "value2")
|
storage.Put(ctx, "findkey2", "value2")
|
||||||
|
|
||||||
var result []interface{}
|
var result []any
|
||||||
iter := storage.Find(ctx, "findkey", f)
|
iter := storage.Find(ctx, "findkey", f)
|
||||||
for iterator.Next(iter) {
|
for iterator.Next(iter) {
|
||||||
result = append(result, iterator.Value(iter))
|
result = append(result, iterator.Value(iter))
|
||||||
|
|
2
cli/smartcontract/testdata/deploy/sub/put.go
vendored
2
cli/smartcontract/testdata/deploy/sub/put.go
vendored
|
@ -4,7 +4,7 @@ import "github.com/nspcc-dev/neo-go/pkg/interop/storage"
|
||||||
|
|
||||||
var Key = "sub"
|
var Key = "sub"
|
||||||
|
|
||||||
func _deploy(data interface{}, isUpdate bool) {
|
func _deploy(data any, isUpdate bool) {
|
||||||
ctx := storage.GetContext()
|
ctx := storage.GetContext()
|
||||||
value := "sub create"
|
value := "sub create"
|
||||||
if isUpdate {
|
if isUpdate {
|
||||||
|
|
13
cli/smartcontract/testdata/gas/gas.go
vendored
13
cli/smartcontract/testdata/gas/gas.go
vendored
|
@ -1,3 +1,5 @@
|
||||||
|
// Code generated by neo-go contract generate-rpcwrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
// Package gastoken contains RPC wrappers for GasToken contract.
|
// Package gastoken contains RPC wrappers for GasToken contract.
|
||||||
package gastoken
|
package gastoken
|
||||||
|
|
||||||
|
@ -25,6 +27,7 @@ type Actor interface {
|
||||||
type ContractReader struct {
|
type ContractReader struct {
|
||||||
nep17.TokenReader
|
nep17.TokenReader
|
||||||
invoker Invoker
|
invoker Invoker
|
||||||
|
hash util.Uint160
|
||||||
}
|
}
|
||||||
|
|
||||||
// Contract implements all contract methods.
|
// Contract implements all contract methods.
|
||||||
|
@ -32,16 +35,18 @@ type Contract struct {
|
||||||
ContractReader
|
ContractReader
|
||||||
nep17.TokenWriter
|
nep17.TokenWriter
|
||||||
actor Actor
|
actor Actor
|
||||||
|
hash util.Uint160
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewReader creates an instance of ContractReader using Hash and the given Invoker.
|
// NewReader creates an instance of ContractReader using Hash and the given Invoker.
|
||||||
func NewReader(invoker Invoker) *ContractReader {
|
func NewReader(invoker Invoker) *ContractReader {
|
||||||
return &ContractReader{*nep17.NewReader(invoker, Hash), invoker}
|
var hash = Hash
|
||||||
|
return &ContractReader{*nep17.NewReader(invoker, hash), invoker, hash}
|
||||||
}
|
}
|
||||||
|
|
||||||
// New creates an instance of Contract using Hash and the given Actor.
|
// New creates an instance of Contract using Hash and the given Actor.
|
||||||
func New(actor Actor) *Contract {
|
func New(actor Actor) *Contract {
|
||||||
var nep17t = nep17.New(actor, Hash)
|
var hash = Hash
|
||||||
return &Contract{ContractReader{nep17t.TokenReader, actor}, nep17t.TokenWriter, actor}
|
var nep17t = nep17.New(actor, hash)
|
||||||
|
return &Contract{ContractReader{nep17t.TokenReader, actor, hash}, nep17t.TokenWriter, actor, hash}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
294
cli/smartcontract/testdata/nameservice/nns.go
vendored
294
cli/smartcontract/testdata/nameservice/nns.go
vendored
|
@ -1,7 +1,11 @@
|
||||||
|
// Code generated by neo-go contract generate-rpcwrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
// Package nameservice contains RPC wrappers for NameService contract.
|
// Package nameservice contains RPC wrappers for NameService contract.
|
||||||
package nameservice
|
package nameservice
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/neorpc/result"
|
"github.com/nspcc-dev/neo-go/pkg/neorpc/result"
|
||||||
|
@ -11,11 +15,26 @@ import (
|
||||||
"github.com/nspcc-dev/neo-go/pkg/util"
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
||||||
"math/big"
|
"math/big"
|
||||||
|
"unicode/utf8"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Hash contains contract hash.
|
// Hash contains contract hash.
|
||||||
var Hash = util.Uint160{0xde, 0x46, 0x5f, 0x5d, 0x50, 0x57, 0xcf, 0x33, 0x28, 0x47, 0x94, 0xc5, 0xcf, 0xc2, 0xc, 0x69, 0x37, 0x1c, 0xac, 0x50}
|
var Hash = util.Uint160{0xde, 0x46, 0x5f, 0x5d, 0x50, 0x57, 0xcf, 0x33, 0x28, 0x47, 0x94, 0xc5, 0xcf, 0xc2, 0xc, 0x69, 0x37, 0x1c, 0xac, 0x50}
|
||||||
|
|
||||||
|
// SetAdminEvent represents "SetAdmin" event emitted by the contract.
|
||||||
|
type SetAdminEvent struct {
|
||||||
|
Name string
|
||||||
|
OldAdmin util.Uint160
|
||||||
|
NewAdmin util.Uint160
|
||||||
|
}
|
||||||
|
|
||||||
|
// RenewEvent represents "Renew" event emitted by the contract.
|
||||||
|
type RenewEvent struct {
|
||||||
|
Name string
|
||||||
|
OldExpiration *big.Int
|
||||||
|
NewExpiration *big.Int
|
||||||
|
}
|
||||||
|
|
||||||
// Invoker is used by ContractReader to call various safe methods.
|
// Invoker is used by ContractReader to call various safe methods.
|
||||||
type Invoker interface {
|
type Invoker interface {
|
||||||
nep11.Invoker
|
nep11.Invoker
|
||||||
|
@ -27,11 +46,11 @@ type Actor interface {
|
||||||
|
|
||||||
nep11.Actor
|
nep11.Actor
|
||||||
|
|
||||||
MakeCall(contract util.Uint160, method string, params ...interface{}) (*transaction.Transaction, error)
|
MakeCall(contract util.Uint160, method string, params ...any) (*transaction.Transaction, error)
|
||||||
MakeRun(script []byte) (*transaction.Transaction, error)
|
MakeRun(script []byte) (*transaction.Transaction, error)
|
||||||
MakeUnsignedCall(contract util.Uint160, method string, attrs []transaction.Attribute, params ...interface{}) (*transaction.Transaction, error)
|
MakeUnsignedCall(contract util.Uint160, method string, attrs []transaction.Attribute, params ...any) (*transaction.Transaction, error)
|
||||||
MakeUnsignedRun(script []byte, attrs []transaction.Attribute) (*transaction.Transaction, error)
|
MakeUnsignedRun(script []byte, attrs []transaction.Attribute) (*transaction.Transaction, error)
|
||||||
SendCall(contract util.Uint160, method string, params ...interface{}) (util.Uint256, uint32, error)
|
SendCall(contract util.Uint160, method string, params ...any) (util.Uint256, uint32, error)
|
||||||
SendRun(script []byte) (util.Uint256, uint32, error)
|
SendRun(script []byte) (util.Uint256, uint32, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,6 +58,7 @@ type Actor interface {
|
||||||
type ContractReader struct {
|
type ContractReader struct {
|
||||||
nep11.NonDivisibleReader
|
nep11.NonDivisibleReader
|
||||||
invoker Invoker
|
invoker Invoker
|
||||||
|
hash util.Uint160
|
||||||
}
|
}
|
||||||
|
|
||||||
// Contract implements all contract methods.
|
// Contract implements all contract methods.
|
||||||
|
@ -46,23 +66,25 @@ type Contract struct {
|
||||||
ContractReader
|
ContractReader
|
||||||
nep11.BaseWriter
|
nep11.BaseWriter
|
||||||
actor Actor
|
actor Actor
|
||||||
|
hash util.Uint160
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewReader creates an instance of ContractReader using Hash and the given Invoker.
|
// NewReader creates an instance of ContractReader using Hash and the given Invoker.
|
||||||
func NewReader(invoker Invoker) *ContractReader {
|
func NewReader(invoker Invoker) *ContractReader {
|
||||||
return &ContractReader{*nep11.NewNonDivisibleReader(invoker, Hash), invoker}
|
var hash = Hash
|
||||||
|
return &ContractReader{*nep11.NewNonDivisibleReader(invoker, hash), invoker, hash}
|
||||||
}
|
}
|
||||||
|
|
||||||
// New creates an instance of Contract using Hash and the given Actor.
|
// New creates an instance of Contract using Hash and the given Actor.
|
||||||
func New(actor Actor) *Contract {
|
func New(actor Actor) *Contract {
|
||||||
var nep11ndt = nep11.NewNonDivisible(actor, Hash)
|
var hash = Hash
|
||||||
return &Contract{ContractReader{nep11ndt.NonDivisibleReader, actor}, nep11ndt.BaseWriter, actor}
|
var nep11ndt = nep11.NewNonDivisible(actor, hash)
|
||||||
|
return &Contract{ContractReader{nep11ndt.NonDivisibleReader, actor, hash}, nep11ndt.BaseWriter, actor, hash}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Roots invokes `roots` method of contract.
|
// Roots invokes `roots` method of contract.
|
||||||
func (c *ContractReader) Roots() (uuid.UUID, result.Iterator, error) {
|
func (c *ContractReader) Roots() (uuid.UUID, result.Iterator, error) {
|
||||||
return unwrap.SessionIterator(c.invoker.Call(Hash, "roots"))
|
return unwrap.SessionIterator(c.invoker.Call(c.hash, "roots"))
|
||||||
}
|
}
|
||||||
|
|
||||||
// RootsExpanded is similar to Roots (uses the same contract
|
// RootsExpanded is similar to Roots (uses the same contract
|
||||||
|
@ -71,27 +93,27 @@ func (c *ContractReader) Roots() (uuid.UUID, result.Iterator, error) {
|
||||||
// number of result items from the iterator right in the VM and return them to
|
// number of result items from the iterator right in the VM and return them to
|
||||||
// you. It's only limited by VM stack and GAS available for RPC invocations.
|
// you. It's only limited by VM stack and GAS available for RPC invocations.
|
||||||
func (c *ContractReader) RootsExpanded(_numOfIteratorItems int) ([]stackitem.Item, error) {
|
func (c *ContractReader) RootsExpanded(_numOfIteratorItems int) ([]stackitem.Item, error) {
|
||||||
return unwrap.Array(c.invoker.CallAndExpandIterator(Hash, "roots", _numOfIteratorItems))
|
return unwrap.Array(c.invoker.CallAndExpandIterator(c.hash, "roots", _numOfIteratorItems))
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetPrice invokes `getPrice` method of contract.
|
// GetPrice invokes `getPrice` method of contract.
|
||||||
func (c *ContractReader) GetPrice(length *big.Int) (*big.Int, error) {
|
func (c *ContractReader) GetPrice(length *big.Int) (*big.Int, error) {
|
||||||
return unwrap.BigInt(c.invoker.Call(Hash, "getPrice", length))
|
return unwrap.BigInt(c.invoker.Call(c.hash, "getPrice", length))
|
||||||
}
|
}
|
||||||
|
|
||||||
// IsAvailable invokes `isAvailable` method of contract.
|
// IsAvailable invokes `isAvailable` method of contract.
|
||||||
func (c *ContractReader) IsAvailable(name string) (bool, error) {
|
func (c *ContractReader) IsAvailable(name string) (bool, error) {
|
||||||
return unwrap.Bool(c.invoker.Call(Hash, "isAvailable", name))
|
return unwrap.Bool(c.invoker.Call(c.hash, "isAvailable", name))
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetRecord invokes `getRecord` method of contract.
|
// GetRecord invokes `getRecord` method of contract.
|
||||||
func (c *ContractReader) GetRecord(name string, typev *big.Int) (string, error) {
|
func (c *ContractReader) GetRecord(name string, typev *big.Int) (string, error) {
|
||||||
return unwrap.UTF8String(c.invoker.Call(Hash, "getRecord", name, typev))
|
return unwrap.UTF8String(c.invoker.Call(c.hash, "getRecord", name, typev))
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetAllRecords invokes `getAllRecords` method of contract.
|
// GetAllRecords invokes `getAllRecords` method of contract.
|
||||||
func (c *ContractReader) GetAllRecords(name string) (uuid.UUID, result.Iterator, error) {
|
func (c *ContractReader) GetAllRecords(name string) (uuid.UUID, result.Iterator, error) {
|
||||||
return unwrap.SessionIterator(c.invoker.Call(Hash, "getAllRecords", name))
|
return unwrap.SessionIterator(c.invoker.Call(c.hash, "getAllRecords", name))
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetAllRecordsExpanded is similar to GetAllRecords (uses the same contract
|
// GetAllRecordsExpanded is similar to GetAllRecords (uses the same contract
|
||||||
|
@ -100,26 +122,26 @@ func (c *ContractReader) GetAllRecords(name string) (uuid.UUID, result.Iterator,
|
||||||
// number of result items from the iterator right in the VM and return them to
|
// number of result items from the iterator right in the VM and return them to
|
||||||
// you. It's only limited by VM stack and GAS available for RPC invocations.
|
// you. It's only limited by VM stack and GAS available for RPC invocations.
|
||||||
func (c *ContractReader) GetAllRecordsExpanded(name string, _numOfIteratorItems int) ([]stackitem.Item, error) {
|
func (c *ContractReader) GetAllRecordsExpanded(name string, _numOfIteratorItems int) ([]stackitem.Item, error) {
|
||||||
return unwrap.Array(c.invoker.CallAndExpandIterator(Hash, "getAllRecords", _numOfIteratorItems, name))
|
return unwrap.Array(c.invoker.CallAndExpandIterator(c.hash, "getAllRecords", _numOfIteratorItems, name))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Resolve invokes `resolve` method of contract.
|
// Resolve invokes `resolve` method of contract.
|
||||||
func (c *ContractReader) Resolve(name string, typev *big.Int) (string, error) {
|
func (c *ContractReader) Resolve(name string, typev *big.Int) (string, error) {
|
||||||
return unwrap.UTF8String(c.invoker.Call(Hash, "resolve", name, typev))
|
return unwrap.UTF8String(c.invoker.Call(c.hash, "resolve", name, typev))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update creates a transaction invoking `update` method of the contract.
|
// Update creates a transaction invoking `update` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) Update(nef []byte, manifest string) (util.Uint256, uint32, error) {
|
func (c *Contract) Update(nef []byte, manifest string) (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "update", nef, manifest)
|
return c.actor.SendCall(c.hash, "update", nef, manifest)
|
||||||
}
|
}
|
||||||
|
|
||||||
// UpdateTransaction creates a transaction invoking `update` method of the contract.
|
// UpdateTransaction creates a transaction invoking `update` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) UpdateTransaction(nef []byte, manifest string) (*transaction.Transaction, error) {
|
func (c *Contract) UpdateTransaction(nef []byte, manifest string) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "update", nef, manifest)
|
return c.actor.MakeCall(c.hash, "update", nef, manifest)
|
||||||
}
|
}
|
||||||
|
|
||||||
// UpdateUnsigned creates a transaction invoking `update` method of the contract.
|
// UpdateUnsigned creates a transaction invoking `update` method of the contract.
|
||||||
|
@ -127,21 +149,21 @@ func (c *Contract) UpdateTransaction(nef []byte, manifest string) (*transaction.
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) UpdateUnsigned(nef []byte, manifest string) (*transaction.Transaction, error) {
|
func (c *Contract) UpdateUnsigned(nef []byte, manifest string) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "update", nil, nef, manifest)
|
return c.actor.MakeUnsignedCall(c.hash, "update", nil, nef, manifest)
|
||||||
}
|
}
|
||||||
|
|
||||||
// AddRoot creates a transaction invoking `addRoot` method of the contract.
|
// AddRoot creates a transaction invoking `addRoot` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) AddRoot(root string) (util.Uint256, uint32, error) {
|
func (c *Contract) AddRoot(root string) (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "addRoot", root)
|
return c.actor.SendCall(c.hash, "addRoot", root)
|
||||||
}
|
}
|
||||||
|
|
||||||
// AddRootTransaction creates a transaction invoking `addRoot` method of the contract.
|
// AddRootTransaction creates a transaction invoking `addRoot` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) AddRootTransaction(root string) (*transaction.Transaction, error) {
|
func (c *Contract) AddRootTransaction(root string) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "addRoot", root)
|
return c.actor.MakeCall(c.hash, "addRoot", root)
|
||||||
}
|
}
|
||||||
|
|
||||||
// AddRootUnsigned creates a transaction invoking `addRoot` method of the contract.
|
// AddRootUnsigned creates a transaction invoking `addRoot` method of the contract.
|
||||||
|
@ -149,40 +171,40 @@ func (c *Contract) AddRootTransaction(root string) (*transaction.Transaction, er
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) AddRootUnsigned(root string) (*transaction.Transaction, error) {
|
func (c *Contract) AddRootUnsigned(root string) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "addRoot", nil, root)
|
return c.actor.MakeUnsignedCall(c.hash, "addRoot", nil, root)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetPrice creates a transaction invoking `setPrice` method of the contract.
|
// SetPrice creates a transaction invoking `setPrice` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) SetPrice(priceList []interface{}) (util.Uint256, uint32, error) {
|
func (c *Contract) SetPrice(priceList []any) (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "setPrice", priceList)
|
return c.actor.SendCall(c.hash, "setPrice", priceList)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetPriceTransaction creates a transaction invoking `setPrice` method of the contract.
|
// SetPriceTransaction creates a transaction invoking `setPrice` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) SetPriceTransaction(priceList []interface{}) (*transaction.Transaction, error) {
|
func (c *Contract) SetPriceTransaction(priceList []any) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "setPrice", priceList)
|
return c.actor.MakeCall(c.hash, "setPrice", priceList)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetPriceUnsigned creates a transaction invoking `setPrice` method of the contract.
|
// SetPriceUnsigned creates a transaction invoking `setPrice` method of the contract.
|
||||||
// This transaction is not signed, it's simply returned to the caller.
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) SetPriceUnsigned(priceList []interface{}) (*transaction.Transaction, error) {
|
func (c *Contract) SetPriceUnsigned(priceList []any) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "setPrice", nil, priceList)
|
return c.actor.MakeUnsignedCall(c.hash, "setPrice", nil, priceList)
|
||||||
}
|
}
|
||||||
|
|
||||||
func scriptForRegister(name string, owner util.Uint160) ([]byte, error) {
|
func (c *Contract) scriptForRegister(name string, owner util.Uint160) ([]byte, error) {
|
||||||
return smartcontract.CreateCallWithAssertScript(Hash, "register", name, owner)
|
return smartcontract.CreateCallWithAssertScript(c.hash, "register", name, owner)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Register creates a transaction invoking `register` method of the contract.
|
// Register creates a transaction invoking `register` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) Register(name string, owner util.Uint160) (util.Uint256, uint32, error) {
|
func (c *Contract) Register(name string, owner util.Uint160) (util.Uint256, uint32, error) {
|
||||||
script, err := scriptForRegister(name, owner)
|
script, err := c.scriptForRegister(name, owner)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return util.Uint256{}, 0, err
|
return util.Uint256{}, 0, err
|
||||||
}
|
}
|
||||||
|
@ -193,7 +215,7 @@ func (c *Contract) Register(name string, owner util.Uint160) (util.Uint256, uint
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) RegisterTransaction(name string, owner util.Uint160) (*transaction.Transaction, error) {
|
func (c *Contract) RegisterTransaction(name string, owner util.Uint160) (*transaction.Transaction, error) {
|
||||||
script, err := scriptForRegister(name, owner)
|
script, err := c.scriptForRegister(name, owner)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -205,7 +227,7 @@ func (c *Contract) RegisterTransaction(name string, owner util.Uint160) (*transa
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) RegisterUnsigned(name string, owner util.Uint160) (*transaction.Transaction, error) {
|
func (c *Contract) RegisterUnsigned(name string, owner util.Uint160) (*transaction.Transaction, error) {
|
||||||
script, err := scriptForRegister(name, owner)
|
script, err := c.scriptForRegister(name, owner)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -216,14 +238,14 @@ func (c *Contract) RegisterUnsigned(name string, owner util.Uint160) (*transacti
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) Renew(name string) (util.Uint256, uint32, error) {
|
func (c *Contract) Renew(name string) (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "renew", name)
|
return c.actor.SendCall(c.hash, "renew", name)
|
||||||
}
|
}
|
||||||
|
|
||||||
// RenewTransaction creates a transaction invoking `renew` method of the contract.
|
// RenewTransaction creates a transaction invoking `renew` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) RenewTransaction(name string) (*transaction.Transaction, error) {
|
func (c *Contract) RenewTransaction(name string) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "renew", name)
|
return c.actor.MakeCall(c.hash, "renew", name)
|
||||||
}
|
}
|
||||||
|
|
||||||
// RenewUnsigned creates a transaction invoking `renew` method of the contract.
|
// RenewUnsigned creates a transaction invoking `renew` method of the contract.
|
||||||
|
@ -231,43 +253,43 @@ func (c *Contract) RenewTransaction(name string) (*transaction.Transaction, erro
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) RenewUnsigned(name string) (*transaction.Transaction, error) {
|
func (c *Contract) RenewUnsigned(name string) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "renew", nil, name)
|
return c.actor.MakeUnsignedCall(c.hash, "renew", nil, name)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Renew_2 creates a transaction invoking `renew` method of the contract.
|
// Renew2 creates a transaction invoking `renew` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) Renew_2(name string, years *big.Int) (util.Uint256, uint32, error) {
|
func (c *Contract) Renew2(name string, years *big.Int) (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "renew", name, years)
|
return c.actor.SendCall(c.hash, "renew", name, years)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Renew_2Transaction creates a transaction invoking `renew` method of the contract.
|
// Renew2Transaction creates a transaction invoking `renew` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) Renew_2Transaction(name string, years *big.Int) (*transaction.Transaction, error) {
|
func (c *Contract) Renew2Transaction(name string, years *big.Int) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "renew", name, years)
|
return c.actor.MakeCall(c.hash, "renew", name, years)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Renew_2Unsigned creates a transaction invoking `renew` method of the contract.
|
// Renew2Unsigned creates a transaction invoking `renew` method of the contract.
|
||||||
// This transaction is not signed, it's simply returned to the caller.
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) Renew_2Unsigned(name string, years *big.Int) (*transaction.Transaction, error) {
|
func (c *Contract) Renew2Unsigned(name string, years *big.Int) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "renew", nil, name, years)
|
return c.actor.MakeUnsignedCall(c.hash, "renew", nil, name, years)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetAdmin creates a transaction invoking `setAdmin` method of the contract.
|
// SetAdmin creates a transaction invoking `setAdmin` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) SetAdmin(name string, admin util.Uint160) (util.Uint256, uint32, error) {
|
func (c *Contract) SetAdmin(name string, admin util.Uint160) (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "setAdmin", name, admin)
|
return c.actor.SendCall(c.hash, "setAdmin", name, admin)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetAdminTransaction creates a transaction invoking `setAdmin` method of the contract.
|
// SetAdminTransaction creates a transaction invoking `setAdmin` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) SetAdminTransaction(name string, admin util.Uint160) (*transaction.Transaction, error) {
|
func (c *Contract) SetAdminTransaction(name string, admin util.Uint160) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "setAdmin", name, admin)
|
return c.actor.MakeCall(c.hash, "setAdmin", name, admin)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetAdminUnsigned creates a transaction invoking `setAdmin` method of the contract.
|
// SetAdminUnsigned creates a transaction invoking `setAdmin` method of the contract.
|
||||||
|
@ -275,21 +297,21 @@ func (c *Contract) SetAdminTransaction(name string, admin util.Uint160) (*transa
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) SetAdminUnsigned(name string, admin util.Uint160) (*transaction.Transaction, error) {
|
func (c *Contract) SetAdminUnsigned(name string, admin util.Uint160) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "setAdmin", nil, name, admin)
|
return c.actor.MakeUnsignedCall(c.hash, "setAdmin", nil, name, admin)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetRecord creates a transaction invoking `setRecord` method of the contract.
|
// SetRecord creates a transaction invoking `setRecord` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) SetRecord(name string, typev *big.Int, data string) (util.Uint256, uint32, error) {
|
func (c *Contract) SetRecord(name string, typev *big.Int, data string) (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "setRecord", name, typev, data)
|
return c.actor.SendCall(c.hash, "setRecord", name, typev, data)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetRecordTransaction creates a transaction invoking `setRecord` method of the contract.
|
// SetRecordTransaction creates a transaction invoking `setRecord` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) SetRecordTransaction(name string, typev *big.Int, data string) (*transaction.Transaction, error) {
|
func (c *Contract) SetRecordTransaction(name string, typev *big.Int, data string) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "setRecord", name, typev, data)
|
return c.actor.MakeCall(c.hash, "setRecord", name, typev, data)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetRecordUnsigned creates a transaction invoking `setRecord` method of the contract.
|
// SetRecordUnsigned creates a transaction invoking `setRecord` method of the contract.
|
||||||
|
@ -297,21 +319,21 @@ func (c *Contract) SetRecordTransaction(name string, typev *big.Int, data string
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) SetRecordUnsigned(name string, typev *big.Int, data string) (*transaction.Transaction, error) {
|
func (c *Contract) SetRecordUnsigned(name string, typev *big.Int, data string) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "setRecord", nil, name, typev, data)
|
return c.actor.MakeUnsignedCall(c.hash, "setRecord", nil, name, typev, data)
|
||||||
}
|
}
|
||||||
|
|
||||||
// DeleteRecord creates a transaction invoking `deleteRecord` method of the contract.
|
// DeleteRecord creates a transaction invoking `deleteRecord` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) DeleteRecord(name string, typev *big.Int) (util.Uint256, uint32, error) {
|
func (c *Contract) DeleteRecord(name string, typev *big.Int) (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "deleteRecord", name, typev)
|
return c.actor.SendCall(c.hash, "deleteRecord", name, typev)
|
||||||
}
|
}
|
||||||
|
|
||||||
// DeleteRecordTransaction creates a transaction invoking `deleteRecord` method of the contract.
|
// DeleteRecordTransaction creates a transaction invoking `deleteRecord` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) DeleteRecordTransaction(name string, typev *big.Int) (*transaction.Transaction, error) {
|
func (c *Contract) DeleteRecordTransaction(name string, typev *big.Int) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "deleteRecord", name, typev)
|
return c.actor.MakeCall(c.hash, "deleteRecord", name, typev)
|
||||||
}
|
}
|
||||||
|
|
||||||
// DeleteRecordUnsigned creates a transaction invoking `deleteRecord` method of the contract.
|
// DeleteRecordUnsigned creates a transaction invoking `deleteRecord` method of the contract.
|
||||||
|
@ -319,5 +341,171 @@ func (c *Contract) DeleteRecordTransaction(name string, typev *big.Int) (*transa
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) DeleteRecordUnsigned(name string, typev *big.Int) (*transaction.Transaction, error) {
|
func (c *Contract) DeleteRecordUnsigned(name string, typev *big.Int) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "deleteRecord", nil, name, typev)
|
return c.actor.MakeUnsignedCall(c.hash, "deleteRecord", nil, name, typev)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetAdminEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "SetAdmin" name from the provided [result.ApplicationLog].
|
||||||
|
func SetAdminEventsFromApplicationLog(log *result.ApplicationLog) ([]*SetAdminEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*SetAdminEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "SetAdmin" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(SetAdminEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize SetAdminEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to SetAdminEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *SetAdminEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 3 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.Name, err = func(item stackitem.Item) (string, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if !utf8.Valid(b) {
|
||||||
|
return "", errors.New("not a UTF-8 string")
|
||||||
|
}
|
||||||
|
return string(b), nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field Name: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
index++
|
||||||
|
e.OldAdmin, err = func(item stackitem.Item) (util.Uint160, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, err
|
||||||
|
}
|
||||||
|
u, err := util.Uint160DecodeBytesBE(b)
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, err
|
||||||
|
}
|
||||||
|
return u, nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field OldAdmin: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
index++
|
||||||
|
e.NewAdmin, err = func(item stackitem.Item) (util.Uint160, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, err
|
||||||
|
}
|
||||||
|
u, err := util.Uint160DecodeBytesBE(b)
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, err
|
||||||
|
}
|
||||||
|
return u, nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field NewAdmin: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// RenewEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "Renew" name from the provided [result.ApplicationLog].
|
||||||
|
func RenewEventsFromApplicationLog(log *result.ApplicationLog) ([]*RenewEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*RenewEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "Renew" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(RenewEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize RenewEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to RenewEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *RenewEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 3 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.Name, err = func(item stackitem.Item) (string, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if !utf8.Valid(b) {
|
||||||
|
return "", errors.New("not a UTF-8 string")
|
||||||
|
}
|
||||||
|
return string(b), nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field Name: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
index++
|
||||||
|
e.OldExpiration, err = arr[index].TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field OldExpiration: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
index++
|
||||||
|
e.NewExpiration, err = arr[index].TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field NewExpiration: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
177
cli/smartcontract/testdata/nex/nex.go
vendored
177
cli/smartcontract/testdata/nex/nex.go
vendored
|
@ -1,18 +1,32 @@
|
||||||
|
// Code generated by neo-go contract generate-rpcwrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
// Package nextoken contains RPC wrappers for NEX Token contract.
|
// Package nextoken contains RPC wrappers for NEX Token contract.
|
||||||
package nextoken
|
package nextoken
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/neorpc/result"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/rpcclient/nep17"
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/nep17"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/rpcclient/unwrap"
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/unwrap"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/util"
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
||||||
"math/big"
|
"math/big"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Hash contains contract hash.
|
// Hash contains contract hash.
|
||||||
var Hash = util.Uint160{0xa8, 0x1a, 0xa1, 0xf0, 0x4b, 0xf, 0xdc, 0x4a, 0xa2, 0xce, 0xd5, 0xbf, 0xc6, 0x22, 0xcf, 0xe8, 0x9, 0x7f, 0xa6, 0xa2}
|
var Hash = util.Uint160{0xa8, 0x1a, 0xa1, 0xf0, 0x4b, 0xf, 0xdc, 0x4a, 0xa2, 0xce, 0xd5, 0xbf, 0xc6, 0x22, 0xcf, 0xe8, 0x9, 0x7f, 0xa6, 0xa2}
|
||||||
|
|
||||||
|
// OnMintEvent represents "OnMint" event emitted by the contract.
|
||||||
|
type OnMintEvent struct {
|
||||||
|
From util.Uint160
|
||||||
|
To util.Uint160
|
||||||
|
Amount *big.Int
|
||||||
|
SwapId *big.Int
|
||||||
|
}
|
||||||
|
|
||||||
// Invoker is used by ContractReader to call various safe methods.
|
// Invoker is used by ContractReader to call various safe methods.
|
||||||
type Invoker interface {
|
type Invoker interface {
|
||||||
nep17.Invoker
|
nep17.Invoker
|
||||||
|
@ -24,11 +38,11 @@ type Actor interface {
|
||||||
|
|
||||||
nep17.Actor
|
nep17.Actor
|
||||||
|
|
||||||
MakeCall(contract util.Uint160, method string, params ...interface{}) (*transaction.Transaction, error)
|
MakeCall(contract util.Uint160, method string, params ...any) (*transaction.Transaction, error)
|
||||||
MakeRun(script []byte) (*transaction.Transaction, error)
|
MakeRun(script []byte) (*transaction.Transaction, error)
|
||||||
MakeUnsignedCall(contract util.Uint160, method string, attrs []transaction.Attribute, params ...interface{}) (*transaction.Transaction, error)
|
MakeUnsignedCall(contract util.Uint160, method string, attrs []transaction.Attribute, params ...any) (*transaction.Transaction, error)
|
||||||
MakeUnsignedRun(script []byte, attrs []transaction.Attribute) (*transaction.Transaction, error)
|
MakeUnsignedRun(script []byte, attrs []transaction.Attribute) (*transaction.Transaction, error)
|
||||||
SendCall(contract util.Uint160, method string, params ...interface{}) (util.Uint256, uint32, error)
|
SendCall(contract util.Uint160, method string, params ...any) (util.Uint256, uint32, error)
|
||||||
SendRun(script []byte) (util.Uint256, uint32, error)
|
SendRun(script []byte) (util.Uint256, uint32, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -36,6 +50,7 @@ type Actor interface {
|
||||||
type ContractReader struct {
|
type ContractReader struct {
|
||||||
nep17.TokenReader
|
nep17.TokenReader
|
||||||
invoker Invoker
|
invoker Invoker
|
||||||
|
hash util.Uint160
|
||||||
}
|
}
|
||||||
|
|
||||||
// Contract implements all contract methods.
|
// Contract implements all contract methods.
|
||||||
|
@ -43,52 +58,54 @@ type Contract struct {
|
||||||
ContractReader
|
ContractReader
|
||||||
nep17.TokenWriter
|
nep17.TokenWriter
|
||||||
actor Actor
|
actor Actor
|
||||||
|
hash util.Uint160
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewReader creates an instance of ContractReader using Hash and the given Invoker.
|
// NewReader creates an instance of ContractReader using Hash and the given Invoker.
|
||||||
func NewReader(invoker Invoker) *ContractReader {
|
func NewReader(invoker Invoker) *ContractReader {
|
||||||
return &ContractReader{*nep17.NewReader(invoker, Hash), invoker}
|
var hash = Hash
|
||||||
|
return &ContractReader{*nep17.NewReader(invoker, hash), invoker, hash}
|
||||||
}
|
}
|
||||||
|
|
||||||
// New creates an instance of Contract using Hash and the given Actor.
|
// New creates an instance of Contract using Hash and the given Actor.
|
||||||
func New(actor Actor) *Contract {
|
func New(actor Actor) *Contract {
|
||||||
var nep17t = nep17.New(actor, Hash)
|
var hash = Hash
|
||||||
return &Contract{ContractReader{nep17t.TokenReader, actor}, nep17t.TokenWriter, actor}
|
var nep17t = nep17.New(actor, hash)
|
||||||
|
return &Contract{ContractReader{nep17t.TokenReader, actor, hash}, nep17t.TokenWriter, actor, hash}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Cap invokes `cap` method of contract.
|
// Cap invokes `cap` method of contract.
|
||||||
func (c *ContractReader) Cap() (*big.Int, error) {
|
func (c *ContractReader) Cap() (*big.Int, error) {
|
||||||
return unwrap.BigInt(c.invoker.Call(Hash, "cap"))
|
return unwrap.BigInt(c.invoker.Call(c.hash, "cap"))
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetMinter invokes `getMinter` method of contract.
|
// GetMinter invokes `getMinter` method of contract.
|
||||||
func (c *ContractReader) GetMinter() (*keys.PublicKey, error) {
|
func (c *ContractReader) GetMinter() (*keys.PublicKey, error) {
|
||||||
return unwrap.PublicKey(c.invoker.Call(Hash, "getMinter"))
|
return unwrap.PublicKey(c.invoker.Call(c.hash, "getMinter"))
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetOwner invokes `getOwner` method of contract.
|
// GetOwner invokes `getOwner` method of contract.
|
||||||
func (c *ContractReader) GetOwner() (util.Uint160, error) {
|
func (c *ContractReader) GetOwner() (util.Uint160, error) {
|
||||||
return unwrap.Uint160(c.invoker.Call(Hash, "getOwner"))
|
return unwrap.Uint160(c.invoker.Call(c.hash, "getOwner"))
|
||||||
}
|
}
|
||||||
|
|
||||||
// TotalMinted invokes `totalMinted` method of contract.
|
// TotalMinted invokes `totalMinted` method of contract.
|
||||||
func (c *ContractReader) TotalMinted() (*big.Int, error) {
|
func (c *ContractReader) TotalMinted() (*big.Int, error) {
|
||||||
return unwrap.BigInt(c.invoker.Call(Hash, "totalMinted"))
|
return unwrap.BigInt(c.invoker.Call(c.hash, "totalMinted"))
|
||||||
}
|
}
|
||||||
|
|
||||||
// ChangeMinter creates a transaction invoking `changeMinter` method of the contract.
|
// ChangeMinter creates a transaction invoking `changeMinter` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) ChangeMinter(newMinter *keys.PublicKey) (util.Uint256, uint32, error) {
|
func (c *Contract) ChangeMinter(newMinter *keys.PublicKey) (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "changeMinter", newMinter)
|
return c.actor.SendCall(c.hash, "changeMinter", newMinter)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ChangeMinterTransaction creates a transaction invoking `changeMinter` method of the contract.
|
// ChangeMinterTransaction creates a transaction invoking `changeMinter` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) ChangeMinterTransaction(newMinter *keys.PublicKey) (*transaction.Transaction, error) {
|
func (c *Contract) ChangeMinterTransaction(newMinter *keys.PublicKey) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "changeMinter", newMinter)
|
return c.actor.MakeCall(c.hash, "changeMinter", newMinter)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ChangeMinterUnsigned creates a transaction invoking `changeMinter` method of the contract.
|
// ChangeMinterUnsigned creates a transaction invoking `changeMinter` method of the contract.
|
||||||
|
@ -96,21 +113,21 @@ func (c *Contract) ChangeMinterTransaction(newMinter *keys.PublicKey) (*transact
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) ChangeMinterUnsigned(newMinter *keys.PublicKey) (*transaction.Transaction, error) {
|
func (c *Contract) ChangeMinterUnsigned(newMinter *keys.PublicKey) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "changeMinter", nil, newMinter)
|
return c.actor.MakeUnsignedCall(c.hash, "changeMinter", nil, newMinter)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ChangeOwner creates a transaction invoking `changeOwner` method of the contract.
|
// ChangeOwner creates a transaction invoking `changeOwner` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) ChangeOwner(newOwner util.Uint160) (util.Uint256, uint32, error) {
|
func (c *Contract) ChangeOwner(newOwner util.Uint160) (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "changeOwner", newOwner)
|
return c.actor.SendCall(c.hash, "changeOwner", newOwner)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ChangeOwnerTransaction creates a transaction invoking `changeOwner` method of the contract.
|
// ChangeOwnerTransaction creates a transaction invoking `changeOwner` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) ChangeOwnerTransaction(newOwner util.Uint160) (*transaction.Transaction, error) {
|
func (c *Contract) ChangeOwnerTransaction(newOwner util.Uint160) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "changeOwner", newOwner)
|
return c.actor.MakeCall(c.hash, "changeOwner", newOwner)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ChangeOwnerUnsigned creates a transaction invoking `changeOwner` method of the contract.
|
// ChangeOwnerUnsigned creates a transaction invoking `changeOwner` method of the contract.
|
||||||
|
@ -118,21 +135,21 @@ func (c *Contract) ChangeOwnerTransaction(newOwner util.Uint160) (*transaction.T
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) ChangeOwnerUnsigned(newOwner util.Uint160) (*transaction.Transaction, error) {
|
func (c *Contract) ChangeOwnerUnsigned(newOwner util.Uint160) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "changeOwner", nil, newOwner)
|
return c.actor.MakeUnsignedCall(c.hash, "changeOwner", nil, newOwner)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Destroy creates a transaction invoking `destroy` method of the contract.
|
// Destroy creates a transaction invoking `destroy` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) Destroy() (util.Uint256, uint32, error) {
|
func (c *Contract) Destroy() (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "destroy")
|
return c.actor.SendCall(c.hash, "destroy")
|
||||||
}
|
}
|
||||||
|
|
||||||
// DestroyTransaction creates a transaction invoking `destroy` method of the contract.
|
// DestroyTransaction creates a transaction invoking `destroy` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) DestroyTransaction() (*transaction.Transaction, error) {
|
func (c *Contract) DestroyTransaction() (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "destroy")
|
return c.actor.MakeCall(c.hash, "destroy")
|
||||||
}
|
}
|
||||||
|
|
||||||
// DestroyUnsigned creates a transaction invoking `destroy` method of the contract.
|
// DestroyUnsigned creates a transaction invoking `destroy` method of the contract.
|
||||||
|
@ -140,21 +157,21 @@ func (c *Contract) DestroyTransaction() (*transaction.Transaction, error) {
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) DestroyUnsigned() (*transaction.Transaction, error) {
|
func (c *Contract) DestroyUnsigned() (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "destroy", nil)
|
return c.actor.MakeUnsignedCall(c.hash, "destroy", nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
// MaxSupply creates a transaction invoking `maxSupply` method of the contract.
|
// MaxSupply creates a transaction invoking `maxSupply` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) MaxSupply() (util.Uint256, uint32, error) {
|
func (c *Contract) MaxSupply() (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "maxSupply")
|
return c.actor.SendCall(c.hash, "maxSupply")
|
||||||
}
|
}
|
||||||
|
|
||||||
// MaxSupplyTransaction creates a transaction invoking `maxSupply` method of the contract.
|
// MaxSupplyTransaction creates a transaction invoking `maxSupply` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) MaxSupplyTransaction() (*transaction.Transaction, error) {
|
func (c *Contract) MaxSupplyTransaction() (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "maxSupply")
|
return c.actor.MakeCall(c.hash, "maxSupply")
|
||||||
}
|
}
|
||||||
|
|
||||||
// MaxSupplyUnsigned creates a transaction invoking `maxSupply` method of the contract.
|
// MaxSupplyUnsigned creates a transaction invoking `maxSupply` method of the contract.
|
||||||
|
@ -162,43 +179,43 @@ func (c *Contract) MaxSupplyTransaction() (*transaction.Transaction, error) {
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) MaxSupplyUnsigned() (*transaction.Transaction, error) {
|
func (c *Contract) MaxSupplyUnsigned() (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "maxSupply", nil)
|
return c.actor.MakeUnsignedCall(c.hash, "maxSupply", nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Mint creates a transaction invoking `mint` method of the contract.
|
// Mint creates a transaction invoking `mint` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) Mint(from util.Uint160, to util.Uint160, amount *big.Int, swapId *big.Int, signature []byte, data interface{}) (util.Uint256, uint32, error) {
|
func (c *Contract) Mint(from util.Uint160, to util.Uint160, amount *big.Int, swapId *big.Int, signature []byte, data any) (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "mint", from, to, amount, swapId, signature, data)
|
return c.actor.SendCall(c.hash, "mint", from, to, amount, swapId, signature, data)
|
||||||
}
|
}
|
||||||
|
|
||||||
// MintTransaction creates a transaction invoking `mint` method of the contract.
|
// MintTransaction creates a transaction invoking `mint` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) MintTransaction(from util.Uint160, to util.Uint160, amount *big.Int, swapId *big.Int, signature []byte, data interface{}) (*transaction.Transaction, error) {
|
func (c *Contract) MintTransaction(from util.Uint160, to util.Uint160, amount *big.Int, swapId *big.Int, signature []byte, data any) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "mint", from, to, amount, swapId, signature, data)
|
return c.actor.MakeCall(c.hash, "mint", from, to, amount, swapId, signature, data)
|
||||||
}
|
}
|
||||||
|
|
||||||
// MintUnsigned creates a transaction invoking `mint` method of the contract.
|
// MintUnsigned creates a transaction invoking `mint` method of the contract.
|
||||||
// This transaction is not signed, it's simply returned to the caller.
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) MintUnsigned(from util.Uint160, to util.Uint160, amount *big.Int, swapId *big.Int, signature []byte, data interface{}) (*transaction.Transaction, error) {
|
func (c *Contract) MintUnsigned(from util.Uint160, to util.Uint160, amount *big.Int, swapId *big.Int, signature []byte, data any) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "mint", nil, from, to, amount, swapId, signature, data)
|
return c.actor.MakeUnsignedCall(c.hash, "mint", nil, from, to, amount, swapId, signature, data)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update creates a transaction invoking `update` method of the contract.
|
// Update creates a transaction invoking `update` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) Update(nef []byte, manifest []byte) (util.Uint256, uint32, error) {
|
func (c *Contract) Update(nef []byte, manifest []byte) (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "update", nef, manifest)
|
return c.actor.SendCall(c.hash, "update", nef, manifest)
|
||||||
}
|
}
|
||||||
|
|
||||||
// UpdateTransaction creates a transaction invoking `update` method of the contract.
|
// UpdateTransaction creates a transaction invoking `update` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) UpdateTransaction(nef []byte, manifest []byte) (*transaction.Transaction, error) {
|
func (c *Contract) UpdateTransaction(nef []byte, manifest []byte) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "update", nef, manifest)
|
return c.actor.MakeCall(c.hash, "update", nef, manifest)
|
||||||
}
|
}
|
||||||
|
|
||||||
// UpdateUnsigned creates a transaction invoking `update` method of the contract.
|
// UpdateUnsigned creates a transaction invoking `update` method of the contract.
|
||||||
|
@ -206,21 +223,21 @@ func (c *Contract) UpdateTransaction(nef []byte, manifest []byte) (*transaction.
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) UpdateUnsigned(nef []byte, manifest []byte) (*transaction.Transaction, error) {
|
func (c *Contract) UpdateUnsigned(nef []byte, manifest []byte) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "update", nil, nef, manifest)
|
return c.actor.MakeUnsignedCall(c.hash, "update", nil, nef, manifest)
|
||||||
}
|
}
|
||||||
|
|
||||||
// UpdateCap creates a transaction invoking `updateCap` method of the contract.
|
// UpdateCap creates a transaction invoking `updateCap` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) UpdateCap(newCap *big.Int) (util.Uint256, uint32, error) {
|
func (c *Contract) UpdateCap(newCap *big.Int) (util.Uint256, uint32, error) {
|
||||||
return c.actor.SendCall(Hash, "updateCap", newCap)
|
return c.actor.SendCall(c.hash, "updateCap", newCap)
|
||||||
}
|
}
|
||||||
|
|
||||||
// UpdateCapTransaction creates a transaction invoking `updateCap` method of the contract.
|
// UpdateCapTransaction creates a transaction invoking `updateCap` method of the contract.
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) UpdateCapTransaction(newCap *big.Int) (*transaction.Transaction, error) {
|
func (c *Contract) UpdateCapTransaction(newCap *big.Int) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeCall(Hash, "updateCap", newCap)
|
return c.actor.MakeCall(c.hash, "updateCap", newCap)
|
||||||
}
|
}
|
||||||
|
|
||||||
// UpdateCapUnsigned creates a transaction invoking `updateCap` method of the contract.
|
// UpdateCapUnsigned creates a transaction invoking `updateCap` method of the contract.
|
||||||
|
@ -228,5 +245,95 @@ func (c *Contract) UpdateCapTransaction(newCap *big.Int) (*transaction.Transacti
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) UpdateCapUnsigned(newCap *big.Int) (*transaction.Transaction, error) {
|
func (c *Contract) UpdateCapUnsigned(newCap *big.Int) (*transaction.Transaction, error) {
|
||||||
return c.actor.MakeUnsignedCall(Hash, "updateCap", nil, newCap)
|
return c.actor.MakeUnsignedCall(c.hash, "updateCap", nil, newCap)
|
||||||
|
}
|
||||||
|
|
||||||
|
// OnMintEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "OnMint" name from the provided [result.ApplicationLog].
|
||||||
|
func OnMintEventsFromApplicationLog(log *result.ApplicationLog) ([]*OnMintEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*OnMintEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "OnMint" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(OnMintEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize OnMintEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to OnMintEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *OnMintEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 4 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.From, err = func(item stackitem.Item) (util.Uint160, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, err
|
||||||
|
}
|
||||||
|
u, err := util.Uint160DecodeBytesBE(b)
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, err
|
||||||
|
}
|
||||||
|
return u, nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field From: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
index++
|
||||||
|
e.To, err = func(item stackitem.Item) (util.Uint160, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, err
|
||||||
|
}
|
||||||
|
u, err := util.Uint160DecodeBytesBE(b)
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, err
|
||||||
|
}
|
||||||
|
return u, nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field To: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
index++
|
||||||
|
e.Amount, err = arr[index].TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field Amount: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
index++
|
||||||
|
e.SwapId, err = arr[index].TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field SwapId: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
19
cli/smartcontract/testdata/nonepiter/iter.go
vendored
19
cli/smartcontract/testdata/nonepiter/iter.go
vendored
|
@ -1,3 +1,5 @@
|
||||||
|
// Code generated by neo-go contract generate-rpcwrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
// Package nonnepxxcontractwithiterators contains RPC wrappers for Non-NEPXX contract with iterators contract.
|
// Package nonnepxxcontractwithiterators contains RPC wrappers for Non-NEPXX contract with iterators contract.
|
||||||
package nonnepxxcontractwithiterators
|
package nonnepxxcontractwithiterators
|
||||||
|
|
||||||
|
@ -14,8 +16,8 @@ var Hash = util.Uint160{0x33, 0x22, 0x11, 0x0, 0xff, 0xee, 0xdd, 0xcc, 0xbb, 0xa
|
||||||
|
|
||||||
// Invoker is used by ContractReader to call various safe methods.
|
// Invoker is used by ContractReader to call various safe methods.
|
||||||
type Invoker interface {
|
type Invoker interface {
|
||||||
Call(contract util.Uint160, operation string, params ...interface{}) (*result.Invoke, error)
|
Call(contract util.Uint160, operation string, params ...any) (*result.Invoke, error)
|
||||||
CallAndExpandIterator(contract util.Uint160, method string, maxItems int, params ...interface{}) (*result.Invoke, error)
|
CallAndExpandIterator(contract util.Uint160, method string, maxItems int, params ...any) (*result.Invoke, error)
|
||||||
TerminateSession(sessionID uuid.UUID) error
|
TerminateSession(sessionID uuid.UUID) error
|
||||||
TraverseIterator(sessionID uuid.UUID, iterator *result.Iterator, num int) ([]stackitem.Item, error)
|
TraverseIterator(sessionID uuid.UUID, iterator *result.Iterator, num int) ([]stackitem.Item, error)
|
||||||
}
|
}
|
||||||
|
@ -23,17 +25,18 @@ type Invoker interface {
|
||||||
// ContractReader implements safe contract methods.
|
// ContractReader implements safe contract methods.
|
||||||
type ContractReader struct {
|
type ContractReader struct {
|
||||||
invoker Invoker
|
invoker Invoker
|
||||||
|
hash util.Uint160
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewReader creates an instance of ContractReader using Hash and the given Invoker.
|
// NewReader creates an instance of ContractReader using Hash and the given Invoker.
|
||||||
func NewReader(invoker Invoker) *ContractReader {
|
func NewReader(invoker Invoker) *ContractReader {
|
||||||
return &ContractReader{invoker}
|
var hash = Hash
|
||||||
|
return &ContractReader{invoker, hash}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Tokens invokes `tokens` method of contract.
|
// Tokens invokes `tokens` method of contract.
|
||||||
func (c *ContractReader) Tokens() (uuid.UUID, result.Iterator, error) {
|
func (c *ContractReader) Tokens() (uuid.UUID, result.Iterator, error) {
|
||||||
return unwrap.SessionIterator(c.invoker.Call(Hash, "tokens"))
|
return unwrap.SessionIterator(c.invoker.Call(c.hash, "tokens"))
|
||||||
}
|
}
|
||||||
|
|
||||||
// TokensExpanded is similar to Tokens (uses the same contract
|
// TokensExpanded is similar to Tokens (uses the same contract
|
||||||
|
@ -42,12 +45,12 @@ func (c *ContractReader) Tokens() (uuid.UUID, result.Iterator, error) {
|
||||||
// number of result items from the iterator right in the VM and return them to
|
// number of result items from the iterator right in the VM and return them to
|
||||||
// you. It's only limited by VM stack and GAS available for RPC invocations.
|
// you. It's only limited by VM stack and GAS available for RPC invocations.
|
||||||
func (c *ContractReader) TokensExpanded(_numOfIteratorItems int) ([]stackitem.Item, error) {
|
func (c *ContractReader) TokensExpanded(_numOfIteratorItems int) ([]stackitem.Item, error) {
|
||||||
return unwrap.Array(c.invoker.CallAndExpandIterator(Hash, "tokens", _numOfIteratorItems))
|
return unwrap.Array(c.invoker.CallAndExpandIterator(c.hash, "tokens", _numOfIteratorItems))
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetAllRecords invokes `getAllRecords` method of contract.
|
// GetAllRecords invokes `getAllRecords` method of contract.
|
||||||
func (c *ContractReader) GetAllRecords(name string) (uuid.UUID, result.Iterator, error) {
|
func (c *ContractReader) GetAllRecords(name string) (uuid.UUID, result.Iterator, error) {
|
||||||
return unwrap.SessionIterator(c.invoker.Call(Hash, "getAllRecords", name))
|
return unwrap.SessionIterator(c.invoker.Call(c.hash, "getAllRecords", name))
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetAllRecordsExpanded is similar to GetAllRecords (uses the same contract
|
// GetAllRecordsExpanded is similar to GetAllRecords (uses the same contract
|
||||||
|
@ -56,5 +59,5 @@ func (c *ContractReader) GetAllRecords(name string) (uuid.UUID, result.Iterator,
|
||||||
// number of result items from the iterator right in the VM and return them to
|
// number of result items from the iterator right in the VM and return them to
|
||||||
// you. It's only limited by VM stack and GAS available for RPC invocations.
|
// you. It's only limited by VM stack and GAS available for RPC invocations.
|
||||||
func (c *ContractReader) GetAllRecordsExpanded(name string, _numOfIteratorItems int) ([]stackitem.Item, error) {
|
func (c *ContractReader) GetAllRecordsExpanded(name string, _numOfIteratorItems int) ([]stackitem.Item, error) {
|
||||||
return unwrap.Array(c.invoker.CallAndExpandIterator(Hash, "getAllRecords", _numOfIteratorItems, name))
|
return unwrap.Array(c.invoker.CallAndExpandIterator(c.hash, "getAllRecords", _numOfIteratorItems, name))
|
||||||
}
|
}
|
||||||
|
|
7
cli/smartcontract/testdata/rpcbindings/invalid1/invalid.go
vendored
Normal file
7
cli/smartcontract/testdata/rpcbindings/invalid1/invalid.go
vendored
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
package invalid1
|
||||||
|
|
||||||
|
import "github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
||||||
|
|
||||||
|
func Main() {
|
||||||
|
runtime.Notify("Non declared event")
|
||||||
|
}
|
1
cli/smartcontract/testdata/rpcbindings/invalid1/invalid.yml
vendored
Normal file
1
cli/smartcontract/testdata/rpcbindings/invalid1/invalid.yml
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
name: Test undeclared event
|
7
cli/smartcontract/testdata/rpcbindings/invalid2/invalid.go
vendored
Normal file
7
cli/smartcontract/testdata/rpcbindings/invalid2/invalid.go
vendored
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
package invalid2
|
||||||
|
|
||||||
|
import "github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
||||||
|
|
||||||
|
func Main() {
|
||||||
|
runtime.Notify("SomeEvent", "p1", "p2")
|
||||||
|
}
|
6
cli/smartcontract/testdata/rpcbindings/invalid2/invalid.yml
vendored
Normal file
6
cli/smartcontract/testdata/rpcbindings/invalid2/invalid.yml
vendored
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
name: Test undeclared event
|
||||||
|
events:
|
||||||
|
- name: SomeEvent
|
||||||
|
parameters:
|
||||||
|
- name: p1
|
||||||
|
type: String
|
7
cli/smartcontract/testdata/rpcbindings/invalid3/invalid.go
vendored
Normal file
7
cli/smartcontract/testdata/rpcbindings/invalid3/invalid.go
vendored
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
package invalid3
|
||||||
|
|
||||||
|
import "github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
||||||
|
|
||||||
|
func Main() {
|
||||||
|
runtime.Notify("SomeEvent", "p1", 5)
|
||||||
|
}
|
8
cli/smartcontract/testdata/rpcbindings/invalid3/invalid.yml
vendored
Normal file
8
cli/smartcontract/testdata/rpcbindings/invalid3/invalid.yml
vendored
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
name: Test undeclared event
|
||||||
|
events:
|
||||||
|
- name: SomeEvent
|
||||||
|
parameters:
|
||||||
|
- name: p1
|
||||||
|
type: String
|
||||||
|
- name: p2
|
||||||
|
type: String
|
17
cli/smartcontract/testdata/rpcbindings/invalid4/invalid.go
vendored
Normal file
17
cli/smartcontract/testdata/rpcbindings/invalid4/invalid.go
vendored
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
package invalid4
|
||||||
|
|
||||||
|
import "github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
||||||
|
|
||||||
|
type SomeStruct1 struct {
|
||||||
|
Field1 int
|
||||||
|
}
|
||||||
|
|
||||||
|
type SomeStruct2 struct {
|
||||||
|
Field2 string
|
||||||
|
}
|
||||||
|
|
||||||
|
func Main() {
|
||||||
|
// Inconsistent event params usages (different named types throughout the usages).
|
||||||
|
runtime.Notify("SomeEvent", SomeStruct1{Field1: 123})
|
||||||
|
runtime.Notify("SomeEvent", SomeStruct2{Field2: "str"})
|
||||||
|
}
|
6
cli/smartcontract/testdata/rpcbindings/invalid4/invalid.yml
vendored
Normal file
6
cli/smartcontract/testdata/rpcbindings/invalid4/invalid.yml
vendored
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
name: Test undeclared event
|
||||||
|
events:
|
||||||
|
- name: SomeEvent
|
||||||
|
parameters:
|
||||||
|
- name: p1
|
||||||
|
type: Array
|
12
cli/smartcontract/testdata/rpcbindings/invalid5/invalid.go
vendored
Normal file
12
cli/smartcontract/testdata/rpcbindings/invalid5/invalid.go
vendored
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
package invalid5
|
||||||
|
|
||||||
|
import "github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
||||||
|
|
||||||
|
type NamedStruct struct {
|
||||||
|
SomeInt int
|
||||||
|
}
|
||||||
|
|
||||||
|
func Main() NamedStruct {
|
||||||
|
runtime.Notify("SomeEvent", []interface{}{123})
|
||||||
|
return NamedStruct{SomeInt: 123}
|
||||||
|
}
|
16
cli/smartcontract/testdata/rpcbindings/invalid5/invalid.yml
vendored
Normal file
16
cli/smartcontract/testdata/rpcbindings/invalid5/invalid.yml
vendored
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
name: Test undeclared event
|
||||||
|
events:
|
||||||
|
- name: SomeEvent
|
||||||
|
parameters:
|
||||||
|
- name: p1
|
||||||
|
type: Array
|
||||||
|
extendedtype:
|
||||||
|
base: Array
|
||||||
|
name: invalid5.NamedStruct
|
||||||
|
namedtypes:
|
||||||
|
invalid5.NamedStruct:
|
||||||
|
base: Array
|
||||||
|
name: invalid5.NamedStruct
|
||||||
|
fields:
|
||||||
|
- field: SomeInt
|
||||||
|
base: Integer
|
14
cli/smartcontract/testdata/rpcbindings/invalid6/invalid.go
vendored
Normal file
14
cli/smartcontract/testdata/rpcbindings/invalid6/invalid.go
vendored
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
package invalid6
|
||||||
|
|
||||||
|
import "github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
||||||
|
|
||||||
|
type SomeStruct struct {
|
||||||
|
Field int
|
||||||
|
// RPC binding generator will convert this field into exported, which matches
|
||||||
|
// exactly the existing Field.
|
||||||
|
field int
|
||||||
|
}
|
||||||
|
|
||||||
|
func Main() {
|
||||||
|
runtime.Notify("SomeEvent", SomeStruct{Field: 123, field: 123})
|
||||||
|
}
|
18
cli/smartcontract/testdata/rpcbindings/invalid6/invalid.yml
vendored
Normal file
18
cli/smartcontract/testdata/rpcbindings/invalid6/invalid.yml
vendored
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
name: Test duplicating event fields
|
||||||
|
events:
|
||||||
|
- name: SomeEvent
|
||||||
|
parameters:
|
||||||
|
- name: p1
|
||||||
|
type: Struct
|
||||||
|
extendedtype:
|
||||||
|
base: Struct
|
||||||
|
name: SomeStruct
|
||||||
|
namedtypes:
|
||||||
|
SomeStruct:
|
||||||
|
base: Struct
|
||||||
|
name: SomeStruct
|
||||||
|
fields:
|
||||||
|
- field: Field
|
||||||
|
base: Integer
|
||||||
|
- field: field
|
||||||
|
base: Integer
|
14
cli/smartcontract/testdata/rpcbindings/invalid7/invalid.go
vendored
Normal file
14
cli/smartcontract/testdata/rpcbindings/invalid7/invalid.go
vendored
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
package invalid7
|
||||||
|
|
||||||
|
import "github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
||||||
|
|
||||||
|
type SomeStruct struct {
|
||||||
|
Field int
|
||||||
|
// RPC binding generator will convert this field into exported, which matches
|
||||||
|
// exactly the existing Field.
|
||||||
|
field int
|
||||||
|
}
|
||||||
|
|
||||||
|
func Main() {
|
||||||
|
runtime.Notify("SomeEvent", SomeStruct{Field: 123, field: 123})
|
||||||
|
}
|
6
cli/smartcontract/testdata/rpcbindings/invalid7/invalid.yml
vendored
Normal file
6
cli/smartcontract/testdata/rpcbindings/invalid7/invalid.yml
vendored
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
name: Test duplicating autogenerated event fields
|
||||||
|
events:
|
||||||
|
- name: SomeEvent
|
||||||
|
parameters:
|
||||||
|
- name: p1
|
||||||
|
type: Struct
|
16
cli/smartcontract/testdata/rpcbindings/invalid8/invalid.go
vendored
Normal file
16
cli/smartcontract/testdata/rpcbindings/invalid8/invalid.go
vendored
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
package invalid8
|
||||||
|
|
||||||
|
type SomeStruct struct {
|
||||||
|
Field int
|
||||||
|
// RPC binding generator will convert this field into exported, which matches
|
||||||
|
// exactly the existing Field.
|
||||||
|
field int
|
||||||
|
}
|
||||||
|
|
||||||
|
func Main() SomeStruct {
|
||||||
|
s := SomeStruct{
|
||||||
|
Field: 1,
|
||||||
|
field: 2,
|
||||||
|
}
|
||||||
|
return s
|
||||||
|
}
|
1
cli/smartcontract/testdata/rpcbindings/invalid8/invalid.yml
vendored
Normal file
1
cli/smartcontract/testdata/rpcbindings/invalid8/invalid.yml
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
name: Test duplicating struct fields
|
23
cli/smartcontract/testdata/rpcbindings/notifications/config.yml
vendored
Normal file
23
cli/smartcontract/testdata/rpcbindings/notifications/config.yml
vendored
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
name: "Notifications"
|
||||||
|
sourceurl: https://github.com/nspcc-dev/neo-go/
|
||||||
|
events:
|
||||||
|
- name: "! complicated name %$#"
|
||||||
|
parameters:
|
||||||
|
- name: ! complicated param @#$%
|
||||||
|
type: String
|
||||||
|
- name: "SomeMap"
|
||||||
|
parameters:
|
||||||
|
- name: m
|
||||||
|
type: Map
|
||||||
|
- name: "SomeStruct"
|
||||||
|
parameters:
|
||||||
|
- name: s
|
||||||
|
type: Struct
|
||||||
|
- name: "SomeArray"
|
||||||
|
parameters:
|
||||||
|
- name: a
|
||||||
|
type: Array
|
||||||
|
- name: "SomeUnexportedField"
|
||||||
|
parameters:
|
||||||
|
- name: s
|
||||||
|
type: Struct
|
60
cli/smartcontract/testdata/rpcbindings/notifications/config_extended.yml
vendored
Normal file
60
cli/smartcontract/testdata/rpcbindings/notifications/config_extended.yml
vendored
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
name: "Notifications"
|
||||||
|
sourceurl: https://github.com/nspcc-dev/neo-go/
|
||||||
|
events:
|
||||||
|
- name: "! complicated name %$#"
|
||||||
|
parameters:
|
||||||
|
- name: ! complicated param @#$%
|
||||||
|
type: String
|
||||||
|
- name: "SomeMap"
|
||||||
|
parameters:
|
||||||
|
- name: m
|
||||||
|
type: Map
|
||||||
|
extendedtype:
|
||||||
|
base: Map
|
||||||
|
key: Integer
|
||||||
|
value:
|
||||||
|
base: Map
|
||||||
|
key: String
|
||||||
|
value:
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: Hash160
|
||||||
|
- name: "SomeStruct"
|
||||||
|
parameters:
|
||||||
|
- name: s
|
||||||
|
type: Struct
|
||||||
|
extendedtype:
|
||||||
|
base: Struct
|
||||||
|
name: crazyStruct
|
||||||
|
- name: "SomeArray"
|
||||||
|
parameters:
|
||||||
|
- name: a
|
||||||
|
type: Array
|
||||||
|
extendedtype:
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: Integer
|
||||||
|
- name: "SomeUnexportedField"
|
||||||
|
parameters:
|
||||||
|
- name: s
|
||||||
|
type: Struct
|
||||||
|
extendedtype:
|
||||||
|
base: Struct
|
||||||
|
name: simpleStruct
|
||||||
|
namedtypes:
|
||||||
|
crazyStruct:
|
||||||
|
base: Struct
|
||||||
|
name: crazyStruct
|
||||||
|
fields:
|
||||||
|
- field: I
|
||||||
|
base: Integer
|
||||||
|
- field: B
|
||||||
|
base: Boolean
|
||||||
|
simpleStruct:
|
||||||
|
base: Struct
|
||||||
|
name: simpleStruct
|
||||||
|
fields:
|
||||||
|
- field: i
|
||||||
|
base: Integer
|
23
cli/smartcontract/testdata/rpcbindings/notifications/config_guessed.yml
vendored
Normal file
23
cli/smartcontract/testdata/rpcbindings/notifications/config_guessed.yml
vendored
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
name: "Notifications"
|
||||||
|
sourceurl: https://github.com/nspcc-dev/neo-go/
|
||||||
|
events:
|
||||||
|
- name: "! complicated name %$#"
|
||||||
|
parameters:
|
||||||
|
- name: ! complicated param @#$%
|
||||||
|
type: String
|
||||||
|
- name: "SomeMap"
|
||||||
|
parameters:
|
||||||
|
- name: m
|
||||||
|
type: Map
|
||||||
|
- name: "SomeStruct"
|
||||||
|
parameters:
|
||||||
|
- name: s
|
||||||
|
type: Struct
|
||||||
|
- name: "SomeArray"
|
||||||
|
parameters:
|
||||||
|
- name: a
|
||||||
|
type: Array
|
||||||
|
- name: "SomeUnexportedField"
|
||||||
|
parameters:
|
||||||
|
- name: s
|
||||||
|
type: Struct
|
33
cli/smartcontract/testdata/rpcbindings/notifications/notifications.go
vendored
Normal file
33
cli/smartcontract/testdata/rpcbindings/notifications/notifications.go
vendored
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
package structs
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/interop"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
||||||
|
)
|
||||||
|
|
||||||
|
func Main() {
|
||||||
|
runtime.Notify("! complicated name %$#", "str1")
|
||||||
|
}
|
||||||
|
|
||||||
|
func CrazyMap() {
|
||||||
|
runtime.Notify("SomeMap", map[int][]map[string][]interop.Hash160{})
|
||||||
|
}
|
||||||
|
|
||||||
|
func Struct() {
|
||||||
|
runtime.Notify("SomeStruct", struct {
|
||||||
|
I int
|
||||||
|
B bool
|
||||||
|
}{I: 123, B: true})
|
||||||
|
}
|
||||||
|
|
||||||
|
func Array() {
|
||||||
|
runtime.Notify("SomeArray", [][]int{})
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnexportedField emits notification with unexported field that must be converted
|
||||||
|
// to exported in the resulting RPC binding.
|
||||||
|
func UnexportedField() {
|
||||||
|
runtime.Notify("SomeUnexportedField", struct {
|
||||||
|
i int
|
||||||
|
}{i: 123})
|
||||||
|
}
|
500
cli/smartcontract/testdata/rpcbindings/notifications/rpcbindings.out
vendored
Normal file
500
cli/smartcontract/testdata/rpcbindings/notifications/rpcbindings.out
vendored
Normal file
|
@ -0,0 +1,500 @@
|
||||||
|
// Code generated by neo-go contract generate-rpcwrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
|
// Package structs contains RPC wrappers for Notifications contract.
|
||||||
|
package structs
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/neorpc/result"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
||||||
|
"unicode/utf8"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Hash contains contract hash.
|
||||||
|
var Hash = util.Uint160{0x33, 0x22, 0x11, 0x0, 0xff, 0xee, 0xdd, 0xcc, 0xbb, 0xaa, 0x99, 0x88, 0x77, 0x66, 0x55, 0x44, 0x33, 0x22, 0x11, 0x0}
|
||||||
|
|
||||||
|
// ComplicatedNameEvent represents "! complicated name %$#" event emitted by the contract.
|
||||||
|
type ComplicatedNameEvent struct {
|
||||||
|
ComplicatedParam string
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeMapEvent represents "SomeMap" event emitted by the contract.
|
||||||
|
type SomeMapEvent struct {
|
||||||
|
M map[any]any
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeStructEvent represents "SomeStruct" event emitted by the contract.
|
||||||
|
type SomeStructEvent struct {
|
||||||
|
S []any
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeArrayEvent represents "SomeArray" event emitted by the contract.
|
||||||
|
type SomeArrayEvent struct {
|
||||||
|
A []any
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeUnexportedFieldEvent represents "SomeUnexportedField" event emitted by the contract.
|
||||||
|
type SomeUnexportedFieldEvent struct {
|
||||||
|
S []any
|
||||||
|
}
|
||||||
|
|
||||||
|
// Actor is used by Contract to call state-changing methods.
|
||||||
|
type Actor interface {
|
||||||
|
MakeCall(contract util.Uint160, method string, params ...any) (*transaction.Transaction, error)
|
||||||
|
MakeRun(script []byte) (*transaction.Transaction, error)
|
||||||
|
MakeUnsignedCall(contract util.Uint160, method string, attrs []transaction.Attribute, params ...any) (*transaction.Transaction, error)
|
||||||
|
MakeUnsignedRun(script []byte, attrs []transaction.Attribute) (*transaction.Transaction, error)
|
||||||
|
SendCall(contract util.Uint160, method string, params ...any) (util.Uint256, uint32, error)
|
||||||
|
SendRun(script []byte) (util.Uint256, uint32, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Contract implements all contract methods.
|
||||||
|
type Contract struct {
|
||||||
|
actor Actor
|
||||||
|
hash util.Uint160
|
||||||
|
}
|
||||||
|
|
||||||
|
// New creates an instance of Contract using Hash and the given Actor.
|
||||||
|
func New(actor Actor) *Contract {
|
||||||
|
var hash = Hash
|
||||||
|
return &Contract{actor, hash}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Array creates a transaction invoking `array` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) Array() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "array")
|
||||||
|
}
|
||||||
|
|
||||||
|
// ArrayTransaction creates a transaction invoking `array` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) ArrayTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "array")
|
||||||
|
}
|
||||||
|
|
||||||
|
// ArrayUnsigned creates a transaction invoking `array` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) ArrayUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "array", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CrazyMap creates a transaction invoking `crazyMap` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) CrazyMap() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "crazyMap")
|
||||||
|
}
|
||||||
|
|
||||||
|
// CrazyMapTransaction creates a transaction invoking `crazyMap` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) CrazyMapTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "crazyMap")
|
||||||
|
}
|
||||||
|
|
||||||
|
// CrazyMapUnsigned creates a transaction invoking `crazyMap` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) CrazyMapUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "crazyMap", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Main creates a transaction invoking `main` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) Main() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "main")
|
||||||
|
}
|
||||||
|
|
||||||
|
// MainTransaction creates a transaction invoking `main` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) MainTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "main")
|
||||||
|
}
|
||||||
|
|
||||||
|
// MainUnsigned creates a transaction invoking `main` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) MainUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "main", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Struct creates a transaction invoking `struct` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) Struct() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "struct")
|
||||||
|
}
|
||||||
|
|
||||||
|
// StructTransaction creates a transaction invoking `struct` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) StructTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "struct")
|
||||||
|
}
|
||||||
|
|
||||||
|
// StructUnsigned creates a transaction invoking `struct` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) StructUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "struct", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnexportedField creates a transaction invoking `unexportedField` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) UnexportedField() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "unexportedField")
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnexportedFieldTransaction creates a transaction invoking `unexportedField` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) UnexportedFieldTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "unexportedField")
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnexportedFieldUnsigned creates a transaction invoking `unexportedField` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) UnexportedFieldUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "unexportedField", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ComplicatedNameEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "! complicated name %$#" name from the provided [result.ApplicationLog].
|
||||||
|
func ComplicatedNameEventsFromApplicationLog(log *result.ApplicationLog) ([]*ComplicatedNameEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*ComplicatedNameEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "! complicated name %$#" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(ComplicatedNameEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize ComplicatedNameEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to ComplicatedNameEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *ComplicatedNameEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.ComplicatedParam, err = func(item stackitem.Item) (string, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if !utf8.Valid(b) {
|
||||||
|
return "", errors.New("not a UTF-8 string")
|
||||||
|
}
|
||||||
|
return string(b), nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field ComplicatedParam: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeMapEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "SomeMap" name from the provided [result.ApplicationLog].
|
||||||
|
func SomeMapEventsFromApplicationLog(log *result.ApplicationLog) ([]*SomeMapEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*SomeMapEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "SomeMap" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(SomeMapEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize SomeMapEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to SomeMapEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *SomeMapEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.M, err = func(item stackitem.Item) (map[any]any, error) {
|
||||||
|
m, ok := item.Value().([]stackitem.MapElement)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("%s is not a map", item.Type().String())
|
||||||
|
}
|
||||||
|
res := make(map[any]any)
|
||||||
|
for i := range m {
|
||||||
|
k, err := m[i].Key.Value(), error(nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("key %d: %w", i, err)
|
||||||
|
}
|
||||||
|
v, err := m[i].Value.Value(), error(nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("value %d: %w", i, err)
|
||||||
|
}
|
||||||
|
res[k] = v
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field M: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeStructEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "SomeStruct" name from the provided [result.ApplicationLog].
|
||||||
|
func SomeStructEventsFromApplicationLog(log *result.ApplicationLog) ([]*SomeStructEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*SomeStructEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "SomeStruct" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(SomeStructEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize SomeStructEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to SomeStructEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *SomeStructEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.S, err = func(item stackitem.Item) ([]any, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([]any, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = arr[i].Value(), error(nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field S: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeArrayEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "SomeArray" name from the provided [result.ApplicationLog].
|
||||||
|
func SomeArrayEventsFromApplicationLog(log *result.ApplicationLog) ([]*SomeArrayEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*SomeArrayEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "SomeArray" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(SomeArrayEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize SomeArrayEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to SomeArrayEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *SomeArrayEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.A, err = func(item stackitem.Item) ([]any, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([]any, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = arr[i].Value(), error(nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field A: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeUnexportedFieldEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "SomeUnexportedField" name from the provided [result.ApplicationLog].
|
||||||
|
func SomeUnexportedFieldEventsFromApplicationLog(log *result.ApplicationLog) ([]*SomeUnexportedFieldEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*SomeUnexportedFieldEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "SomeUnexportedField" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(SomeUnexportedFieldEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize SomeUnexportedFieldEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to SomeUnexportedFieldEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *SomeUnexportedFieldEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.S, err = func(item stackitem.Item) ([]any, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([]any, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = arr[i].Value(), error(nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field S: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
623
cli/smartcontract/testdata/rpcbindings/notifications/rpcbindings_extended.out
vendored
Executable file
623
cli/smartcontract/testdata/rpcbindings/notifications/rpcbindings_extended.out
vendored
Executable file
|
@ -0,0 +1,623 @@
|
||||||
|
// Code generated by neo-go contract generate-rpcwrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
|
// Package structs contains RPC wrappers for Notifications contract.
|
||||||
|
package structs
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/neorpc/result"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
||||||
|
"math/big"
|
||||||
|
"unicode/utf8"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Hash contains contract hash.
|
||||||
|
var Hash = util.Uint160{0x33, 0x22, 0x11, 0x0, 0xff, 0xee, 0xdd, 0xcc, 0xbb, 0xaa, 0x99, 0x88, 0x77, 0x66, 0x55, 0x44, 0x33, 0x22, 0x11, 0x0}
|
||||||
|
|
||||||
|
// CrazyStruct is a contract-specific crazyStruct type used by its methods.
|
||||||
|
type CrazyStruct struct {
|
||||||
|
I *big.Int
|
||||||
|
B bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// SimpleStruct is a contract-specific simpleStruct type used by its methods.
|
||||||
|
type SimpleStruct struct {
|
||||||
|
I *big.Int
|
||||||
|
}
|
||||||
|
|
||||||
|
// ComplicatedNameEvent represents "! complicated name %$#" event emitted by the contract.
|
||||||
|
type ComplicatedNameEvent struct {
|
||||||
|
ComplicatedParam string
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeMapEvent represents "SomeMap" event emitted by the contract.
|
||||||
|
type SomeMapEvent struct {
|
||||||
|
M map[*big.Int]map[string][]util.Uint160
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeStructEvent represents "SomeStruct" event emitted by the contract.
|
||||||
|
type SomeStructEvent struct {
|
||||||
|
S *CrazyStruct
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeArrayEvent represents "SomeArray" event emitted by the contract.
|
||||||
|
type SomeArrayEvent struct {
|
||||||
|
A [][]*big.Int
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeUnexportedFieldEvent represents "SomeUnexportedField" event emitted by the contract.
|
||||||
|
type SomeUnexportedFieldEvent struct {
|
||||||
|
S *SimpleStruct
|
||||||
|
}
|
||||||
|
|
||||||
|
// Actor is used by Contract to call state-changing methods.
|
||||||
|
type Actor interface {
|
||||||
|
MakeCall(contract util.Uint160, method string, params ...any) (*transaction.Transaction, error)
|
||||||
|
MakeRun(script []byte) (*transaction.Transaction, error)
|
||||||
|
MakeUnsignedCall(contract util.Uint160, method string, attrs []transaction.Attribute, params ...any) (*transaction.Transaction, error)
|
||||||
|
MakeUnsignedRun(script []byte, attrs []transaction.Attribute) (*transaction.Transaction, error)
|
||||||
|
SendCall(contract util.Uint160, method string, params ...any) (util.Uint256, uint32, error)
|
||||||
|
SendRun(script []byte) (util.Uint256, uint32, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Contract implements all contract methods.
|
||||||
|
type Contract struct {
|
||||||
|
actor Actor
|
||||||
|
hash util.Uint160
|
||||||
|
}
|
||||||
|
|
||||||
|
// New creates an instance of Contract using Hash and the given Actor.
|
||||||
|
func New(actor Actor) *Contract {
|
||||||
|
var hash = Hash
|
||||||
|
return &Contract{actor, hash}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Array creates a transaction invoking `array` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) Array() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "array")
|
||||||
|
}
|
||||||
|
|
||||||
|
// ArrayTransaction creates a transaction invoking `array` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) ArrayTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "array")
|
||||||
|
}
|
||||||
|
|
||||||
|
// ArrayUnsigned creates a transaction invoking `array` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) ArrayUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "array", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CrazyMap creates a transaction invoking `crazyMap` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) CrazyMap() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "crazyMap")
|
||||||
|
}
|
||||||
|
|
||||||
|
// CrazyMapTransaction creates a transaction invoking `crazyMap` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) CrazyMapTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "crazyMap")
|
||||||
|
}
|
||||||
|
|
||||||
|
// CrazyMapUnsigned creates a transaction invoking `crazyMap` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) CrazyMapUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "crazyMap", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Main creates a transaction invoking `main` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) Main() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "main")
|
||||||
|
}
|
||||||
|
|
||||||
|
// MainTransaction creates a transaction invoking `main` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) MainTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "main")
|
||||||
|
}
|
||||||
|
|
||||||
|
// MainUnsigned creates a transaction invoking `main` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) MainUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "main", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Struct creates a transaction invoking `struct` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) Struct() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "struct")
|
||||||
|
}
|
||||||
|
|
||||||
|
// StructTransaction creates a transaction invoking `struct` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) StructTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "struct")
|
||||||
|
}
|
||||||
|
|
||||||
|
// StructUnsigned creates a transaction invoking `struct` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) StructUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "struct", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnexportedField creates a transaction invoking `unexportedField` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) UnexportedField() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "unexportedField")
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnexportedFieldTransaction creates a transaction invoking `unexportedField` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) UnexportedFieldTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "unexportedField")
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnexportedFieldUnsigned creates a transaction invoking `unexportedField` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) UnexportedFieldUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "unexportedField", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// itemToCrazyStruct converts stack item into *CrazyStruct.
|
||||||
|
func itemToCrazyStruct(item stackitem.Item, err error) (*CrazyStruct, error) {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var res = new(CrazyStruct)
|
||||||
|
err = res.FromStackItem(item)
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem retrieves fields of CrazyStruct from the given
|
||||||
|
// [stackitem.Item] or returns an error if it's not possible to do to so.
|
||||||
|
func (res *CrazyStruct) FromStackItem(item stackitem.Item) error {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 2 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
res.I, err = arr[index].TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field I: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
index++
|
||||||
|
res.B, err = arr[index].TryBool()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field B: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// itemToSimpleStruct converts stack item into *SimpleStruct.
|
||||||
|
func itemToSimpleStruct(item stackitem.Item, err error) (*SimpleStruct, error) {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var res = new(SimpleStruct)
|
||||||
|
err = res.FromStackItem(item)
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem retrieves fields of SimpleStruct from the given
|
||||||
|
// [stackitem.Item] or returns an error if it's not possible to do to so.
|
||||||
|
func (res *SimpleStruct) FromStackItem(item stackitem.Item) error {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
res.I, err = arr[index].TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field I: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ComplicatedNameEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "! complicated name %$#" name from the provided [result.ApplicationLog].
|
||||||
|
func ComplicatedNameEventsFromApplicationLog(log *result.ApplicationLog) ([]*ComplicatedNameEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*ComplicatedNameEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "! complicated name %$#" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(ComplicatedNameEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize ComplicatedNameEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to ComplicatedNameEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *ComplicatedNameEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.ComplicatedParam, err = func(item stackitem.Item) (string, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if !utf8.Valid(b) {
|
||||||
|
return "", errors.New("not a UTF-8 string")
|
||||||
|
}
|
||||||
|
return string(b), nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field ComplicatedParam: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeMapEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "SomeMap" name from the provided [result.ApplicationLog].
|
||||||
|
func SomeMapEventsFromApplicationLog(log *result.ApplicationLog) ([]*SomeMapEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*SomeMapEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "SomeMap" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(SomeMapEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize SomeMapEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to SomeMapEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *SomeMapEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.M, err = func(item stackitem.Item) (map[*big.Int]map[string][]util.Uint160, error) {
|
||||||
|
m, ok := item.Value().([]stackitem.MapElement)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("%s is not a map", item.Type().String())
|
||||||
|
}
|
||||||
|
res := make(map[*big.Int]map[string][]util.Uint160)
|
||||||
|
for i := range m {
|
||||||
|
k, err := m[i].Key.TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("key %d: %w", i, err)
|
||||||
|
}
|
||||||
|
v, err := func(item stackitem.Item) (map[string][]util.Uint160, error) {
|
||||||
|
m, ok := item.Value().([]stackitem.MapElement)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("%s is not a map", item.Type().String())
|
||||||
|
}
|
||||||
|
res := make(map[string][]util.Uint160)
|
||||||
|
for i := range m {
|
||||||
|
k, err := func(item stackitem.Item) (string, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if !utf8.Valid(b) {
|
||||||
|
return "", errors.New("not a UTF-8 string")
|
||||||
|
}
|
||||||
|
return string(b), nil
|
||||||
|
}(m[i].Key)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("key %d: %w", i, err)
|
||||||
|
}
|
||||||
|
v, err := func(item stackitem.Item) ([]util.Uint160, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([]util.Uint160, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = func(item stackitem.Item) (util.Uint160, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, err
|
||||||
|
}
|
||||||
|
u, err := util.Uint160DecodeBytesBE(b)
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, err
|
||||||
|
}
|
||||||
|
return u, nil
|
||||||
|
}(arr[i])
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(m[i].Value)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("value %d: %w", i, err)
|
||||||
|
}
|
||||||
|
res[k] = v
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(m[i].Value)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("value %d: %w", i, err)
|
||||||
|
}
|
||||||
|
res[k] = v
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field M: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeStructEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "SomeStruct" name from the provided [result.ApplicationLog].
|
||||||
|
func SomeStructEventsFromApplicationLog(log *result.ApplicationLog) ([]*SomeStructEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*SomeStructEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "SomeStruct" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(SomeStructEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize SomeStructEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to SomeStructEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *SomeStructEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.S, err = itemToCrazyStruct(arr[index], nil)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field S: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeArrayEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "SomeArray" name from the provided [result.ApplicationLog].
|
||||||
|
func SomeArrayEventsFromApplicationLog(log *result.ApplicationLog) ([]*SomeArrayEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*SomeArrayEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "SomeArray" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(SomeArrayEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize SomeArrayEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to SomeArrayEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *SomeArrayEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.A, err = func(item stackitem.Item) ([][]*big.Int, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([][]*big.Int, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = func(item stackitem.Item) ([]*big.Int, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([]*big.Int, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = arr[i].TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[i])
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field A: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeUnexportedFieldEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "SomeUnexportedField" name from the provided [result.ApplicationLog].
|
||||||
|
func SomeUnexportedFieldEventsFromApplicationLog(log *result.ApplicationLog) ([]*SomeUnexportedFieldEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*SomeUnexportedFieldEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "SomeUnexportedField" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(SomeUnexportedFieldEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize SomeUnexportedFieldEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to SomeUnexportedFieldEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *SomeUnexportedFieldEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.S, err = itemToSimpleStruct(arr[index], nil)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field S: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
636
cli/smartcontract/testdata/rpcbindings/notifications/rpcbindings_guessed.out
vendored
Executable file
636
cli/smartcontract/testdata/rpcbindings/notifications/rpcbindings_guessed.out
vendored
Executable file
|
@ -0,0 +1,636 @@
|
||||||
|
// Code generated by neo-go contract generate-rpcwrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
|
// Package structs contains RPC wrappers for Notifications contract.
|
||||||
|
package structs
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/neorpc/result"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
||||||
|
"math/big"
|
||||||
|
"unicode/utf8"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Hash contains contract hash.
|
||||||
|
var Hash = util.Uint160{0x33, 0x22, 0x11, 0x0, 0xff, 0xee, 0xdd, 0xcc, 0xbb, 0xaa, 0x99, 0x88, 0x77, 0x66, 0x55, 0x44, 0x33, 0x22, 0x11, 0x0}
|
||||||
|
|
||||||
|
// Unnamed is a contract-specific unnamed type used by its methods.
|
||||||
|
type Unnamed struct {
|
||||||
|
I *big.Int
|
||||||
|
B bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnnamedX is a contract-specific unnamedX type used by its methods.
|
||||||
|
type UnnamedX struct {
|
||||||
|
I *big.Int
|
||||||
|
}
|
||||||
|
|
||||||
|
// ComplicatedNameEvent represents "! complicated name %$#" event emitted by the contract.
|
||||||
|
type ComplicatedNameEvent struct {
|
||||||
|
ComplicatedParam string
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeMapEvent represents "SomeMap" event emitted by the contract.
|
||||||
|
type SomeMapEvent struct {
|
||||||
|
M map[*big.Int][]map[string][]util.Uint160
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeStructEvent represents "SomeStruct" event emitted by the contract.
|
||||||
|
type SomeStructEvent struct {
|
||||||
|
S *Unnamed
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeArrayEvent represents "SomeArray" event emitted by the contract.
|
||||||
|
type SomeArrayEvent struct {
|
||||||
|
A [][]*big.Int
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeUnexportedFieldEvent represents "SomeUnexportedField" event emitted by the contract.
|
||||||
|
type SomeUnexportedFieldEvent struct {
|
||||||
|
S *UnnamedX
|
||||||
|
}
|
||||||
|
|
||||||
|
// Actor is used by Contract to call state-changing methods.
|
||||||
|
type Actor interface {
|
||||||
|
MakeCall(contract util.Uint160, method string, params ...any) (*transaction.Transaction, error)
|
||||||
|
MakeRun(script []byte) (*transaction.Transaction, error)
|
||||||
|
MakeUnsignedCall(contract util.Uint160, method string, attrs []transaction.Attribute, params ...any) (*transaction.Transaction, error)
|
||||||
|
MakeUnsignedRun(script []byte, attrs []transaction.Attribute) (*transaction.Transaction, error)
|
||||||
|
SendCall(contract util.Uint160, method string, params ...any) (util.Uint256, uint32, error)
|
||||||
|
SendRun(script []byte) (util.Uint256, uint32, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Contract implements all contract methods.
|
||||||
|
type Contract struct {
|
||||||
|
actor Actor
|
||||||
|
hash util.Uint160
|
||||||
|
}
|
||||||
|
|
||||||
|
// New creates an instance of Contract using Hash and the given Actor.
|
||||||
|
func New(actor Actor) *Contract {
|
||||||
|
var hash = Hash
|
||||||
|
return &Contract{actor, hash}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Array creates a transaction invoking `array` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) Array() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "array")
|
||||||
|
}
|
||||||
|
|
||||||
|
// ArrayTransaction creates a transaction invoking `array` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) ArrayTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "array")
|
||||||
|
}
|
||||||
|
|
||||||
|
// ArrayUnsigned creates a transaction invoking `array` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) ArrayUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "array", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CrazyMap creates a transaction invoking `crazyMap` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) CrazyMap() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "crazyMap")
|
||||||
|
}
|
||||||
|
|
||||||
|
// CrazyMapTransaction creates a transaction invoking `crazyMap` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) CrazyMapTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "crazyMap")
|
||||||
|
}
|
||||||
|
|
||||||
|
// CrazyMapUnsigned creates a transaction invoking `crazyMap` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) CrazyMapUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "crazyMap", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Main creates a transaction invoking `main` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) Main() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "main")
|
||||||
|
}
|
||||||
|
|
||||||
|
// MainTransaction creates a transaction invoking `main` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) MainTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "main")
|
||||||
|
}
|
||||||
|
|
||||||
|
// MainUnsigned creates a transaction invoking `main` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) MainUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "main", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Struct creates a transaction invoking `struct` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) Struct() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "struct")
|
||||||
|
}
|
||||||
|
|
||||||
|
// StructTransaction creates a transaction invoking `struct` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) StructTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "struct")
|
||||||
|
}
|
||||||
|
|
||||||
|
// StructUnsigned creates a transaction invoking `struct` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) StructUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "struct", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnexportedField creates a transaction invoking `unexportedField` method of the contract.
|
||||||
|
// This transaction is signed and immediately sent to the network.
|
||||||
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
|
func (c *Contract) UnexportedField() (util.Uint256, uint32, error) {
|
||||||
|
return c.actor.SendCall(c.hash, "unexportedField")
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnexportedFieldTransaction creates a transaction invoking `unexportedField` method of the contract.
|
||||||
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
|
// returned to the caller.
|
||||||
|
func (c *Contract) UnexportedFieldTransaction() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeCall(c.hash, "unexportedField")
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnexportedFieldUnsigned creates a transaction invoking `unexportedField` method of the contract.
|
||||||
|
// This transaction is not signed, it's simply returned to the caller.
|
||||||
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
|
func (c *Contract) UnexportedFieldUnsigned() (*transaction.Transaction, error) {
|
||||||
|
return c.actor.MakeUnsignedCall(c.hash, "unexportedField", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// itemToUnnamed converts stack item into *Unnamed.
|
||||||
|
func itemToUnnamed(item stackitem.Item, err error) (*Unnamed, error) {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var res = new(Unnamed)
|
||||||
|
err = res.FromStackItem(item)
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem retrieves fields of Unnamed from the given
|
||||||
|
// [stackitem.Item] or returns an error if it's not possible to do to so.
|
||||||
|
func (res *Unnamed) FromStackItem(item stackitem.Item) error {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 2 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
res.I, err = arr[index].TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field I: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
index++
|
||||||
|
res.B, err = arr[index].TryBool()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field B: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// itemToUnnamedX converts stack item into *UnnamedX.
|
||||||
|
func itemToUnnamedX(item stackitem.Item, err error) (*UnnamedX, error) {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var res = new(UnnamedX)
|
||||||
|
err = res.FromStackItem(item)
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem retrieves fields of UnnamedX from the given
|
||||||
|
// [stackitem.Item] or returns an error if it's not possible to do to so.
|
||||||
|
func (res *UnnamedX) FromStackItem(item stackitem.Item) error {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
res.I, err = arr[index].TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field I: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ComplicatedNameEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "! complicated name %$#" name from the provided [result.ApplicationLog].
|
||||||
|
func ComplicatedNameEventsFromApplicationLog(log *result.ApplicationLog) ([]*ComplicatedNameEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*ComplicatedNameEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "! complicated name %$#" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(ComplicatedNameEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize ComplicatedNameEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to ComplicatedNameEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *ComplicatedNameEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.ComplicatedParam, err = func(item stackitem.Item) (string, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if !utf8.Valid(b) {
|
||||||
|
return "", errors.New("not a UTF-8 string")
|
||||||
|
}
|
||||||
|
return string(b), nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field ComplicatedParam: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeMapEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "SomeMap" name from the provided [result.ApplicationLog].
|
||||||
|
func SomeMapEventsFromApplicationLog(log *result.ApplicationLog) ([]*SomeMapEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*SomeMapEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "SomeMap" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(SomeMapEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize SomeMapEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to SomeMapEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *SomeMapEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.M, err = func(item stackitem.Item) (map[*big.Int][]map[string][]util.Uint160, error) {
|
||||||
|
m, ok := item.Value().([]stackitem.MapElement)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("%s is not a map", item.Type().String())
|
||||||
|
}
|
||||||
|
res := make(map[*big.Int][]map[string][]util.Uint160)
|
||||||
|
for i := range m {
|
||||||
|
k, err := m[i].Key.TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("key %d: %w", i, err)
|
||||||
|
}
|
||||||
|
v, err := func(item stackitem.Item) ([]map[string][]util.Uint160, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([]map[string][]util.Uint160, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = func(item stackitem.Item) (map[string][]util.Uint160, error) {
|
||||||
|
m, ok := item.Value().([]stackitem.MapElement)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("%s is not a map", item.Type().String())
|
||||||
|
}
|
||||||
|
res := make(map[string][]util.Uint160)
|
||||||
|
for i := range m {
|
||||||
|
k, err := func(item stackitem.Item) (string, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if !utf8.Valid(b) {
|
||||||
|
return "", errors.New("not a UTF-8 string")
|
||||||
|
}
|
||||||
|
return string(b), nil
|
||||||
|
}(m[i].Key)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("key %d: %w", i, err)
|
||||||
|
}
|
||||||
|
v, err := func(item stackitem.Item) ([]util.Uint160, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([]util.Uint160, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = func(item stackitem.Item) (util.Uint160, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, err
|
||||||
|
}
|
||||||
|
u, err := util.Uint160DecodeBytesBE(b)
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, err
|
||||||
|
}
|
||||||
|
return u, nil
|
||||||
|
}(arr[i])
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(m[i].Value)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("value %d: %w", i, err)
|
||||||
|
}
|
||||||
|
res[k] = v
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[i])
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(m[i].Value)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("value %d: %w", i, err)
|
||||||
|
}
|
||||||
|
res[k] = v
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field M: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeStructEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "SomeStruct" name from the provided [result.ApplicationLog].
|
||||||
|
func SomeStructEventsFromApplicationLog(log *result.ApplicationLog) ([]*SomeStructEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*SomeStructEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "SomeStruct" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(SomeStructEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize SomeStructEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to SomeStructEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *SomeStructEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.S, err = itemToUnnamed(arr[index], nil)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field S: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeArrayEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "SomeArray" name from the provided [result.ApplicationLog].
|
||||||
|
func SomeArrayEventsFromApplicationLog(log *result.ApplicationLog) ([]*SomeArrayEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*SomeArrayEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "SomeArray" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(SomeArrayEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize SomeArrayEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to SomeArrayEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *SomeArrayEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.A, err = func(item stackitem.Item) ([][]*big.Int, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([][]*big.Int, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = func(item stackitem.Item) ([]*big.Int, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([]*big.Int, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = arr[i].TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[i])
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field A: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SomeUnexportedFieldEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "SomeUnexportedField" name from the provided [result.ApplicationLog].
|
||||||
|
func SomeUnexportedFieldEventsFromApplicationLog(log *result.ApplicationLog) ([]*SomeUnexportedFieldEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*SomeUnexportedFieldEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "SomeUnexportedField" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(SomeUnexportedFieldEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize SomeUnexportedFieldEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to SomeUnexportedFieldEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *SomeUnexportedFieldEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.S, err = itemToUnnamedX(arr[index], nil)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field S: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
File diff suppressed because it is too large
Load diff
1251
cli/smartcontract/testdata/rpcbindings/structs/rpcbindings_dynamic_hash.out
vendored
Executable file
1251
cli/smartcontract/testdata/rpcbindings/structs/rpcbindings_dynamic_hash.out
vendored
Executable file
File diff suppressed because it is too large
Load diff
40
cli/smartcontract/testdata/rpcbindings/structs/structs.go
vendored
Normal file
40
cli/smartcontract/testdata/rpcbindings/structs/structs.go
vendored
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
package structs
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/interop"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/interop/native/ledger"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/interop/native/management"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Internal struct {
|
||||||
|
Bool bool
|
||||||
|
Int int
|
||||||
|
Bytes []byte
|
||||||
|
String string
|
||||||
|
H160 interop.Hash160
|
||||||
|
H256 interop.Hash256
|
||||||
|
PK interop.PublicKey
|
||||||
|
PubKey interop.PublicKey
|
||||||
|
Sign interop.Signature
|
||||||
|
ArrOfBytes [][]byte
|
||||||
|
ArrOfH160 []interop.Hash160
|
||||||
|
Map map[int][]interop.PublicKey
|
||||||
|
Struct *Internal
|
||||||
|
unexportedField int // this one should be exported in the resulting RPC binding.
|
||||||
|
}
|
||||||
|
|
||||||
|
func Contract(mc management.Contract) management.Contract {
|
||||||
|
return mc
|
||||||
|
}
|
||||||
|
|
||||||
|
func Block(b *ledger.Block) *ledger.Block {
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
|
||||||
|
func Transaction(t *ledger.Transaction) *ledger.Transaction {
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
func Struct(s *Internal) *Internal {
|
||||||
|
return s
|
||||||
|
}
|
|
@ -1,3 +1,3 @@
|
||||||
name: "Types"
|
name: "Types"
|
||||||
sourceurl: https://github.com/nspcc-dev/neo-go/
|
sourceurl: https://github.com/nspcc-dev/neo-go/
|
||||||
safemethods: ["bool", "int", "bytes", "string", "any", "hash160", "hash256", "publicKey", "signature", "bools", "ints", "bytess", "strings", "hash160s", "hash256s", "publicKeys", "signatures", "aAAStrings", "maps", "crazyMaps"]
|
safemethods: ["bool", "int", "bytes", "string", "any", "hash160", "hash256", "publicKey", "signature", "bools", "ints", "bytess", "strings", "hash160s", "hash256s", "publicKeys", "signatures", "aAAStrings", "maps", "crazyMaps", "anyMaps", "unnamedStructs", "unnamedStructsX"]
|
|
@ -1,3 +1,5 @@
|
||||||
|
// Code generated by neo-go contract generate-rpcwrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
// Package types contains RPC wrappers for Types contract.
|
// Package types contains RPC wrappers for Types contract.
|
||||||
package types
|
package types
|
||||||
|
|
||||||
|
@ -16,22 +18,34 @@ import (
|
||||||
// Hash contains contract hash.
|
// Hash contains contract hash.
|
||||||
var Hash = util.Uint160{0x33, 0x22, 0x11, 0x0, 0xff, 0xee, 0xdd, 0xcc, 0xbb, 0xaa, 0x99, 0x88, 0x77, 0x66, 0x55, 0x44, 0x33, 0x22, 0x11, 0x0}
|
var Hash = util.Uint160{0x33, 0x22, 0x11, 0x0, 0xff, 0xee, 0xdd, 0xcc, 0xbb, 0xaa, 0x99, 0x88, 0x77, 0x66, 0x55, 0x44, 0x33, 0x22, 0x11, 0x0}
|
||||||
|
|
||||||
|
// Unnamed is a contract-specific unnamed type used by its methods.
|
||||||
|
type Unnamed struct {
|
||||||
|
I *big.Int
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnnamedX is a contract-specific unnamedX type used by its methods.
|
||||||
|
type UnnamedX struct {
|
||||||
|
I *big.Int
|
||||||
|
B bool
|
||||||
|
}
|
||||||
|
|
||||||
// Invoker is used by ContractReader to call various safe methods.
|
// Invoker is used by ContractReader to call various safe methods.
|
||||||
type Invoker interface {
|
type Invoker interface {
|
||||||
Call(contract util.Uint160, operation string, params ...interface{}) (*result.Invoke, error)
|
Call(contract util.Uint160, operation string, params ...any) (*result.Invoke, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ContractReader implements safe contract methods.
|
// ContractReader implements safe contract methods.
|
||||||
type ContractReader struct {
|
type ContractReader struct {
|
||||||
invoker Invoker
|
invoker Invoker
|
||||||
|
hash util.Uint160
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewReader creates an instance of ContractReader using Hash and the given Invoker.
|
// NewReader creates an instance of ContractReader using Hash and the given Invoker.
|
||||||
func NewReader(invoker Invoker) *ContractReader {
|
func NewReader(invoker Invoker) *ContractReader {
|
||||||
return &ContractReader{invoker}
|
var hash = Hash
|
||||||
|
return &ContractReader{invoker, hash}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// AAAStrings invokes `aAAStrings` method of contract.
|
// AAAStrings invokes `aAAStrings` method of contract.
|
||||||
func (c *ContractReader) AAAStrings(s [][][]string) ([][][]string, error) {
|
func (c *ContractReader) AAAStrings(s [][][]string) ([][][]string, error) {
|
||||||
return func(item stackitem.Item, err error) ([][][]string, error) {
|
return func(item stackitem.Item, err error) ([][][]string, error) {
|
||||||
|
@ -87,37 +101,65 @@ func (c *ContractReader) AAAStrings(s [][][]string) ([][][]string, error) {
|
||||||
}
|
}
|
||||||
return res, nil
|
return res, nil
|
||||||
}(item)
|
}(item)
|
||||||
} (unwrap.Item(c.invoker.Call(Hash, "aAAStrings", s)))
|
}(unwrap.Item(c.invoker.Call(c.hash, "aAAStrings", s)))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Any invokes `any` method of contract.
|
// Any invokes `any` method of contract.
|
||||||
func (c *ContractReader) Any(a interface{}) (interface{}, error) {
|
func (c *ContractReader) Any(a any) (any, error) {
|
||||||
return func (item stackitem.Item, err error) (interface{}, error) {
|
return func(item stackitem.Item, err error) (any, error) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return item.Value(), nil
|
return item.Value(), error(nil)
|
||||||
} (unwrap.Item(c.invoker.Call(Hash, "any", a)))
|
}(unwrap.Item(c.invoker.Call(c.hash, "any", a)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// AnyMaps invokes `anyMaps` method of contract.
|
||||||
|
func (c *ContractReader) AnyMaps(m map[*big.Int]any) (map[*big.Int]any, error) {
|
||||||
|
return func(item stackitem.Item, err error) (map[*big.Int]any, error) {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return func(item stackitem.Item) (map[*big.Int]any, error) {
|
||||||
|
m, ok := item.Value().([]stackitem.MapElement)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("%s is not a map", item.Type().String())
|
||||||
|
}
|
||||||
|
res := make(map[*big.Int]any)
|
||||||
|
for i := range m {
|
||||||
|
k, err := m[i].Key.TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("key %d: %w", i, err)
|
||||||
|
}
|
||||||
|
v, err := m[i].Value.Value(), error(nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("value %d: %w", i, err)
|
||||||
|
}
|
||||||
|
res[k] = v
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(item)
|
||||||
|
}(unwrap.Item(c.invoker.Call(c.hash, "anyMaps", m)))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bool invokes `bool` method of contract.
|
// Bool invokes `bool` method of contract.
|
||||||
func (c *ContractReader) Bool(b bool) (bool, error) {
|
func (c *ContractReader) Bool(b bool) (bool, error) {
|
||||||
return unwrap.Bool(c.invoker.Call(Hash, "bool", b))
|
return unwrap.Bool(c.invoker.Call(c.hash, "bool", b))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bools invokes `bools` method of contract.
|
// Bools invokes `bools` method of contract.
|
||||||
func (c *ContractReader) Bools(b []bool) ([]bool, error) {
|
func (c *ContractReader) Bools(b []bool) ([]bool, error) {
|
||||||
return unwrap.ArrayOfBools(c.invoker.Call(Hash, "bools", b))
|
return unwrap.ArrayOfBools(c.invoker.Call(c.hash, "bools", b))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bytes invokes `bytes` method of contract.
|
// Bytes invokes `bytes` method of contract.
|
||||||
func (c *ContractReader) Bytes(b []byte) ([]byte, error) {
|
func (c *ContractReader) Bytes(b []byte) ([]byte, error) {
|
||||||
return unwrap.Bytes(c.invoker.Call(Hash, "bytes", b))
|
return unwrap.Bytes(c.invoker.Call(c.hash, "bytes", b))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bytess invokes `bytess` method of contract.
|
// Bytess invokes `bytess` method of contract.
|
||||||
func (c *ContractReader) Bytess(b [][]byte) ([][]byte, error) {
|
func (c *ContractReader) Bytess(b [][]byte) ([][]byte, error) {
|
||||||
return unwrap.ArrayOfBytes(c.invoker.Call(Hash, "bytess", b))
|
return unwrap.ArrayOfBytes(c.invoker.Call(c.hash, "bytess", b))
|
||||||
}
|
}
|
||||||
|
|
||||||
// CrazyMaps invokes `crazyMaps` method of contract.
|
// CrazyMaps invokes `crazyMaps` method of contract.
|
||||||
|
@ -208,37 +250,37 @@ func (c *ContractReader) CrazyMaps(m map[*big.Int][]map[string][]util.Uint160) (
|
||||||
}
|
}
|
||||||
return res, nil
|
return res, nil
|
||||||
}(item)
|
}(item)
|
||||||
} (unwrap.Item(c.invoker.Call(Hash, "crazyMaps", m)))
|
}(unwrap.Item(c.invoker.Call(c.hash, "crazyMaps", m)))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Hash160 invokes `hash160` method of contract.
|
// Hash160 invokes `hash160` method of contract.
|
||||||
func (c *ContractReader) Hash160(h util.Uint160) (util.Uint160, error) {
|
func (c *ContractReader) Hash160(h util.Uint160) (util.Uint160, error) {
|
||||||
return unwrap.Uint160(c.invoker.Call(Hash, "hash160", h))
|
return unwrap.Uint160(c.invoker.Call(c.hash, "hash160", h))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Hash160s invokes `hash160s` method of contract.
|
// Hash160s invokes `hash160s` method of contract.
|
||||||
func (c *ContractReader) Hash160s(h []util.Uint160) ([]util.Uint160, error) {
|
func (c *ContractReader) Hash160s(h []util.Uint160) ([]util.Uint160, error) {
|
||||||
return unwrap.ArrayOfUint160(c.invoker.Call(Hash, "hash160s", h))
|
return unwrap.ArrayOfUint160(c.invoker.Call(c.hash, "hash160s", h))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Hash256 invokes `hash256` method of contract.
|
// Hash256 invokes `hash256` method of contract.
|
||||||
func (c *ContractReader) Hash256(h util.Uint256) (util.Uint256, error) {
|
func (c *ContractReader) Hash256(h util.Uint256) (util.Uint256, error) {
|
||||||
return unwrap.Uint256(c.invoker.Call(Hash, "hash256", h))
|
return unwrap.Uint256(c.invoker.Call(c.hash, "hash256", h))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Hash256s invokes `hash256s` method of contract.
|
// Hash256s invokes `hash256s` method of contract.
|
||||||
func (c *ContractReader) Hash256s(h []util.Uint256) ([]util.Uint256, error) {
|
func (c *ContractReader) Hash256s(h []util.Uint256) ([]util.Uint256, error) {
|
||||||
return unwrap.ArrayOfUint256(c.invoker.Call(Hash, "hash256s", h))
|
return unwrap.ArrayOfUint256(c.invoker.Call(c.hash, "hash256s", h))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Int invokes `int` method of contract.
|
// Int invokes `int` method of contract.
|
||||||
func (c *ContractReader) Int(i *big.Int) (*big.Int, error) {
|
func (c *ContractReader) Int(i *big.Int) (*big.Int, error) {
|
||||||
return unwrap.BigInt(c.invoker.Call(Hash, "int", i))
|
return unwrap.BigInt(c.invoker.Call(c.hash, "int", i))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ints invokes `ints` method of contract.
|
// Ints invokes `ints` method of contract.
|
||||||
func (c *ContractReader) Ints(i []*big.Int) ([]*big.Int, error) {
|
func (c *ContractReader) Ints(i []*big.Int) ([]*big.Int, error) {
|
||||||
return unwrap.ArrayOfBigInts(c.invoker.Call(Hash, "ints", i))
|
return unwrap.ArrayOfBigInts(c.invoker.Call(c.hash, "ints", i))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Maps invokes `maps` method of contract.
|
// Maps invokes `maps` method of contract.
|
||||||
|
@ -284,35 +326,119 @@ func (c *ContractReader) Maps(m map[string]string) (map[string]string, error) {
|
||||||
}
|
}
|
||||||
return res, nil
|
return res, nil
|
||||||
}(item)
|
}(item)
|
||||||
} (unwrap.Item(c.invoker.Call(Hash, "maps", m)))
|
}(unwrap.Item(c.invoker.Call(c.hash, "maps", m)))
|
||||||
}
|
}
|
||||||
|
|
||||||
// PublicKey invokes `publicKey` method of contract.
|
// PublicKey invokes `publicKey` method of contract.
|
||||||
func (c *ContractReader) PublicKey(k *keys.PublicKey) (*keys.PublicKey, error) {
|
func (c *ContractReader) PublicKey(k *keys.PublicKey) (*keys.PublicKey, error) {
|
||||||
return unwrap.PublicKey(c.invoker.Call(Hash, "publicKey", k))
|
return unwrap.PublicKey(c.invoker.Call(c.hash, "publicKey", k))
|
||||||
}
|
}
|
||||||
|
|
||||||
// PublicKeys invokes `publicKeys` method of contract.
|
// PublicKeys invokes `publicKeys` method of contract.
|
||||||
func (c *ContractReader) PublicKeys(k keys.PublicKeys) (keys.PublicKeys, error) {
|
func (c *ContractReader) PublicKeys(k keys.PublicKeys) (keys.PublicKeys, error) {
|
||||||
return unwrap.ArrayOfPublicKeys(c.invoker.Call(Hash, "publicKeys", k))
|
return unwrap.ArrayOfPublicKeys(c.invoker.Call(c.hash, "publicKeys", k))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Signature invokes `signature` method of contract.
|
// Signature invokes `signature` method of contract.
|
||||||
func (c *ContractReader) Signature(s []byte) ([]byte, error) {
|
func (c *ContractReader) Signature(s []byte) ([]byte, error) {
|
||||||
return unwrap.Bytes(c.invoker.Call(Hash, "signature", s))
|
return unwrap.Bytes(c.invoker.Call(c.hash, "signature", s))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Signatures invokes `signatures` method of contract.
|
// Signatures invokes `signatures` method of contract.
|
||||||
func (c *ContractReader) Signatures(s [][]byte) ([][]byte, error) {
|
func (c *ContractReader) Signatures(s [][]byte) ([][]byte, error) {
|
||||||
return unwrap.ArrayOfBytes(c.invoker.Call(Hash, "signatures", s))
|
return unwrap.ArrayOfBytes(c.invoker.Call(c.hash, "signatures", s))
|
||||||
}
|
}
|
||||||
|
|
||||||
// String invokes `string` method of contract.
|
// String invokes `string` method of contract.
|
||||||
func (c *ContractReader) String(s string) (string, error) {
|
func (c *ContractReader) String(s string) (string, error) {
|
||||||
return unwrap.UTF8String(c.invoker.Call(Hash, "string", s))
|
return unwrap.UTF8String(c.invoker.Call(c.hash, "string", s))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Strings invokes `strings` method of contract.
|
// Strings invokes `strings` method of contract.
|
||||||
func (c *ContractReader) Strings(s []string) ([]string, error) {
|
func (c *ContractReader) Strings(s []string) ([]string, error) {
|
||||||
return unwrap.ArrayOfUTF8Strings(c.invoker.Call(Hash, "strings", s))
|
return unwrap.ArrayOfUTF8Strings(c.invoker.Call(c.hash, "strings", s))
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnnamedStructs invokes `unnamedStructs` method of contract.
|
||||||
|
func (c *ContractReader) UnnamedStructs() (*Unnamed, error) {
|
||||||
|
return itemToUnnamed(unwrap.Item(c.invoker.Call(c.hash, "unnamedStructs")))
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnnamedStructsX invokes `unnamedStructsX` method of contract.
|
||||||
|
func (c *ContractReader) UnnamedStructsX() (*UnnamedX, error) {
|
||||||
|
return itemToUnnamedX(unwrap.Item(c.invoker.Call(c.hash, "unnamedStructsX")))
|
||||||
|
}
|
||||||
|
|
||||||
|
// itemToUnnamed converts stack item into *Unnamed.
|
||||||
|
func itemToUnnamed(item stackitem.Item, err error) (*Unnamed, error) {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var res = new(Unnamed)
|
||||||
|
err = res.FromStackItem(item)
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem retrieves fields of Unnamed from the given
|
||||||
|
// [stackitem.Item] or returns an error if it's not possible to do to so.
|
||||||
|
func (res *Unnamed) FromStackItem(item stackitem.Item) error {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
res.I, err = arr[index].TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field I: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// itemToUnnamedX converts stack item into *UnnamedX.
|
||||||
|
func itemToUnnamedX(item stackitem.Item, err error) (*UnnamedX, error) {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var res = new(UnnamedX)
|
||||||
|
err = res.FromStackItem(item)
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem retrieves fields of UnnamedX from the given
|
||||||
|
// [stackitem.Item] or returns an error if it's not possible to do to so.
|
||||||
|
func (res *UnnamedX) FromStackItem(item stackitem.Item) error {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 2 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
res.I, err = arr[index].TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field I: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
index++
|
||||||
|
res.B, err = arr[index].TryBool()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field B: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
440
cli/smartcontract/testdata/rpcbindings/types/rpcbindings_dynamic_hash.out
vendored
Executable file
440
cli/smartcontract/testdata/rpcbindings/types/rpcbindings_dynamic_hash.out
vendored
Executable file
|
@ -0,0 +1,440 @@
|
||||||
|
// Code generated by neo-go contract generate-rpcwrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
|
// Package types contains RPC wrappers for Types contract.
|
||||||
|
package types
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/neorpc/result"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/unwrap"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
||||||
|
"math/big"
|
||||||
|
"unicode/utf8"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Unnamed is a contract-specific unnamed type used by its methods.
|
||||||
|
type Unnamed struct {
|
||||||
|
I *big.Int
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnnamedX is a contract-specific unnamedX type used by its methods.
|
||||||
|
type UnnamedX struct {
|
||||||
|
I *big.Int
|
||||||
|
B bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// Invoker is used by ContractReader to call various safe methods.
|
||||||
|
type Invoker interface {
|
||||||
|
Call(contract util.Uint160, operation string, params ...any) (*result.Invoke, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ContractReader implements safe contract methods.
|
||||||
|
type ContractReader struct {
|
||||||
|
invoker Invoker
|
||||||
|
hash util.Uint160
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewReader creates an instance of ContractReader using provided contract hash and the given Invoker.
|
||||||
|
func NewReader(invoker Invoker, hash util.Uint160) *ContractReader {
|
||||||
|
return &ContractReader{invoker, hash}
|
||||||
|
}
|
||||||
|
|
||||||
|
// AAAStrings invokes `aAAStrings` method of contract.
|
||||||
|
func (c *ContractReader) AAAStrings(s [][][]string) ([][][]string, error) {
|
||||||
|
return func(item stackitem.Item, err error) ([][][]string, error) {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return func(item stackitem.Item) ([][][]string, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([][][]string, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = func(item stackitem.Item) ([][]string, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([][]string, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = func(item stackitem.Item) ([]string, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([]string, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = func(item stackitem.Item) (string, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if !utf8.Valid(b) {
|
||||||
|
return "", errors.New("not a UTF-8 string")
|
||||||
|
}
|
||||||
|
return string(b), nil
|
||||||
|
}(arr[i])
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[i])
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[i])
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(item)
|
||||||
|
}(unwrap.Item(c.invoker.Call(c.hash, "aAAStrings", s)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Any invokes `any` method of contract.
|
||||||
|
func (c *ContractReader) Any(a any) (any, error) {
|
||||||
|
return func(item stackitem.Item, err error) (any, error) {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return item.Value(), error(nil)
|
||||||
|
}(unwrap.Item(c.invoker.Call(c.hash, "any", a)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// AnyMaps invokes `anyMaps` method of contract.
|
||||||
|
func (c *ContractReader) AnyMaps(m map[*big.Int]any) (map[*big.Int]any, error) {
|
||||||
|
return func(item stackitem.Item, err error) (map[*big.Int]any, error) {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return func(item stackitem.Item) (map[*big.Int]any, error) {
|
||||||
|
m, ok := item.Value().([]stackitem.MapElement)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("%s is not a map", item.Type().String())
|
||||||
|
}
|
||||||
|
res := make(map[*big.Int]any)
|
||||||
|
for i := range m {
|
||||||
|
k, err := m[i].Key.TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("key %d: %w", i, err)
|
||||||
|
}
|
||||||
|
v, err := m[i].Value.Value(), error(nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("value %d: %w", i, err)
|
||||||
|
}
|
||||||
|
res[k] = v
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(item)
|
||||||
|
}(unwrap.Item(c.invoker.Call(c.hash, "anyMaps", m)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Bool invokes `bool` method of contract.
|
||||||
|
func (c *ContractReader) Bool(b bool) (bool, error) {
|
||||||
|
return unwrap.Bool(c.invoker.Call(c.hash, "bool", b))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Bools invokes `bools` method of contract.
|
||||||
|
func (c *ContractReader) Bools(b []bool) ([]bool, error) {
|
||||||
|
return unwrap.ArrayOfBools(c.invoker.Call(c.hash, "bools", b))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Bytes invokes `bytes` method of contract.
|
||||||
|
func (c *ContractReader) Bytes(b []byte) ([]byte, error) {
|
||||||
|
return unwrap.Bytes(c.invoker.Call(c.hash, "bytes", b))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Bytess invokes `bytess` method of contract.
|
||||||
|
func (c *ContractReader) Bytess(b [][]byte) ([][]byte, error) {
|
||||||
|
return unwrap.ArrayOfBytes(c.invoker.Call(c.hash, "bytess", b))
|
||||||
|
}
|
||||||
|
|
||||||
|
// CrazyMaps invokes `crazyMaps` method of contract.
|
||||||
|
func (c *ContractReader) CrazyMaps(m map[*big.Int][]map[string][]util.Uint160) (map[*big.Int][]map[string][]util.Uint160, error) {
|
||||||
|
return func(item stackitem.Item, err error) (map[*big.Int][]map[string][]util.Uint160, error) {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return func(item stackitem.Item) (map[*big.Int][]map[string][]util.Uint160, error) {
|
||||||
|
m, ok := item.Value().([]stackitem.MapElement)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("%s is not a map", item.Type().String())
|
||||||
|
}
|
||||||
|
res := make(map[*big.Int][]map[string][]util.Uint160)
|
||||||
|
for i := range m {
|
||||||
|
k, err := m[i].Key.TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("key %d: %w", i, err)
|
||||||
|
}
|
||||||
|
v, err := func(item stackitem.Item) ([]map[string][]util.Uint160, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([]map[string][]util.Uint160, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = func(item stackitem.Item) (map[string][]util.Uint160, error) {
|
||||||
|
m, ok := item.Value().([]stackitem.MapElement)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("%s is not a map", item.Type().String())
|
||||||
|
}
|
||||||
|
res := make(map[string][]util.Uint160)
|
||||||
|
for i := range m {
|
||||||
|
k, err := func(item stackitem.Item) (string, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if !utf8.Valid(b) {
|
||||||
|
return "", errors.New("not a UTF-8 string")
|
||||||
|
}
|
||||||
|
return string(b), nil
|
||||||
|
}(m[i].Key)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("key %d: %w", i, err)
|
||||||
|
}
|
||||||
|
v, err := func(item stackitem.Item) ([]util.Uint160, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([]util.Uint160, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = func(item stackitem.Item) (util.Uint160, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, err
|
||||||
|
}
|
||||||
|
u, err := util.Uint160DecodeBytesBE(b)
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, err
|
||||||
|
}
|
||||||
|
return u, nil
|
||||||
|
}(arr[i])
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(m[i].Value)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("value %d: %w", i, err)
|
||||||
|
}
|
||||||
|
res[k] = v
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[i])
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(m[i].Value)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("value %d: %w", i, err)
|
||||||
|
}
|
||||||
|
res[k] = v
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(item)
|
||||||
|
}(unwrap.Item(c.invoker.Call(c.hash, "crazyMaps", m)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Hash160 invokes `hash160` method of contract.
|
||||||
|
func (c *ContractReader) Hash160(h util.Uint160) (util.Uint160, error) {
|
||||||
|
return unwrap.Uint160(c.invoker.Call(c.hash, "hash160", h))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Hash160s invokes `hash160s` method of contract.
|
||||||
|
func (c *ContractReader) Hash160s(h []util.Uint160) ([]util.Uint160, error) {
|
||||||
|
return unwrap.ArrayOfUint160(c.invoker.Call(c.hash, "hash160s", h))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Hash256 invokes `hash256` method of contract.
|
||||||
|
func (c *ContractReader) Hash256(h util.Uint256) (util.Uint256, error) {
|
||||||
|
return unwrap.Uint256(c.invoker.Call(c.hash, "hash256", h))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Hash256s invokes `hash256s` method of contract.
|
||||||
|
func (c *ContractReader) Hash256s(h []util.Uint256) ([]util.Uint256, error) {
|
||||||
|
return unwrap.ArrayOfUint256(c.invoker.Call(c.hash, "hash256s", h))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Int invokes `int` method of contract.
|
||||||
|
func (c *ContractReader) Int(i *big.Int) (*big.Int, error) {
|
||||||
|
return unwrap.BigInt(c.invoker.Call(c.hash, "int", i))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ints invokes `ints` method of contract.
|
||||||
|
func (c *ContractReader) Ints(i []*big.Int) ([]*big.Int, error) {
|
||||||
|
return unwrap.ArrayOfBigInts(c.invoker.Call(c.hash, "ints", i))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Maps invokes `maps` method of contract.
|
||||||
|
func (c *ContractReader) Maps(m map[string]string) (map[string]string, error) {
|
||||||
|
return func(item stackitem.Item, err error) (map[string]string, error) {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return func(item stackitem.Item) (map[string]string, error) {
|
||||||
|
m, ok := item.Value().([]stackitem.MapElement)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("%s is not a map", item.Type().String())
|
||||||
|
}
|
||||||
|
res := make(map[string]string)
|
||||||
|
for i := range m {
|
||||||
|
k, err := func(item stackitem.Item) (string, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if !utf8.Valid(b) {
|
||||||
|
return "", errors.New("not a UTF-8 string")
|
||||||
|
}
|
||||||
|
return string(b), nil
|
||||||
|
}(m[i].Key)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("key %d: %w", i, err)
|
||||||
|
}
|
||||||
|
v, err := func(item stackitem.Item) (string, error) {
|
||||||
|
b, err := item.TryBytes()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if !utf8.Valid(b) {
|
||||||
|
return "", errors.New("not a UTF-8 string")
|
||||||
|
}
|
||||||
|
return string(b), nil
|
||||||
|
}(m[i].Value)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("value %d: %w", i, err)
|
||||||
|
}
|
||||||
|
res[k] = v
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(item)
|
||||||
|
}(unwrap.Item(c.invoker.Call(c.hash, "maps", m)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// PublicKey invokes `publicKey` method of contract.
|
||||||
|
func (c *ContractReader) PublicKey(k *keys.PublicKey) (*keys.PublicKey, error) {
|
||||||
|
return unwrap.PublicKey(c.invoker.Call(c.hash, "publicKey", k))
|
||||||
|
}
|
||||||
|
|
||||||
|
// PublicKeys invokes `publicKeys` method of contract.
|
||||||
|
func (c *ContractReader) PublicKeys(k keys.PublicKeys) (keys.PublicKeys, error) {
|
||||||
|
return unwrap.ArrayOfPublicKeys(c.invoker.Call(c.hash, "publicKeys", k))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Signature invokes `signature` method of contract.
|
||||||
|
func (c *ContractReader) Signature(s []byte) ([]byte, error) {
|
||||||
|
return unwrap.Bytes(c.invoker.Call(c.hash, "signature", s))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Signatures invokes `signatures` method of contract.
|
||||||
|
func (c *ContractReader) Signatures(s [][]byte) ([][]byte, error) {
|
||||||
|
return unwrap.ArrayOfBytes(c.invoker.Call(c.hash, "signatures", s))
|
||||||
|
}
|
||||||
|
|
||||||
|
// String invokes `string` method of contract.
|
||||||
|
func (c *ContractReader) String(s string) (string, error) {
|
||||||
|
return unwrap.UTF8String(c.invoker.Call(c.hash, "string", s))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Strings invokes `strings` method of contract.
|
||||||
|
func (c *ContractReader) Strings(s []string) ([]string, error) {
|
||||||
|
return unwrap.ArrayOfUTF8Strings(c.invoker.Call(c.hash, "strings", s))
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnnamedStructs invokes `unnamedStructs` method of contract.
|
||||||
|
func (c *ContractReader) UnnamedStructs() (*Unnamed, error) {
|
||||||
|
return itemToUnnamed(unwrap.Item(c.invoker.Call(c.hash, "unnamedStructs")))
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnnamedStructsX invokes `unnamedStructsX` method of contract.
|
||||||
|
func (c *ContractReader) UnnamedStructsX() (*UnnamedX, error) {
|
||||||
|
return itemToUnnamedX(unwrap.Item(c.invoker.Call(c.hash, "unnamedStructsX")))
|
||||||
|
}
|
||||||
|
|
||||||
|
// itemToUnnamed converts stack item into *Unnamed.
|
||||||
|
func itemToUnnamed(item stackitem.Item, err error) (*Unnamed, error) {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var res = new(Unnamed)
|
||||||
|
err = res.FromStackItem(item)
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem retrieves fields of Unnamed from the given
|
||||||
|
// [stackitem.Item] or returns an error if it's not possible to do to so.
|
||||||
|
func (res *Unnamed) FromStackItem(item stackitem.Item) error {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
res.I, err = arr[index].TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field I: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// itemToUnnamedX converts stack item into *UnnamedX.
|
||||||
|
func itemToUnnamedX(item stackitem.Item, err error) (*UnnamedX, error) {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var res = new(UnnamedX)
|
||||||
|
err = res.FromStackItem(item)
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem retrieves fields of UnnamedX from the given
|
||||||
|
// [stackitem.Item] or returns an error if it's not possible to do to so.
|
||||||
|
func (res *UnnamedX) FromStackItem(item stackitem.Item) error {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 2 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
res.I, err = arr[index].TryInteger()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field I: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
index++
|
||||||
|
res.B, err = arr[index].TryBool()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field B: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -20,7 +20,7 @@ func String(s string) string {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
func Any(a interface{}) interface{} {
|
func Any(a any) any {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -83,3 +83,21 @@ func Maps(m map[string]string) map[string]string {
|
||||||
func CrazyMaps(m map[int][]map[string][]interop.Hash160) map[int][]map[string][]interop.Hash160 {
|
func CrazyMaps(m map[int][]map[string][]interop.Hash160) map[int][]map[string][]interop.Hash160 {
|
||||||
return m
|
return m
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func AnyMaps(m map[int]any) map[int]any {
|
||||||
|
return m
|
||||||
|
}
|
||||||
|
|
||||||
|
func UnnamedStructs() struct{ I int } {
|
||||||
|
return struct{ I int }{I: 123}
|
||||||
|
}
|
||||||
|
|
||||||
|
func UnnamedStructsX() struct {
|
||||||
|
I int
|
||||||
|
B bool
|
||||||
|
} {
|
||||||
|
return struct {
|
||||||
|
I int
|
||||||
|
B bool
|
||||||
|
}{I: 123, B: true}
|
||||||
|
}
|
39
cli/smartcontract/testdata/structs/structs.go
vendored
39
cli/smartcontract/testdata/structs/structs.go
vendored
|
@ -1,39 +0,0 @@
|
||||||
package structs
|
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/interop"
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/interop/native/ledger"
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/interop/native/management"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Internal struct {
|
|
||||||
Bool bool
|
|
||||||
Int int
|
|
||||||
Bytes []byte
|
|
||||||
String string
|
|
||||||
H160 interop.Hash160
|
|
||||||
H256 interop.Hash256
|
|
||||||
PK interop.PublicKey
|
|
||||||
PubKey interop.PublicKey
|
|
||||||
Sign interop.Signature
|
|
||||||
ArrOfBytes [][]byte
|
|
||||||
ArrOfH160 []interop.Hash160
|
|
||||||
Map map[int][]interop.PublicKey
|
|
||||||
Struct *Internal
|
|
||||||
}
|
|
||||||
|
|
||||||
func Contract(mc management.Contract) management.Contract {
|
|
||||||
return mc
|
|
||||||
}
|
|
||||||
|
|
||||||
func Block(b *ledger.Block) *ledger.Block {
|
|
||||||
return b
|
|
||||||
}
|
|
||||||
|
|
||||||
func Transaction(t *ledger.Transaction) *ledger.Transaction {
|
|
||||||
return t
|
|
||||||
}
|
|
||||||
|
|
||||||
func Struct(s *Internal) *Internal {
|
|
||||||
return s
|
|
||||||
}
|
|
297
cli/smartcontract/testdata/verify.bindings.yml
vendored
Executable file
297
cli/smartcontract/testdata/verify.bindings.yml
vendored
Executable file
|
@ -0,0 +1,297 @@
|
||||||
|
package: testdata
|
||||||
|
hash: "0x0000000000000000000000000000000000000000"
|
||||||
|
overrides:
|
||||||
|
burnGas.gas: int
|
||||||
|
call: any
|
||||||
|
call.args: '[]any'
|
||||||
|
call.f: any
|
||||||
|
call.method: string
|
||||||
|
call.scriptHash: github.com/nspcc-dev/neo-go/pkg/interop.Hash160
|
||||||
|
callWithToken: any
|
||||||
|
callWithToken.args: '[]any'
|
||||||
|
callWithToken.flags: int
|
||||||
|
callWithToken.method: string
|
||||||
|
callWithToken.scriptHash: string
|
||||||
|
callWithTokenNoRet.args: '[]any'
|
||||||
|
callWithTokenNoRet.flags: int
|
||||||
|
callWithTokenNoRet.method: string
|
||||||
|
callWithTokenNoRet.scriptHash: string
|
||||||
|
checkWitness: bool
|
||||||
|
checkWitness.hashOrKey: '[]byte'
|
||||||
|
createMultisigAccount: '[]byte'
|
||||||
|
createMultisigAccount.m: int
|
||||||
|
createMultisigAccount.pubs: '[]github.com/nspcc-dev/neo-go/pkg/interop.PublicKey'
|
||||||
|
createStandardAccount: '[]byte'
|
||||||
|
createStandardAccount.pub: github.com/nspcc-dev/neo-go/pkg/interop.PublicKey
|
||||||
|
currentHash: github.com/nspcc-dev/neo-go/pkg/interop.Hash256
|
||||||
|
currentIndex: int
|
||||||
|
currentSigners: '[]github.com/nspcc-dev/neo-go/pkg/interop/native/ledger.TransactionSigner'
|
||||||
|
equals: bool
|
||||||
|
equals.b: any
|
||||||
|
gasLeft: int
|
||||||
|
getAddressVersion: int
|
||||||
|
getBlock: '*github.com/nspcc-dev/neo-go/pkg/interop/native/ledger.Block'
|
||||||
|
getBlock.indexOrHash: any
|
||||||
|
getCallFlags: any
|
||||||
|
getCallingScriptHash: github.com/nspcc-dev/neo-go/pkg/interop.Hash160
|
||||||
|
getEntryScriptHash: github.com/nspcc-dev/neo-go/pkg/interop.Hash160
|
||||||
|
getExecutingScriptHash: github.com/nspcc-dev/neo-go/pkg/interop.Hash160
|
||||||
|
getInvocationCounter: int
|
||||||
|
getNetwork: int
|
||||||
|
getNotifications: '[][]any'
|
||||||
|
getNotifications.h: github.com/nspcc-dev/neo-go/pkg/interop.Hash160
|
||||||
|
getRandom: int
|
||||||
|
getScriptContainer: '*github.com/nspcc-dev/neo-go/pkg/interop/native/ledger.Transaction'
|
||||||
|
getTime: int
|
||||||
|
getTransaction: '*github.com/nspcc-dev/neo-go/pkg/interop/native/ledger.Transaction'
|
||||||
|
getTransaction.hash: github.com/nspcc-dev/neo-go/pkg/interop.Hash256
|
||||||
|
getTransactionFromBlock: '*github.com/nspcc-dev/neo-go/pkg/interop/native/ledger.Transaction'
|
||||||
|
getTransactionFromBlock.indexOrHash: any
|
||||||
|
getTransactionFromBlock.txIndex: int
|
||||||
|
getTransactionHeight: int
|
||||||
|
getTransactionHeight.hash: github.com/nspcc-dev/neo-go/pkg/interop.Hash256
|
||||||
|
getTransactionSigners: '[]github.com/nspcc-dev/neo-go/pkg/interop/native/ledger.TransactionSigner'
|
||||||
|
getTransactionSigners.hash: github.com/nspcc-dev/neo-go/pkg/interop.Hash256
|
||||||
|
getTransactionVMState: int
|
||||||
|
getTransactionVMState.hash: github.com/nspcc-dev/neo-go/pkg/interop.Hash256
|
||||||
|
getTrigger: int
|
||||||
|
loadScript: any
|
||||||
|
loadScript.args: '[]any'
|
||||||
|
loadScript.f: any
|
||||||
|
loadScript.script: '[]byte'
|
||||||
|
log.message: string
|
||||||
|
notify.args: '[]any'
|
||||||
|
notify.name: string
|
||||||
|
onNEP11Payment.amount: int
|
||||||
|
onNEP11Payment.data: any
|
||||||
|
onNEP11Payment.from: github.com/nspcc-dev/neo-go/pkg/interop.Hash160
|
||||||
|
onNEP11Payment.token: '[]byte'
|
||||||
|
onNEP17Payment.amount: int
|
||||||
|
onNEP17Payment.data: any
|
||||||
|
onNEP17Payment.from: github.com/nspcc-dev/neo-go/pkg/interop.Hash160
|
||||||
|
opcode0NoReturn.op: string
|
||||||
|
opcode1: any
|
||||||
|
opcode1NoReturn.arg: any
|
||||||
|
opcode1NoReturn.op: string
|
||||||
|
opcode1.arg: any
|
||||||
|
opcode1.op: string
|
||||||
|
opcode2: any
|
||||||
|
opcode2NoReturn.arg1: any
|
||||||
|
opcode2NoReturn.arg2: any
|
||||||
|
opcode2NoReturn.op: string
|
||||||
|
opcode2.arg1: any
|
||||||
|
opcode2.arg2: any
|
||||||
|
opcode2.op: string
|
||||||
|
opcode3: any
|
||||||
|
opcode3.arg1: any
|
||||||
|
opcode3.arg2: any
|
||||||
|
opcode3.arg3: any
|
||||||
|
opcode3.op: string
|
||||||
|
platform: '[]byte'
|
||||||
|
syscall0: any
|
||||||
|
syscall0NoReturn.name: string
|
||||||
|
syscall0.name: string
|
||||||
|
syscall1: any
|
||||||
|
syscall1NoReturn.arg: any
|
||||||
|
syscall1NoReturn.name: string
|
||||||
|
syscall1.arg: any
|
||||||
|
syscall1.name: string
|
||||||
|
syscall2: any
|
||||||
|
syscall2NoReturn.arg1: any
|
||||||
|
syscall2NoReturn.arg2: any
|
||||||
|
syscall2NoReturn.name: string
|
||||||
|
syscall2.arg1: any
|
||||||
|
syscall2.arg2: any
|
||||||
|
syscall2.name: string
|
||||||
|
syscall3: any
|
||||||
|
syscall3NoReturn.arg1: any
|
||||||
|
syscall3NoReturn.arg2: any
|
||||||
|
syscall3NoReturn.arg3: any
|
||||||
|
syscall3NoReturn.name: string
|
||||||
|
syscall3.arg1: any
|
||||||
|
syscall3.arg2: any
|
||||||
|
syscall3.arg3: any
|
||||||
|
syscall3.name: string
|
||||||
|
syscall4: any
|
||||||
|
syscall4NoReturn.arg1: any
|
||||||
|
syscall4NoReturn.arg2: any
|
||||||
|
syscall4NoReturn.arg3: any
|
||||||
|
syscall4NoReturn.arg4: any
|
||||||
|
syscall4NoReturn.name: string
|
||||||
|
syscall4.arg1: any
|
||||||
|
syscall4.arg2: any
|
||||||
|
syscall4.arg3: any
|
||||||
|
syscall4.arg4: any
|
||||||
|
syscall4.name: string
|
||||||
|
toBlockSR: '*github.com/nspcc-dev/neo-go/pkg/interop/native/ledger.BlockSR'
|
||||||
|
verify: bool
|
||||||
|
namedtypes:
|
||||||
|
ledger.Block:
|
||||||
|
base: Array
|
||||||
|
name: ledger.Block
|
||||||
|
fields:
|
||||||
|
- field: Hash
|
||||||
|
base: Hash256
|
||||||
|
- field: Version
|
||||||
|
base: Integer
|
||||||
|
- field: PrevHash
|
||||||
|
base: Hash256
|
||||||
|
- field: MerkleRoot
|
||||||
|
base: Hash256
|
||||||
|
- field: Timestamp
|
||||||
|
base: Integer
|
||||||
|
- field: Nonce
|
||||||
|
base: Integer
|
||||||
|
- field: Index
|
||||||
|
base: Integer
|
||||||
|
- field: NextConsensus
|
||||||
|
base: Hash160
|
||||||
|
- field: TransactionsLength
|
||||||
|
base: Integer
|
||||||
|
ledger.BlockSR:
|
||||||
|
base: Array
|
||||||
|
name: ledger.BlockSR
|
||||||
|
fields:
|
||||||
|
- field: Hash
|
||||||
|
base: Hash256
|
||||||
|
- field: Version
|
||||||
|
base: Integer
|
||||||
|
- field: PrevHash
|
||||||
|
base: Hash256
|
||||||
|
- field: MerkleRoot
|
||||||
|
base: Hash256
|
||||||
|
- field: Timestamp
|
||||||
|
base: Integer
|
||||||
|
- field: Nonce
|
||||||
|
base: Integer
|
||||||
|
- field: Index
|
||||||
|
base: Integer
|
||||||
|
- field: NextConsensus
|
||||||
|
base: Hash160
|
||||||
|
- field: TransactionsLength
|
||||||
|
base: Integer
|
||||||
|
- field: PrevStateRoot
|
||||||
|
base: Hash256
|
||||||
|
ledger.Transaction:
|
||||||
|
base: Array
|
||||||
|
name: ledger.Transaction
|
||||||
|
fields:
|
||||||
|
- field: Hash
|
||||||
|
base: Hash256
|
||||||
|
- field: Version
|
||||||
|
base: Integer
|
||||||
|
- field: Nonce
|
||||||
|
base: Integer
|
||||||
|
- field: Sender
|
||||||
|
base: Hash160
|
||||||
|
- field: SysFee
|
||||||
|
base: Integer
|
||||||
|
- field: NetFee
|
||||||
|
base: Integer
|
||||||
|
- field: ValidUntilBlock
|
||||||
|
base: Integer
|
||||||
|
- field: Script
|
||||||
|
base: ByteArray
|
||||||
|
ledger.TransactionSigner:
|
||||||
|
base: Array
|
||||||
|
name: ledger.TransactionSigner
|
||||||
|
fields:
|
||||||
|
- field: Account
|
||||||
|
base: Hash160
|
||||||
|
- field: Scopes
|
||||||
|
base: Integer
|
||||||
|
- field: AllowedContracts
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: Hash160
|
||||||
|
- field: AllowedGroups
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: PublicKey
|
||||||
|
- field: Rules
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: Array
|
||||||
|
name: ledger.WitnessRule
|
||||||
|
ledger.WitnessCondition:
|
||||||
|
base: Array
|
||||||
|
name: ledger.WitnessCondition
|
||||||
|
fields:
|
||||||
|
- field: Type
|
||||||
|
base: Integer
|
||||||
|
- field: Value
|
||||||
|
base: Any
|
||||||
|
ledger.WitnessRule:
|
||||||
|
base: Array
|
||||||
|
name: ledger.WitnessRule
|
||||||
|
fields:
|
||||||
|
- field: Action
|
||||||
|
base: Integer
|
||||||
|
- field: Condition
|
||||||
|
base: Array
|
||||||
|
name: ledger.WitnessCondition
|
||||||
|
types:
|
||||||
|
call.args:
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: Any
|
||||||
|
call.f:
|
||||||
|
base: InteropInterface
|
||||||
|
interface: iterator
|
||||||
|
callWithToken.args:
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: Any
|
||||||
|
callWithTokenNoRet.args:
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: Any
|
||||||
|
createMultisigAccount.pubs:
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: PublicKey
|
||||||
|
currentSigners:
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: Array
|
||||||
|
name: ledger.TransactionSigner
|
||||||
|
getBlock:
|
||||||
|
base: Array
|
||||||
|
name: ledger.Block
|
||||||
|
getCallFlags:
|
||||||
|
base: InteropInterface
|
||||||
|
interface: iterator
|
||||||
|
getNotifications:
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: Any
|
||||||
|
getScriptContainer:
|
||||||
|
base: Array
|
||||||
|
name: ledger.Transaction
|
||||||
|
getTransaction:
|
||||||
|
base: Array
|
||||||
|
name: ledger.Transaction
|
||||||
|
getTransactionFromBlock:
|
||||||
|
base: Array
|
||||||
|
name: ledger.Transaction
|
||||||
|
getTransactionSigners:
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: Array
|
||||||
|
name: ledger.TransactionSigner
|
||||||
|
loadScript.args:
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: Any
|
||||||
|
loadScript.f:
|
||||||
|
base: InteropInterface
|
||||||
|
interface: iterator
|
||||||
|
notify.args:
|
||||||
|
base: Array
|
||||||
|
value:
|
||||||
|
base: Any
|
||||||
|
toBlockSR:
|
||||||
|
base: Array
|
||||||
|
name: ledger.BlockSR
|
4
cli/smartcontract/testdata/verify.go
vendored
4
cli/smartcontract/testdata/verify.go
vendored
|
@ -9,12 +9,12 @@ func Verify() bool {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
func OnNEP17Payment(from interop.Hash160, amount int, data interface{}) {
|
func OnNEP17Payment(from interop.Hash160, amount int, data any) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// OnNEP11Payment notifies about NEP-11 payment. You don't call this method directly,
|
// OnNEP11Payment notifies about NEP-11 payment. You don't call this method directly,
|
||||||
// instead it's called by NEP-11 contract when you transfer funds from your address
|
// instead it's called by NEP-11 contract when you transfer funds from your address
|
||||||
// to the address of this NFT contract.
|
// to the address of this NFT contract.
|
||||||
func OnNEP11Payment(from interop.Hash160, amount int, token []byte, data interface{}) {
|
func OnNEP11Payment(from interop.Hash160, amount int, token []byte, data any) {
|
||||||
runtime.Notify("OnNEP11Payment", from, amount, token, data)
|
runtime.Notify("OnNEP11Payment", from, amount, token, data)
|
||||||
}
|
}
|
||||||
|
|
97
cli/smartcontract/testdata/verifyrpc/verify.go
vendored
97
cli/smartcontract/testdata/verifyrpc/verify.go
vendored
|
@ -1,45 +1,57 @@
|
||||||
|
// Code generated by neo-go contract generate-rpcwrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
||||||
|
|
||||||
// Package verify contains RPC wrappers for verify contract.
|
// Package verify contains RPC wrappers for verify contract.
|
||||||
package verify
|
package verify
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/neorpc/result"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/util"
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Hash contains contract hash.
|
// Hash contains contract hash.
|
||||||
var Hash = util.Uint160{0x33, 0x22, 0x11, 0x0, 0xff, 0xee, 0xdd, 0xcc, 0xbb, 0xaa, 0x99, 0x88, 0x77, 0x66, 0x55, 0x44, 0x33, 0x22, 0x11, 0x0}
|
var Hash = util.Uint160{0x33, 0x22, 0x11, 0x0, 0xff, 0xee, 0xdd, 0xcc, 0xbb, 0xaa, 0x99, 0x88, 0x77, 0x66, 0x55, 0x44, 0x33, 0x22, 0x11, 0x0}
|
||||||
|
|
||||||
|
// HelloWorldEvent represents "Hello world!" event emitted by the contract.
|
||||||
|
type HelloWorldEvent struct {
|
||||||
|
Args []any
|
||||||
|
}
|
||||||
|
|
||||||
// Actor is used by Contract to call state-changing methods.
|
// Actor is used by Contract to call state-changing methods.
|
||||||
type Actor interface {
|
type Actor interface {
|
||||||
MakeCall(contract util.Uint160, method string, params ...interface{}) (*transaction.Transaction, error)
|
MakeCall(contract util.Uint160, method string, params ...any) (*transaction.Transaction, error)
|
||||||
MakeRun(script []byte) (*transaction.Transaction, error)
|
MakeRun(script []byte) (*transaction.Transaction, error)
|
||||||
MakeUnsignedCall(contract util.Uint160, method string, attrs []transaction.Attribute, params ...interface{}) (*transaction.Transaction, error)
|
MakeUnsignedCall(contract util.Uint160, method string, attrs []transaction.Attribute, params ...any) (*transaction.Transaction, error)
|
||||||
MakeUnsignedRun(script []byte, attrs []transaction.Attribute) (*transaction.Transaction, error)
|
MakeUnsignedRun(script []byte, attrs []transaction.Attribute) (*transaction.Transaction, error)
|
||||||
SendCall(contract util.Uint160, method string, params ...interface{}) (util.Uint256, uint32, error)
|
SendCall(contract util.Uint160, method string, params ...any) (util.Uint256, uint32, error)
|
||||||
SendRun(script []byte) (util.Uint256, uint32, error)
|
SendRun(script []byte) (util.Uint256, uint32, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Contract implements all contract methods.
|
// Contract implements all contract methods.
|
||||||
type Contract struct {
|
type Contract struct {
|
||||||
actor Actor
|
actor Actor
|
||||||
|
hash util.Uint160
|
||||||
}
|
}
|
||||||
|
|
||||||
// New creates an instance of Contract using Hash and the given Actor.
|
// New creates an instance of Contract using Hash and the given Actor.
|
||||||
func New(actor Actor) *Contract {
|
func New(actor Actor) *Contract {
|
||||||
return &Contract{actor}
|
var hash = Hash
|
||||||
|
return &Contract{actor, hash}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *Contract) scriptForVerify() ([]byte, error) {
|
||||||
func scriptForVerify() ([]byte, error) {
|
return smartcontract.CreateCallWithAssertScript(c.hash, "verify")
|
||||||
return smartcontract.CreateCallWithAssertScript(Hash, "verify")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Verify creates a transaction invoking `verify` method of the contract.
|
// Verify creates a transaction invoking `verify` method of the contract.
|
||||||
// This transaction is signed and immediately sent to the network.
|
// This transaction is signed and immediately sent to the network.
|
||||||
// The values returned are its hash, ValidUntilBlock value and error if any.
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
||||||
func (c *Contract) Verify() (util.Uint256, uint32, error) {
|
func (c *Contract) Verify() (util.Uint256, uint32, error) {
|
||||||
script, err := scriptForVerify()
|
script, err := c.scriptForVerify()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return util.Uint256{}, 0, err
|
return util.Uint256{}, 0, err
|
||||||
}
|
}
|
||||||
|
@ -50,7 +62,7 @@ func (c *Contract) Verify() (util.Uint256, uint32, error) {
|
||||||
// This transaction is signed, but not sent to the network, instead it's
|
// This transaction is signed, but not sent to the network, instead it's
|
||||||
// returned to the caller.
|
// returned to the caller.
|
||||||
func (c *Contract) VerifyTransaction() (*transaction.Transaction, error) {
|
func (c *Contract) VerifyTransaction() (*transaction.Transaction, error) {
|
||||||
script, err := scriptForVerify()
|
script, err := c.scriptForVerify()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -62,9 +74,74 @@ func (c *Contract) VerifyTransaction() (*transaction.Transaction, error) {
|
||||||
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
||||||
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
||||||
func (c *Contract) VerifyUnsigned() (*transaction.Transaction, error) {
|
func (c *Contract) VerifyUnsigned() (*transaction.Transaction, error) {
|
||||||
script, err := scriptForVerify()
|
script, err := c.scriptForVerify()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return c.actor.MakeUnsignedRun(script, nil)
|
return c.actor.MakeUnsignedRun(script, nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// HelloWorldEventsFromApplicationLog retrieves a set of all emitted events
|
||||||
|
// with "Hello world!" name from the provided [result.ApplicationLog].
|
||||||
|
func HelloWorldEventsFromApplicationLog(log *result.ApplicationLog) ([]*HelloWorldEvent, error) {
|
||||||
|
if log == nil {
|
||||||
|
return nil, errors.New("nil application log")
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []*HelloWorldEvent
|
||||||
|
for i, ex := range log.Executions {
|
||||||
|
for j, e := range ex.Events {
|
||||||
|
if e.Name != "Hello world!" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
event := new(HelloWorldEvent)
|
||||||
|
err := event.FromStackItem(e.Item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to deserialize HelloWorldEvent from stackitem (execution #%d, event #%d): %w", i, j, err)
|
||||||
|
}
|
||||||
|
res = append(res, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FromStackItem converts provided [stackitem.Array] to HelloWorldEvent or
|
||||||
|
// returns an error if it's not possible to do to so.
|
||||||
|
func (e *HelloWorldEvent) FromStackItem(item *stackitem.Array) error {
|
||||||
|
if item == nil {
|
||||||
|
return errors.New("nil item")
|
||||||
|
}
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("not an array")
|
||||||
|
}
|
||||||
|
if len(arr) != 1 {
|
||||||
|
return errors.New("wrong number of structure elements")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
index = -1
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
index++
|
||||||
|
e.Args, err = func(item stackitem.Item) ([]any, error) {
|
||||||
|
arr, ok := item.Value().([]stackitem.Item)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("not an array")
|
||||||
|
}
|
||||||
|
res := make([]any, len(arr))
|
||||||
|
for i := range res {
|
||||||
|
res[i], err = arr[i].Value(), error(nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}(arr[index])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("field Args: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
64
cli/testdata/testwallet_multi.json
vendored
Normal file
64
cli/testdata/testwallet_multi.json
vendored
Normal file
|
@ -0,0 +1,64 @@
|
||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"accounts": [
|
||||||
|
{
|
||||||
|
"address": "NgHcPxgEKZQV4QBedzyASJrgiANhJqBVLw",
|
||||||
|
"key": "6PYTbVq2P3AJQwWU5SFMKLjHYco7QABtNRo4ZvLvXhyaYjwMcuZm6xKokT",
|
||||||
|
"label": "one",
|
||||||
|
"contract": {
|
||||||
|
"script": "DCECnmSGVirDOqMr57EHaYz0YMTjaHQtO9FQYu8DMTCDw6VBVuezJw==",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "parameter0",
|
||||||
|
"type": "Signature"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"deployed": false
|
||||||
|
},
|
||||||
|
"lock": false,
|
||||||
|
"isDefault": false
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"address": "NLvHRfKAifjio2z9HiwLo9ZnpRPHUbAHgH",
|
||||||
|
"key": "6PYUjQ8TgR3cduEpG5niUNuPEWi3tYiQsnC4Jha9nGAJ6tAQGUmcrZXsLF",
|
||||||
|
"label": "two",
|
||||||
|
"contract": {
|
||||||
|
"script": "DCECgk91c1ABAX3A1uJNnxhGlp7NwUJScwJzJhrsYrXIbgNBVuezJw==",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "parameter0",
|
||||||
|
"type": "Signature"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"deployed": false
|
||||||
|
},
|
||||||
|
"lock": false,
|
||||||
|
"isDefault": false
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"address": "NcDfG8foJx79XSihcDDrx1df7cHAoJBfXj",
|
||||||
|
"key": "6PYRkUQKWFrTovHyeQZ7X4nWoDXKohtFRKW51LiCz317pwCjmB1cVwpcxz",
|
||||||
|
"label": "three",
|
||||||
|
"contract": {
|
||||||
|
"script": "DCEC9v0ZqBg8f4jJX9WR891M0bazf0FYTNu7MEgpSHrb9CVBVuezJw==",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "parameter0",
|
||||||
|
"type": "Signature"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"deployed": false
|
||||||
|
},
|
||||||
|
"lock": false,
|
||||||
|
"isDefault": false
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"scrypt": {
|
||||||
|
"n": 2,
|
||||||
|
"r": 1,
|
||||||
|
"p": 1
|
||||||
|
},
|
||||||
|
"extra": {
|
||||||
|
"Tokens": null
|
||||||
|
}
|
||||||
|
}
|
|
@ -5,13 +5,16 @@ package txctx
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/cli/flags"
|
"github.com/nspcc-dev/neo-go/cli/flags"
|
||||||
"github.com/nspcc-dev/neo-go/cli/input"
|
"github.com/nspcc-dev/neo-go/cli/input"
|
||||||
"github.com/nspcc-dev/neo-go/cli/paramcontext"
|
"github.com/nspcc-dev/neo-go/cli/paramcontext"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/rpcclient/actor"
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/actor"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/wallet"
|
"github.com/nspcc-dev/neo-go/pkg/wallet"
|
||||||
"github.com/urfave/cli"
|
"github.com/urfave/cli"
|
||||||
)
|
)
|
||||||
|
@ -37,6 +40,11 @@ var (
|
||||||
Name: "force",
|
Name: "force",
|
||||||
Usage: "Do not ask for a confirmation (and ignore errors)",
|
Usage: "Do not ask for a confirmation (and ignore errors)",
|
||||||
}
|
}
|
||||||
|
// AwaitFlag is a flag used to wait for the transaction to be included in a block.
|
||||||
|
AwaitFlag = cli.BoolFlag{
|
||||||
|
Name: "await",
|
||||||
|
Usage: "wait for the transaction to be included in a block",
|
||||||
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
// SignAndSend adds network and system fees to the provided transaction and
|
// SignAndSend adds network and system fees to the provided transaction and
|
||||||
|
@ -48,6 +56,7 @@ func SignAndSend(ctx *cli.Context, act *actor.Actor, acc *wallet.Account, tx *tr
|
||||||
gas = flags.Fixed8FromContext(ctx, "gas")
|
gas = flags.Fixed8FromContext(ctx, "gas")
|
||||||
sysgas = flags.Fixed8FromContext(ctx, "sysgas")
|
sysgas = flags.Fixed8FromContext(ctx, "sysgas")
|
||||||
ver = act.GetVersion()
|
ver = act.GetVersion()
|
||||||
|
aer *state.AppExecResult
|
||||||
)
|
)
|
||||||
|
|
||||||
tx.SystemFee += int64(sysgas)
|
tx.SystemFee += int64(sysgas)
|
||||||
|
@ -66,14 +75,39 @@ func SignAndSend(ctx *cli.Context, act *actor.Actor, acc *wallet.Account, tx *tr
|
||||||
}
|
}
|
||||||
waitTime := time.Since(promptTime)
|
waitTime := time.Since(promptTime)
|
||||||
// Compensate for confirmation waiting.
|
// Compensate for confirmation waiting.
|
||||||
tx.ValidUntilBlock += uint32((waitTime.Milliseconds() / int64(ver.Protocol.MillisecondsPerBlock))) + 1
|
tx.ValidUntilBlock += uint32(waitTime.Milliseconds()/int64(ver.Protocol.MillisecondsPerBlock)) + 2
|
||||||
|
}
|
||||||
|
var (
|
||||||
|
resTx util.Uint256
|
||||||
|
vub uint32
|
||||||
|
)
|
||||||
|
resTx, vub, err = act.SignAndSend(tx)
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(err, 1)
|
||||||
|
}
|
||||||
|
if ctx.Bool("await") {
|
||||||
|
aer, err = act.Wait(resTx, vub, err)
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(fmt.Errorf("failed to await transaction %s: %w", resTx.StringLE(), err), 1)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
_, _, err = act.SignAndSend(tx)
|
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(err, 1)
|
return cli.NewExitError(err, 1)
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Fprintln(ctx.App.Writer, tx.Hash().StringLE())
|
DumpTransactionInfo(ctx.App.Writer, tx.Hash(), aer)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// DumpTransactionInfo prints transaction info to the given writer.
|
||||||
|
func DumpTransactionInfo(w io.Writer, h util.Uint256, res *state.AppExecResult) {
|
||||||
|
fmt.Fprintln(w, h.StringLE())
|
||||||
|
if res != nil {
|
||||||
|
fmt.Fprintf(w, "OnChain:\t%t\n", res != nil)
|
||||||
|
fmt.Fprintf(w, "VMState:\t%s\n", res.VMState.String())
|
||||||
|
if res.FaultException != "" {
|
||||||
|
fmt.Fprintf(w, "FaultException:\t%s\n", res.FaultException)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
103
cli/util/cancel.go
Normal file
103
cli/util/cancel.go
Normal file
|
@ -0,0 +1,103 @@
|
||||||
|
package util
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/nspcc-dev/neo-go/cli/cmdargs"
|
||||||
|
"github.com/nspcc-dev/neo-go/cli/flags"
|
||||||
|
"github.com/nspcc-dev/neo-go/cli/options"
|
||||||
|
"github.com/nspcc-dev/neo-go/cli/txctx"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/neorpc/result"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/actor"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/waiter"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/vm/opcode"
|
||||||
|
"github.com/urfave/cli"
|
||||||
|
)
|
||||||
|
|
||||||
|
func cancelTx(ctx *cli.Context) error {
|
||||||
|
args := ctx.Args()
|
||||||
|
if len(args) == 0 {
|
||||||
|
return cli.NewExitError("transaction hash is missing", 1)
|
||||||
|
} else if len(args) > 1 {
|
||||||
|
return cli.NewExitError("only one transaction hash is accepted", 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
txHash, err := util.Uint256DecodeStringLE(strings.TrimPrefix(args[0], "0x"))
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(fmt.Sprintf("invalid tx hash: %s", args[0]), 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
gctx, cancel := options.GetTimeoutContext(ctx)
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
acc, w, err := options.GetAccFromContext(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(fmt.Errorf("failed to get account from context to sign the conflicting transaction: %w", err), 1)
|
||||||
|
}
|
||||||
|
defer w.Close()
|
||||||
|
|
||||||
|
signers, err := cmdargs.GetSignersAccounts(acc, w, nil, transaction.CalledByEntry)
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(fmt.Errorf("invalid signers: %w", err), 1)
|
||||||
|
}
|
||||||
|
c, a, exitErr := options.GetRPCWithActor(gctx, ctx, signers)
|
||||||
|
if exitErr != nil {
|
||||||
|
return exitErr
|
||||||
|
}
|
||||||
|
|
||||||
|
mainTx, _ := c.GetRawTransactionVerbose(txHash)
|
||||||
|
if mainTx != nil && !mainTx.Blockhash.Equals(util.Uint256{}) {
|
||||||
|
return cli.NewExitError(fmt.Errorf("target transaction %s is accepted at block %s", txHash, mainTx.Blockhash.StringLE()), 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
if mainTx != nil && !mainTx.HasSigner(acc.ScriptHash()) {
|
||||||
|
return cli.NewExitError(fmt.Errorf("account %s is not a signer of the conflicting transaction", acc.Address), 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
resHash, resVub, err := a.SendTunedRun([]byte{byte(opcode.RET)}, []transaction.Attribute{{Type: transaction.ConflictsT, Value: &transaction.Conflicts{Hash: txHash}}}, func(r *result.Invoke, t *transaction.Transaction) error {
|
||||||
|
err := actor.DefaultCheckerModifier(r, t)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if mainTx != nil && t.NetworkFee < mainTx.NetworkFee+1 {
|
||||||
|
t.NetworkFee = mainTx.NetworkFee + 1
|
||||||
|
}
|
||||||
|
t.NetworkFee += int64(flags.Fixed8FromContext(ctx, "gas"))
|
||||||
|
if mainTx != nil {
|
||||||
|
t.ValidUntilBlock = mainTx.ValidUntilBlock
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(fmt.Errorf("failed to send conflicting transaction: %w", err), 1)
|
||||||
|
}
|
||||||
|
var res *state.AppExecResult
|
||||||
|
if ctx.Bool("await") {
|
||||||
|
res, err = a.WaitAny(gctx, resVub, txHash, resHash)
|
||||||
|
if err != nil {
|
||||||
|
if errors.Is(err, waiter.ErrTxNotAccepted) {
|
||||||
|
if mainTx == nil {
|
||||||
|
return cli.NewExitError(fmt.Errorf("neither target nor conflicting transaction is accepted before the current height %d (ValidUntilBlock value of conlicting transaction). Main transaction is unknown to the provided RPC node, thus still has chances to be accepted, you may try cancellation again", resVub), 1)
|
||||||
|
}
|
||||||
|
fmt.Fprintf(ctx.App.Writer, "Neither target nor conflicting transaction is accepted before the current height %d (ValidUntilBlock value of both target and conflicting transactions). Main transaction is not valid anymore, cancellation is successful\n", resVub)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return cli.NewExitError(fmt.Errorf("failed to await target/ conflicting transaction %s/ %s: %w", txHash.StringLE(), resHash.StringLE(), err), 1)
|
||||||
|
}
|
||||||
|
if txHash.Equals(res.Container) {
|
||||||
|
tx, err := c.GetRawTransactionVerbose(txHash)
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(fmt.Errorf("target transaction %s is accepted", txHash), 1)
|
||||||
|
}
|
||||||
|
return cli.NewExitError(fmt.Errorf("target transaction %s is accepted at block %s", txHash, tx.Blockhash.StringLE()), 1)
|
||||||
|
}
|
||||||
|
fmt.Fprintln(ctx.App.Writer, "Conflicting transaction accepted")
|
||||||
|
}
|
||||||
|
txctx.DumpTransactionInfo(ctx.App.Writer, resHash, res)
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -1,16 +1,32 @@
|
||||||
package util
|
package util
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"encoding/base64"
|
||||||
|
"encoding/hex"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"github.com/nspcc-dev/neo-go/cli/flags"
|
||||||
"github.com/nspcc-dev/neo-go/cli/options"
|
"github.com/nspcc-dev/neo-go/cli/options"
|
||||||
|
"github.com/nspcc-dev/neo-go/cli/txctx"
|
||||||
vmcli "github.com/nspcc-dev/neo-go/cli/vm"
|
vmcli "github.com/nspcc-dev/neo-go/cli/vm"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/vm"
|
||||||
"github.com/urfave/cli"
|
"github.com/urfave/cli"
|
||||||
)
|
)
|
||||||
|
|
||||||
// NewCommands returns util commands for neo-go CLI.
|
// NewCommands returns util commands for neo-go CLI.
|
||||||
func NewCommands() []cli.Command {
|
func NewCommands() []cli.Command {
|
||||||
txDumpFlags := append([]cli.Flag{}, options.RPC...)
|
txDumpFlags := append([]cli.Flag{}, options.RPC...)
|
||||||
|
txSendFlags := append(txDumpFlags, txctx.AwaitFlag)
|
||||||
|
txCancelFlags := append([]cli.Flag{
|
||||||
|
flags.AddressFlag{
|
||||||
|
Name: "address, a",
|
||||||
|
Usage: "address to use as conflicting transaction signee (and gas source)",
|
||||||
|
},
|
||||||
|
txctx.GasFlag,
|
||||||
|
txctx.AwaitFlag,
|
||||||
|
}, options.RPC...)
|
||||||
|
txCancelFlags = append(txCancelFlags, options.Wallet...)
|
||||||
return []cli.Command{
|
return []cli.Command{
|
||||||
{
|
{
|
||||||
Name: "util",
|
Name: "util",
|
||||||
|
@ -28,14 +44,35 @@ func NewCommands() []cli.Command {
|
||||||
{
|
{
|
||||||
Name: "sendtx",
|
Name: "sendtx",
|
||||||
Usage: "Send complete transaction stored in a context file",
|
Usage: "Send complete transaction stored in a context file",
|
||||||
UsageText: "sendtx [-r <endpoint>] <file.in>",
|
UsageText: "sendtx [-r <endpoint>] <file.in> [--await]",
|
||||||
Description: `Sends the transaction from the given context file to the given RPC node if it's
|
Description: `Sends the transaction from the given context file to the given RPC node if it's
|
||||||
completely signed and ready. This command expects a ContractParametersContext
|
completely signed and ready. This command expects a ContractParametersContext
|
||||||
JSON file for input, it can't handle binary (or hex- or base64-encoded)
|
JSON file for input, it can't handle binary (or hex- or base64-encoded)
|
||||||
transactions.
|
transactions. If the --await flag is included, the command waits for the
|
||||||
|
transaction to be included in a block before exiting.
|
||||||
`,
|
`,
|
||||||
Action: sendTx,
|
Action: sendTx,
|
||||||
Flags: txDumpFlags,
|
Flags: txSendFlags,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "canceltx",
|
||||||
|
Usage: "Cancel transaction by sending conflicting transaction",
|
||||||
|
UsageText: "canceltx <txid> -r <endpoint> --wallet <wallet> [--account <account>] [--wallet-config <path>] [--gas <gas>] [--await]",
|
||||||
|
Description: `Aims to prevent a transaction from being added to the blockchain by dispatching a more
|
||||||
|
prioritized conflicting transaction to the specified RPC node. The input for this command should
|
||||||
|
be the transaction hash. If another account is not specified, the conflicting transaction is
|
||||||
|
automatically generated and signed by the default account in the wallet. If the target transaction
|
||||||
|
is in the memory pool of the provided RPC node, the NetworkFee value of the conflicting transaction
|
||||||
|
is set to the target transaction's NetworkFee value plus one (if it's sufficient for the
|
||||||
|
conflicting transaction itself), the ValidUntilBlock value of the conflicting transaction is set to the
|
||||||
|
target transaction's ValidUntilBlock value. If the target transaction is not in the memory pool, standard
|
||||||
|
NetworkFee calculations are performed based on the calculatenetworkfee RPC request. If the --gas
|
||||||
|
flag is included, the specified value is added to the resulting conflicting transaction network fee
|
||||||
|
in both scenarios. When the --await flag is included, the command waits for one of the conflicting
|
||||||
|
or target transactions to be included in a block.
|
||||||
|
`,
|
||||||
|
Action: cancelTx,
|
||||||
|
Flags: txCancelFlags,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "txdump",
|
Name: "txdump",
|
||||||
|
@ -43,6 +80,28 @@ func NewCommands() []cli.Command {
|
||||||
UsageText: "txdump [-r <endpoint>] <file.in>",
|
UsageText: "txdump [-r <endpoint>] <file.in>",
|
||||||
Action: txDump,
|
Action: txDump,
|
||||||
Flags: txDumpFlags,
|
Flags: txDumpFlags,
|
||||||
|
Description: `Dumps the transaction from the given parameter context file to
|
||||||
|
the output. This command expects a ContractParametersContext JSON file for input, it can't handle
|
||||||
|
binary (or hex- or base64-encoded) transactions. If --rpc-endpoint flag is specified the result
|
||||||
|
of the given script after running it true the VM will be printed. Otherwise only transaction will
|
||||||
|
be printed.
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "ops",
|
||||||
|
Usage: "Pretty-print VM opcodes of the given base64- or hex- encoded script (base64 is checked first). If the input file is specified, then the script is taken from the file.",
|
||||||
|
UsageText: "ops <base64/hex-encoded script> [-i path-to-file] [--hex]",
|
||||||
|
Action: handleOps,
|
||||||
|
Flags: []cli.Flag{
|
||||||
|
cli.StringFlag{
|
||||||
|
Name: "in, i",
|
||||||
|
Usage: "input file containing base64- or hex- encoded script representation",
|
||||||
|
},
|
||||||
|
cli.BoolFlag{
|
||||||
|
Name: "hex",
|
||||||
|
Usage: "use hex encoding and do not check base64",
|
||||||
|
},
|
||||||
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -57,3 +116,35 @@ func handleParse(ctx *cli.Context) error {
|
||||||
fmt.Fprint(ctx.App.Writer, res)
|
fmt.Fprint(ctx.App.Writer, res)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func handleOps(ctx *cli.Context) error {
|
||||||
|
var (
|
||||||
|
s string
|
||||||
|
err error
|
||||||
|
b []byte
|
||||||
|
)
|
||||||
|
in := ctx.String("in")
|
||||||
|
if len(in) != 0 {
|
||||||
|
b, err := os.ReadFile(in)
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(fmt.Errorf("failed to read file: %w", err), 1)
|
||||||
|
}
|
||||||
|
s = string(b)
|
||||||
|
} else {
|
||||||
|
if !ctx.Args().Present() {
|
||||||
|
return cli.NewExitError("missing script", 1)
|
||||||
|
}
|
||||||
|
s = ctx.Args()[0]
|
||||||
|
}
|
||||||
|
b, err = base64.StdEncoding.DecodeString(s)
|
||||||
|
if err != nil || ctx.Bool("hex") {
|
||||||
|
b, err = hex.DecodeString(s)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError("unknown encoding: base64 or hex are supported", 1)
|
||||||
|
}
|
||||||
|
v := vm.New()
|
||||||
|
v.LoadScript(b)
|
||||||
|
v.PrintOps(ctx.App.Writer)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
|
@ -29,7 +29,10 @@ func txDump(ctx *cli.Context) error {
|
||||||
return cli.NewExitError("verifiable item is not a transaction", 1)
|
return cli.NewExitError("verifiable item is not a transaction", 1)
|
||||||
}
|
}
|
||||||
|
|
||||||
query.DumpApplicationLog(ctx, nil, tx, nil, true)
|
err = query.DumpApplicationLog(ctx, nil, tx, nil, true)
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(err, 1)
|
||||||
|
}
|
||||||
|
|
||||||
if ctx.String(options.RPCEndpointFlag) != "" {
|
if ctx.String(options.RPCEndpointFlag) != "" {
|
||||||
gctx, cancel := options.GetTimeoutContext(ctx)
|
gctx, cancel := options.GetTimeoutContext(ctx)
|
||||||
|
|
|
@ -5,6 +5,9 @@ import (
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/cli/options"
|
"github.com/nspcc-dev/neo-go/cli/options"
|
||||||
"github.com/nspcc-dev/neo-go/cli/paramcontext"
|
"github.com/nspcc-dev/neo-go/cli/paramcontext"
|
||||||
|
"github.com/nspcc-dev/neo-go/cli/txctx"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/waiter"
|
||||||
"github.com/urfave/cli"
|
"github.com/urfave/cli"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -37,6 +40,14 @@ func sendTx(ctx *cli.Context) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(fmt.Errorf("failed to submit transaction to RPC node: %w", err), 1)
|
return cli.NewExitError(fmt.Errorf("failed to submit transaction to RPC node: %w", err), 1)
|
||||||
}
|
}
|
||||||
fmt.Fprintln(ctx.App.Writer, res.StringLE())
|
var aer *state.AppExecResult
|
||||||
|
if ctx.Bool("await") {
|
||||||
|
version, err := c.GetVersion()
|
||||||
|
aer, err = waiter.New(c, version).Wait(res, tx.ValidUntilBlock, err)
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(fmt.Errorf("failed to await transaction %s: %w", res.StringLE(), err), 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
txctx.DumpTransactionInfo(ctx.App.Writer, res, aer)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,10 +1,19 @@
|
||||||
package util_test
|
package util_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/internal/testcli"
|
"github.com/nspcc-dev/neo-go/internal/testcli"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/trigger"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/util"
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/vm/vmstate"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/wallet"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestUtilConvert(t *testing.T) {
|
func TestUtilConvert(t *testing.T) {
|
||||||
|
@ -24,3 +33,160 @@ func TestUtilConvert(t *testing.T) {
|
||||||
e.CheckNextLine(t, "MDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAzMDIwMQ==") // string to base64
|
e.CheckNextLine(t, "MDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAzMDIwMQ==") // string to base64
|
||||||
e.CheckEOF(t)
|
e.CheckEOF(t)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestUtilOps(t *testing.T) {
|
||||||
|
e := testcli.NewExecutor(t, false)
|
||||||
|
base64Str := "EUA="
|
||||||
|
hexStr := "1140"
|
||||||
|
|
||||||
|
check := func(t *testing.T) {
|
||||||
|
e.CheckNextLine(t, "INDEX.*OPCODE.*PARAMETER")
|
||||||
|
e.CheckNextLine(t, "PUSH1")
|
||||||
|
e.CheckNextLine(t, "RET")
|
||||||
|
e.CheckEOF(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
e.Run(t, "neo-go", "util", "ops", base64Str) // base64
|
||||||
|
check(t)
|
||||||
|
|
||||||
|
e.Run(t, "neo-go", "util", "ops", hexStr) // base64 is checked firstly by default, but it's invalid script if decoded from base64
|
||||||
|
e.CheckNextLine(t, "INDEX.*OPCODE.*PARAMETER")
|
||||||
|
e.CheckNextLine(t, ".*ERROR: incorrect opcode")
|
||||||
|
e.CheckEOF(t)
|
||||||
|
|
||||||
|
e.Run(t, "neo-go", "util", "ops", "--hex", hexStr) // explicitly specify hex encoding
|
||||||
|
check(t)
|
||||||
|
|
||||||
|
e.RunWithError(t, "neo-go", "util", "ops", "%&~*") // unknown encoding
|
||||||
|
|
||||||
|
tmp := filepath.Join(t.TempDir(), "script_base64.txt")
|
||||||
|
require.NoError(t, os.WriteFile(tmp, []byte(base64Str), os.ModePerm))
|
||||||
|
e.Run(t, "neo-go", "util", "ops", "--in", tmp) // base64 from file
|
||||||
|
check(t)
|
||||||
|
|
||||||
|
tmp = filepath.Join(t.TempDir(), "script_hex.txt")
|
||||||
|
require.NoError(t, os.WriteFile(tmp, []byte(hexStr), os.ModePerm))
|
||||||
|
e.Run(t, "neo-go", "util", "ops", "--hex", "--in", tmp) // hex from file
|
||||||
|
check(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestUtilCancelTx(t *testing.T) {
|
||||||
|
e := testcli.NewExecutorSuspended(t)
|
||||||
|
|
||||||
|
w, err := wallet.NewWalletFromFile("../testdata/testwallet.json")
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
transferArgs := []string{
|
||||||
|
"neo-go", "wallet", "nep17", "transfer",
|
||||||
|
"--rpc-endpoint", "http://" + e.RPC.Addresses()[0],
|
||||||
|
"--wallet", testcli.ValidatorWallet,
|
||||||
|
"--to", w.Accounts[0].Address,
|
||||||
|
"--token", "NEO",
|
||||||
|
"--from", testcli.ValidatorAddr,
|
||||||
|
"--force",
|
||||||
|
}
|
||||||
|
args := []string{"neo-go", "util", "canceltx",
|
||||||
|
"-r", "http://" + e.RPC.Addresses()[0],
|
||||||
|
"--wallet", testcli.ValidatorWallet,
|
||||||
|
"--address", testcli.ValidatorAddr}
|
||||||
|
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.Run(t, append(transferArgs, "--amount", "1")...)
|
||||||
|
line := e.GetNextLine(t)
|
||||||
|
txHash, err := util.Uint256DecodeStringLE(line)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
_, ok := e.Chain.GetMemPool().TryGetValue(txHash)
|
||||||
|
require.True(t, ok)
|
||||||
|
|
||||||
|
t.Run("invalid", func(t *testing.T) {
|
||||||
|
t.Run("missing tx argument", func(t *testing.T) {
|
||||||
|
e.RunWithError(t, args...)
|
||||||
|
})
|
||||||
|
t.Run("excessive arguments", func(t *testing.T) {
|
||||||
|
e.RunWithError(t, append(args, txHash.StringLE(), txHash.StringLE())...)
|
||||||
|
})
|
||||||
|
t.Run("invalid hash", func(t *testing.T) {
|
||||||
|
e.RunWithError(t, append(args, "notahash")...)
|
||||||
|
})
|
||||||
|
t.Run("not signed by main signer", func(t *testing.T) {
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.RunWithError(t, "neo-go", "util", "canceltx",
|
||||||
|
"-r", "http://"+e.RPC.Addresses()[0],
|
||||||
|
"--wallet", testcli.ValidatorWallet,
|
||||||
|
"--address", testcli.MultisigAddr, txHash.StringLE())
|
||||||
|
})
|
||||||
|
t.Run("wrong rpc endpoint", func(t *testing.T) {
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.RunWithError(t, "neo-go", "util", "canceltx",
|
||||||
|
"-r", "http://localhost:20331",
|
||||||
|
"--wallet", testcli.ValidatorWallet, txHash.StringLE())
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.Run(t, append(args, txHash.StringLE())...)
|
||||||
|
resHash, err := util.Uint256DecodeStringLE(e.GetNextLine(t))
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
_, _, err = e.Chain.GetTransaction(resHash)
|
||||||
|
require.NoError(t, err)
|
||||||
|
e.CheckEOF(t)
|
||||||
|
go e.Chain.Run()
|
||||||
|
|
||||||
|
require.Eventually(t, func() bool {
|
||||||
|
_, aerErr := e.Chain.GetAppExecResults(resHash, trigger.Application)
|
||||||
|
return aerErr == nil
|
||||||
|
}, time.Second*2, time.Millisecond*50)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAwaitUtilCancelTx(t *testing.T) {
|
||||||
|
e := testcli.NewExecutor(t, true)
|
||||||
|
|
||||||
|
w, err := wallet.NewWalletFromFile("../testdata/testwallet.json")
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
transferArgs := []string{
|
||||||
|
"neo-go", "wallet", "nep17", "transfer",
|
||||||
|
"--rpc-endpoint", "http://" + e.RPC.Addresses()[0],
|
||||||
|
"--wallet", testcli.ValidatorWallet,
|
||||||
|
"--to", w.Accounts[0].Address,
|
||||||
|
"--token", "NEO",
|
||||||
|
"--from", testcli.ValidatorAddr,
|
||||||
|
"--force",
|
||||||
|
}
|
||||||
|
args := []string{"neo-go", "util", "canceltx",
|
||||||
|
"-r", "http://" + e.RPC.Addresses()[0],
|
||||||
|
"--wallet", testcli.ValidatorWallet,
|
||||||
|
"--address", testcli.ValidatorAddr,
|
||||||
|
"--await"}
|
||||||
|
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.Run(t, append(transferArgs, "--amount", "1")...)
|
||||||
|
line := e.GetNextLine(t)
|
||||||
|
txHash, err := util.Uint256DecodeStringLE(line)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
_, ok := e.Chain.GetMemPool().TryGetValue(txHash)
|
||||||
|
require.True(t, ok)
|
||||||
|
|
||||||
|
// Allow both cases: either target or conflicting tx acceptance.
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
err = e.RunUnchecked(t, append(args, txHash.StringLE())...)
|
||||||
|
switch {
|
||||||
|
case err == nil:
|
||||||
|
response := e.GetNextLine(t)
|
||||||
|
require.Equal(t, "Conflicting transaction accepted", response)
|
||||||
|
resHash, _ := e.CheckAwaitableTxPersisted(t)
|
||||||
|
require.NotEqual(t, resHash, txHash)
|
||||||
|
case strings.Contains(err.Error(), fmt.Sprintf("target transaction %s is accepted", txHash)) ||
|
||||||
|
strings.Contains(err.Error(), fmt.Sprintf("failed to send conflicting transaction: Invalid transaction attribute (-507) - invalid attribute: conflicting transaction %s is already on chain", txHash)):
|
||||||
|
tx, _ := e.GetTransaction(t, txHash)
|
||||||
|
aer, err := e.Chain.GetAppExecResults(tx.Hash(), trigger.Application)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, 1, len(aer))
|
||||||
|
require.Equal(t, vmstate.Halt, aer[0].VMState)
|
||||||
|
default:
|
||||||
|
t.Fatal(fmt.Errorf("unexpected error: %w", err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
277
cli/vm/cli.go
277
cli/vm/cli.go
|
@ -28,12 +28,14 @@ import (
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/interop"
|
"github.com/nspcc-dev/neo-go/pkg/core/interop"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/interop/runtime"
|
"github.com/nspcc-dev/neo-go/pkg/core/interop/runtime"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/native"
|
"github.com/nspcc-dev/neo-go/pkg/core/native"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/storage"
|
"github.com/nspcc-dev/neo-go/pkg/core/storage"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/storage/dbconfig"
|
"github.com/nspcc-dev/neo-go/pkg/core/storage/dbconfig"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/encoding/bigint"
|
"github.com/nspcc-dev/neo-go/pkg/encoding/bigint"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract/callflag"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/callflag"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract/nef"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/nef"
|
||||||
|
@ -51,7 +53,7 @@ const (
|
||||||
chainKey = "chain"
|
chainKey = "chain"
|
||||||
chainCfgKey = "chainCfg"
|
chainCfgKey = "chainCfg"
|
||||||
icKey = "ic"
|
icKey = "ic"
|
||||||
manifestKey = "manifest"
|
contractStateKey = "contractState"
|
||||||
exitFuncKey = "exitFunc"
|
exitFuncKey = "exitFunc"
|
||||||
readlineInstanceKey = "readlineKey"
|
readlineInstanceKey = "readlineKey"
|
||||||
printLogoKey = "printLogoKey"
|
printLogoKey = "printLogoKey"
|
||||||
|
@ -64,6 +66,7 @@ const (
|
||||||
gasFlagFullName = "gas"
|
gasFlagFullName = "gas"
|
||||||
backwardsFlagFullName = "backwards"
|
backwardsFlagFullName = "backwards"
|
||||||
diffFlagFullName = "diff"
|
diffFlagFullName = "diff"
|
||||||
|
hashFlagFullName = "hash"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
@ -76,6 +79,10 @@ var (
|
||||||
Name: gasFlagFullName,
|
Name: gasFlagFullName,
|
||||||
Usage: "GAS limit for this execution (integer number, satoshi).",
|
Usage: "GAS limit for this execution (integer number, satoshi).",
|
||||||
}
|
}
|
||||||
|
hashFlag = cli.StringFlag{
|
||||||
|
Name: hashFlagFullName,
|
||||||
|
Usage: "Smart-contract hash in LE form or address",
|
||||||
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
var commands = []cli.Command{
|
var commands = []cli.Command{
|
||||||
|
@ -150,10 +157,12 @@ Example:
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "loadnef",
|
Name: "loadnef",
|
||||||
Usage: "Load a NEF-consistent script into the VM optionally attaching to it provided signers with scopes",
|
Usage: "Load a NEF (possibly with a contract hash) into the VM optionally using provided scoped signers in the context",
|
||||||
UsageText: `loadnef [--historic <height>] [--gas <int>] <file> <manifest> [<signer-with-scope>, ...]`,
|
UsageText: `loadnef [--historic <height>] [--gas <int>] [--hash <hash-or-address>] <file> [<manifest>] [-- <signer-with-scope>, ...]`,
|
||||||
Flags: []cli.Flag{historicFlag, gasFlag},
|
Flags: []cli.Flag{historicFlag, gasFlag, hashFlag},
|
||||||
Description: `<file> and <manifest> parameters are mandatory.
|
Description: `<file> parameter is mandatory, <manifest> parameter (if omitted) will
|
||||||
|
be guessed from the <file> parameter by replacing '.nef' suffix with '.manifest.json'
|
||||||
|
suffix.
|
||||||
|
|
||||||
` + cmdargs.SignersParsingDoc + `
|
` + cmdargs.SignersParsingDoc + `
|
||||||
|
|
||||||
|
@ -164,7 +173,7 @@ Example:
|
||||||
{
|
{
|
||||||
Name: "loadbase64",
|
Name: "loadbase64",
|
||||||
Usage: "Load a base64-encoded script string into the VM optionally attaching to it provided signers with scopes",
|
Usage: "Load a base64-encoded script string into the VM optionally attaching to it provided signers with scopes",
|
||||||
UsageText: `loadbase64 [--historic <height>] [--gas <int>] <string> [<signer-with-scope>, ...]`,
|
UsageText: `loadbase64 [--historic <height>] [--gas <int>] <string> [-- <signer-with-scope>, ...]`,
|
||||||
Flags: []cli.Flag{historicFlag, gasFlag},
|
Flags: []cli.Flag{historicFlag, gasFlag},
|
||||||
Description: `<string> is mandatory parameter.
|
Description: `<string> is mandatory parameter.
|
||||||
|
|
||||||
|
@ -177,7 +186,7 @@ Example:
|
||||||
{
|
{
|
||||||
Name: "loadhex",
|
Name: "loadhex",
|
||||||
Usage: "Load a hex-encoded script string into the VM optionally attaching to it provided signers with scopes",
|
Usage: "Load a hex-encoded script string into the VM optionally attaching to it provided signers with scopes",
|
||||||
UsageText: `loadhex [--historic <height>] [--gas <int>] <string> [<signer-with-scope>, ...]`,
|
UsageText: `loadhex [--historic <height>] [--gas <int>] <string> [-- <signer-with-scope>, ...]`,
|
||||||
Flags: []cli.Flag{historicFlag, gasFlag},
|
Flags: []cli.Flag{historicFlag, gasFlag},
|
||||||
Description: `<string> is mandatory parameter.
|
Description: `<string> is mandatory parameter.
|
||||||
|
|
||||||
|
@ -189,9 +198,9 @@ Example:
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "loadgo",
|
Name: "loadgo",
|
||||||
Usage: "Compile and load a Go file with the manifest into the VM optionally attaching to it provided signers with scopes",
|
Usage: "Compile and load a Go file with the manifest into the VM optionally attaching to it provided signers with scopes and setting provided hash",
|
||||||
UsageText: `loadgo [--historic <height>] [--gas <int>] <file> [<signer-with-scope>, ...]`,
|
UsageText: `loadgo [--historic <height>] [--gas <int>] [--hash <hash-or-address>] <file> [-- <signer-with-scope>, ...]`,
|
||||||
Flags: []cli.Flag{historicFlag, gasFlag},
|
Flags: []cli.Flag{historicFlag, gasFlag, hashFlag},
|
||||||
Description: `<file> is mandatory parameter.
|
Description: `<file> is mandatory parameter.
|
||||||
|
|
||||||
` + cmdargs.SignersParsingDoc + `
|
` + cmdargs.SignersParsingDoc + `
|
||||||
|
@ -220,7 +229,7 @@ Example:
|
||||||
{
|
{
|
||||||
Name: "loaddeployed",
|
Name: "loaddeployed",
|
||||||
Usage: "Load deployed contract into the VM from chain optionally attaching to it provided signers with scopes",
|
Usage: "Load deployed contract into the VM from chain optionally attaching to it provided signers with scopes",
|
||||||
UsageText: `loaddeployed [--historic <height>] [--gas <int>] <hash-or-address-or-id> [<signer-with-scope>, ...]`,
|
UsageText: `loaddeployed [--historic <height>] [--gas <int>] <hash-or-address-or-id> [-- <signer-with-scope>, ...]`,
|
||||||
Flags: []cli.Flag{historicFlag, gasFlag},
|
Flags: []cli.Flag{historicFlag, gasFlag},
|
||||||
Description: `Load deployed contract into the VM from chain optionally attaching to it provided signers with scopes.
|
Description: `Load deployed contract into the VM from chain optionally attaching to it provided signers with scopes.
|
||||||
If '--historic' flag specified, then the historic contract state (historic script and manifest) will be loaded.
|
If '--historic' flag specified, then the historic contract state (historic script and manifest) will be loaded.
|
||||||
|
@ -483,11 +492,11 @@ func NewWithConfig(printLogotype bool, onExit func(int), c *readline.Config, cfg
|
||||||
shell: ctl,
|
shell: ctl,
|
||||||
}
|
}
|
||||||
|
|
||||||
vmcli.shell.Metadata = map[string]interface{}{
|
vmcli.shell.Metadata = map[string]any{
|
||||||
chainKey: chain,
|
chainKey: chain,
|
||||||
chainCfgKey: cfg,
|
chainCfgKey: cfg,
|
||||||
icKey: ic,
|
icKey: ic,
|
||||||
manifestKey: new(manifest.Manifest),
|
contractStateKey: new(state.ContractBase),
|
||||||
exitFuncKey: exitF,
|
exitFuncKey: exitF,
|
||||||
readlineInstanceKey: l,
|
readlineInstanceKey: l,
|
||||||
printLogoKey: printLogotype,
|
printLogoKey: printLogotype,
|
||||||
|
@ -520,8 +529,8 @@ func getInteropContextFromContext(app *cli.App) *interop.Context {
|
||||||
return app.Metadata[icKey].(*interop.Context)
|
return app.Metadata[icKey].(*interop.Context)
|
||||||
}
|
}
|
||||||
|
|
||||||
func getManifestFromContext(app *cli.App) *manifest.Manifest {
|
func getContractStateFromContext(app *cli.App) *state.ContractBase {
|
||||||
return app.Metadata[manifestKey].(*manifest.Manifest)
|
return app.Metadata[contractStateKey].(*state.ContractBase)
|
||||||
}
|
}
|
||||||
|
|
||||||
func getPrintLogoFromContext(app *cli.App) bool {
|
func getPrintLogoFromContext(app *cli.App) bool {
|
||||||
|
@ -532,8 +541,8 @@ func setInteropContextInContext(app *cli.App, ic *interop.Context) {
|
||||||
app.Metadata[icKey] = ic
|
app.Metadata[icKey] = ic
|
||||||
}
|
}
|
||||||
|
|
||||||
func setManifestInContext(app *cli.App, m *manifest.Manifest) {
|
func setContractStateInContext(app *cli.App, cs *state.ContractBase) {
|
||||||
app.Metadata[manifestKey] = m
|
app.Metadata[contractStateKey] = cs
|
||||||
}
|
}
|
||||||
|
|
||||||
func checkVMIsReady(app *cli.App) bool {
|
func checkVMIsReady(app *cli.App) bool {
|
||||||
|
@ -607,7 +616,7 @@ func getInstructionParameter(c *cli.Context) (int, error) {
|
||||||
}
|
}
|
||||||
n, err := strconv.Atoi(args[0])
|
n, err := strconv.Atoi(args[0])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, fmt.Errorf("%w: %s", ErrInvalidParameter, err) //nolint:errorlint // errorlint: non-wrapping format verb for fmt.Errorf. Use `%w` to format errors
|
return 0, fmt.Errorf("%w: %w", ErrInvalidParameter, err)
|
||||||
}
|
}
|
||||||
return n, nil
|
return n, nil
|
||||||
}
|
}
|
||||||
|
@ -669,12 +678,49 @@ func prepareVM(c *cli.Context, tx *transaction.Transaction) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func getHashFlag(c *cli.Context) (util.Uint160, error) {
|
||||||
|
if !c.IsSet(hashFlagFullName) {
|
||||||
|
return util.Uint160{}, nil
|
||||||
|
}
|
||||||
|
h, err := flags.ParseAddress(c.String(hashFlagFullName))
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint160{}, fmt.Errorf("failed to parse contract hash: %w", err)
|
||||||
|
}
|
||||||
|
return h, nil
|
||||||
|
}
|
||||||
|
|
||||||
func handleLoadNEF(c *cli.Context) error {
|
func handleLoadNEF(c *cli.Context) error {
|
||||||
args := c.Args()
|
args := c.Args()
|
||||||
if len(args) < 2 {
|
if len(args) < 1 {
|
||||||
return fmt.Errorf("%w: <file> <manifest>", ErrMissingParameter)
|
return fmt.Errorf("%w: <nef> is required", ErrMissingParameter)
|
||||||
}
|
}
|
||||||
b, err := os.ReadFile(args[0])
|
nefFile := args[0]
|
||||||
|
var (
|
||||||
|
manifestFile string
|
||||||
|
signersStartOffset int
|
||||||
|
)
|
||||||
|
if len(args) == 2 {
|
||||||
|
manifestFile = args[1]
|
||||||
|
} else if len(args) == 3 {
|
||||||
|
if args[1] != cmdargs.CosignersSeparator {
|
||||||
|
return fmt.Errorf("%w: `%s` was expected as the second parameter, got %s", ErrInvalidParameter, cmdargs.CosignersSeparator, args[1])
|
||||||
|
}
|
||||||
|
signersStartOffset = 2
|
||||||
|
} else if len(args) > 3 {
|
||||||
|
if args[1] == cmdargs.CosignersSeparator {
|
||||||
|
signersStartOffset = 2
|
||||||
|
} else {
|
||||||
|
manifestFile = args[1]
|
||||||
|
if args[2] != cmdargs.CosignersSeparator {
|
||||||
|
return fmt.Errorf("%w: `%s` was expected as the third parameter, got %s", ErrInvalidParameter, cmdargs.CosignersSeparator, args[2])
|
||||||
|
}
|
||||||
|
signersStartOffset = 3
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if len(manifestFile) == 0 {
|
||||||
|
manifestFile = strings.TrimSuffix(nefFile, ".nef") + ".manifest.json"
|
||||||
|
}
|
||||||
|
b, err := os.ReadFile(nefFile)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -682,24 +728,34 @@ func handleLoadNEF(c *cli.Context) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("failed to decode NEF file: %w", err)
|
return fmt.Errorf("failed to decode NEF file: %w", err)
|
||||||
}
|
}
|
||||||
m, err := getManifestFromFile(args[1])
|
m, err := getManifestFromFile(manifestFile)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("failed to read manifest: %w", err)
|
return fmt.Errorf("failed to read manifest: %w", err)
|
||||||
}
|
}
|
||||||
var signers []transaction.Signer
|
var signers []transaction.Signer
|
||||||
if len(args) > 2 {
|
if signersStartOffset != 0 && len(args) > signersStartOffset {
|
||||||
signers, err = cmdargs.ParseSigners(c.Args()[2:])
|
signers, err = cmdargs.ParseSigners(c.Args()[signersStartOffset:])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("%w: %v", ErrInvalidParameter, err) //nolint:errorlint // errorlint: non-wrapping format verb for fmt.Errorf. Use `%w` to format errors
|
return fmt.Errorf("%w: failed to parse signers: %w", ErrInvalidParameter, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
err = prepareVM(c, createFakeTransaction(nef.Script, signers))
|
err = prepareVM(c, createFakeTransaction(nef.Script, signers))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
h, err := getHashFlag(c)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
cs := &state.ContractBase{
|
||||||
|
Hash: h,
|
||||||
|
NEF: nef,
|
||||||
|
Manifest: *m,
|
||||||
|
}
|
||||||
|
setContractStateInContext(c.App, cs)
|
||||||
|
|
||||||
v := getVMFromContext(c.App)
|
v := getVMFromContext(c.App)
|
||||||
fmt.Fprintf(c.App.Writer, "READY: loaded %d instructions\n", v.Context().LenInstr())
|
fmt.Fprintf(c.App.Writer, "READY: loaded %d instructions\n", v.Context().LenInstr())
|
||||||
setManifestInContext(c.App, m)
|
|
||||||
changePrompt(c.App)
|
changePrompt(c.App)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -711,13 +767,19 @@ func handleLoadBase64(c *cli.Context) error {
|
||||||
}
|
}
|
||||||
b, err := base64.StdEncoding.DecodeString(args[0])
|
b, err := base64.StdEncoding.DecodeString(args[0])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("%w: %s", ErrInvalidParameter, err) //nolint:errorlint // errorlint: non-wrapping format verb for fmt.Errorf. Use `%w` to format errors
|
return fmt.Errorf("%w: %w", ErrInvalidParameter, err)
|
||||||
}
|
}
|
||||||
var signers []transaction.Signer
|
var signers []transaction.Signer
|
||||||
if len(args) > 1 {
|
if len(args) > 1 {
|
||||||
signers, err = cmdargs.ParseSigners(args[1:])
|
if args[1] != cmdargs.CosignersSeparator {
|
||||||
|
return fmt.Errorf("%w: `%s` was expected as the second parameter, got %s", ErrInvalidParameter, cmdargs.CosignersSeparator, args[1])
|
||||||
|
}
|
||||||
|
if len(args) < 3 {
|
||||||
|
return fmt.Errorf("%w: signers expected after `%s`, got none", ErrInvalidParameter, cmdargs.CosignersSeparator)
|
||||||
|
}
|
||||||
|
signers, err = cmdargs.ParseSigners(args[2:])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("%w: %v", ErrInvalidParameter, err) //nolint:errorlint // errorlint: non-wrapping format verb for fmt.Errorf. Use `%w` to format errors
|
return fmt.Errorf("%w: %w", ErrInvalidParameter, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
err = prepareVM(c, createFakeTransaction(b, signers))
|
err = prepareVM(c, createFakeTransaction(b, signers))
|
||||||
|
@ -745,13 +807,19 @@ func handleLoadHex(c *cli.Context) error {
|
||||||
}
|
}
|
||||||
b, err := hex.DecodeString(args[0])
|
b, err := hex.DecodeString(args[0])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("%w: %s", ErrInvalidParameter, err) //nolint:errorlint // errorlint: non-wrapping format verb for fmt.Errorf. Use `%w` to format errors
|
return fmt.Errorf("%w: %w", ErrInvalidParameter, err)
|
||||||
}
|
}
|
||||||
var signers []transaction.Signer
|
var signers []transaction.Signer
|
||||||
if len(args) > 1 {
|
if len(args) > 1 {
|
||||||
signers, err = cmdargs.ParseSigners(args[1:])
|
if args[1] != cmdargs.CosignersSeparator {
|
||||||
|
return fmt.Errorf("%w: `%s` was expected as the second parameter, got %s", ErrInvalidParameter, cmdargs.CosignersSeparator, args[1])
|
||||||
|
}
|
||||||
|
if len(args) < 3 {
|
||||||
|
return fmt.Errorf("%w: signers expected after `%s`, got none", ErrInvalidParameter, cmdargs.CosignersSeparator)
|
||||||
|
}
|
||||||
|
signers, err = cmdargs.ParseSigners(args[2:])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("%w: %v", ErrInvalidParameter, err) //nolint:errorlint // errorlint: non-wrapping format verb for fmt.Errorf. Use `%w` to format errors
|
return fmt.Errorf("%w: %w", ErrInvalidParameter, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
err = prepareVM(c, createFakeTransaction(b, signers))
|
err = prepareVM(c, createFakeTransaction(b, signers))
|
||||||
|
@ -771,7 +839,7 @@ func handleLoadGo(c *cli.Context) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
name := strings.TrimSuffix(args[0], ".go")
|
name := strings.TrimSuffix(args[0], ".go")
|
||||||
b, di, err := compiler.CompileWithOptions(args[0], nil, &compiler.Options{Name: name})
|
ne, di, err := compiler.CompileWithOptions(args[0], nil, &compiler.Options{Name: name})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("failed to compile: %w", err)
|
return fmt.Errorf("failed to compile: %w", err)
|
||||||
}
|
}
|
||||||
|
@ -783,18 +851,34 @@ func handleLoadGo(c *cli.Context) error {
|
||||||
}
|
}
|
||||||
var signers []transaction.Signer
|
var signers []transaction.Signer
|
||||||
if len(args) > 1 {
|
if len(args) > 1 {
|
||||||
signers, err = cmdargs.ParseSigners(args[1:])
|
if args[1] != cmdargs.CosignersSeparator {
|
||||||
|
return fmt.Errorf("%w: `%s` was expected as the second parameter, got %s", ErrInvalidParameter, cmdargs.CosignersSeparator, args[1])
|
||||||
|
}
|
||||||
|
if len(args) < 3 {
|
||||||
|
return fmt.Errorf("%w: signers expected after `%s`, got none", ErrInvalidParameter, cmdargs.CosignersSeparator)
|
||||||
|
}
|
||||||
|
signers, err = cmdargs.ParseSigners(args[2:])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("%w: %v", ErrInvalidParameter, err) //nolint:errorlint // errorlint: non-wrapping format verb for fmt.Errorf. Use `%w` to format errors
|
return fmt.Errorf("%w: %w", ErrInvalidParameter, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
err = prepareVM(c, createFakeTransaction(b.Script, signers))
|
err = prepareVM(c, createFakeTransaction(ne.Script, signers))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
h, err := getHashFlag(c)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
cs := &state.ContractBase{
|
||||||
|
Hash: h,
|
||||||
|
NEF: *ne,
|
||||||
|
Manifest: *m,
|
||||||
|
}
|
||||||
|
setContractStateInContext(c.App, cs)
|
||||||
|
|
||||||
v := getVMFromContext(c.App)
|
v := getVMFromContext(c.App)
|
||||||
setManifestInContext(c.App, m)
|
|
||||||
fmt.Fprintf(c.App.Writer, "READY: loaded %d instructions\n", v.Context().LenInstr())
|
fmt.Fprintf(c.App.Writer, "READY: loaded %d instructions\n", v.Context().LenInstr())
|
||||||
changePrompt(c.App)
|
changePrompt(c.App)
|
||||||
return nil
|
return nil
|
||||||
|
@ -849,7 +933,8 @@ func handleLoadDeployed(c *cli.Context) error {
|
||||||
if !c.Args().Present() {
|
if !c.Args().Present() {
|
||||||
return errors.New("contract hash, address or ID is mandatory argument")
|
return errors.New("contract hash, address or ID is mandatory argument")
|
||||||
}
|
}
|
||||||
hashOrID := c.Args().Get(0)
|
args := c.Args()
|
||||||
|
hashOrID := args[0]
|
||||||
ic := getInteropContextFromContext(c.App)
|
ic := getInteropContextFromContext(c.App)
|
||||||
h, err := flags.ParseAddress(hashOrID)
|
h, err := flags.ParseAddress(hashOrID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -868,10 +953,16 @@ func handleLoadDeployed(c *cli.Context) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
var signers []transaction.Signer
|
var signers []transaction.Signer
|
||||||
if len(c.Args()) > 1 {
|
if len(args) > 1 {
|
||||||
signers, err = cmdargs.ParseSigners(c.Args()[1:])
|
if args[1] != cmdargs.CosignersSeparator {
|
||||||
|
return fmt.Errorf("%w: %s was expected as the second parameter, got %s", ErrInvalidParameter, cmdargs.CosignersSeparator, args[1])
|
||||||
|
}
|
||||||
|
if len(args) < 3 {
|
||||||
|
return fmt.Errorf("%w: signers expected after `%s`, got none", ErrInvalidParameter, cmdargs.CosignersSeparator)
|
||||||
|
}
|
||||||
|
signers, err = cmdargs.ParseSigners(args[2:])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("%w: %v", ErrInvalidParameter, err) //nolint:errorlint // errorlint: non-wrapping format verb for fmt.Errorf. Use `%w` to format errors
|
return fmt.Errorf("%w: %w", ErrInvalidParameter, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
err = prepareVM(c, createFakeTransaction(cs.NEF.Script, signers)) // prepare VM one more time for proper IC initialization.
|
err = prepareVM(c, createFakeTransaction(cs.NEF.Script, signers)) // prepare VM one more time for proper IC initialization.
|
||||||
|
@ -884,7 +975,7 @@ func handleLoadDeployed(c *cli.Context) error {
|
||||||
ic.VM.GasLimit = gasLimit
|
ic.VM.GasLimit = gasLimit
|
||||||
ic.VM.LoadScriptWithHash(cs.NEF.Script, cs.Hash, callflag.All)
|
ic.VM.LoadScriptWithHash(cs.NEF.Script, cs.Hash, callflag.All)
|
||||||
fmt.Fprintf(c.App.Writer, "READY: loaded %d instructions\n", ic.VM.Context().LenInstr())
|
fmt.Fprintf(c.App.Writer, "READY: loaded %d instructions\n", ic.VM.Context().LenInstr())
|
||||||
setManifestInContext(c.App, &cs.Manifest)
|
setContractStateInContext(c.App, &cs.ContractBase)
|
||||||
changePrompt(c.App)
|
changePrompt(c.App)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -939,9 +1030,9 @@ func resetInteropContext(app *cli.App, tx *transaction.Transaction, height ...ui
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// resetManifest removes manifest from app context.
|
// resetContractState removes loaded contract state from app context.
|
||||||
func resetManifest(app *cli.App) {
|
func resetContractState(app *cli.App) {
|
||||||
setManifestInContext(app, nil)
|
setContractStateInContext(app, nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
// resetState resets state of the app (clear interop context and manifest) so that it's ready
|
// resetState resets state of the app (clear interop context and manifest) so that it's ready
|
||||||
|
@ -951,7 +1042,7 @@ func resetState(app *cli.App, tx *transaction.Transaction, height ...uint32) err
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
resetManifest(app)
|
resetContractState(app)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -970,7 +1061,7 @@ func getManifestFromFile(name string) (*manifest.Manifest, error) {
|
||||||
|
|
||||||
func handleRun(c *cli.Context) error {
|
func handleRun(c *cli.Context) error {
|
||||||
v := getVMFromContext(c.App)
|
v := getVMFromContext(c.App)
|
||||||
m := getManifestFromContext(c.App)
|
cs := getContractStateFromContext(c.App)
|
||||||
args := c.Args()
|
args := c.Args()
|
||||||
if len(args) != 0 {
|
if len(args) != 0 {
|
||||||
var (
|
var (
|
||||||
|
@ -978,11 +1069,12 @@ func handleRun(c *cli.Context) error {
|
||||||
offset int
|
offset int
|
||||||
err error
|
err error
|
||||||
runCurrent = args[0] != "_"
|
runCurrent = args[0] != "_"
|
||||||
|
hasRet bool
|
||||||
)
|
)
|
||||||
|
|
||||||
_, scParams, err := cmdargs.ParseParams(args[1:], true)
|
_, scParams, err := cmdargs.ParseParams(args[1:], true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("%w: %v", ErrInvalidParameter, err) //nolint:errorlint // errorlint: non-wrapping format verb for fmt.Errorf. Use `%w` to format errors
|
return fmt.Errorf("%w: %w", ErrInvalidParameter, err)
|
||||||
}
|
}
|
||||||
params = make([]stackitem.Item, len(scParams))
|
params = make([]stackitem.Item, len(scParams))
|
||||||
for i := range scParams {
|
for i := range scParams {
|
||||||
|
@ -992,27 +1084,35 @@ func handleRun(c *cli.Context) error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if runCurrent {
|
if runCurrent {
|
||||||
if m == nil {
|
if cs == nil {
|
||||||
return fmt.Errorf("manifest is not loaded; either use 'run' command to run loaded script from the start or use 'loadgo' and 'loadnef' commands to provide manifest")
|
return fmt.Errorf("manifest is not loaded; either use 'run' command to run loaded script from the start or use 'loadgo', 'loadnef' or 'loaddeployed' commands to provide manifest")
|
||||||
}
|
}
|
||||||
md := m.ABI.GetMethod(args[0], len(params))
|
md := cs.Manifest.ABI.GetMethod(args[0], len(params))
|
||||||
if md == nil {
|
if md == nil {
|
||||||
return fmt.Errorf("%w: method not found", ErrInvalidParameter)
|
return fmt.Errorf("%w: method not found", ErrInvalidParameter)
|
||||||
}
|
}
|
||||||
|
hasRet = md.ReturnType != smartcontract.VoidType
|
||||||
offset = md.Offset
|
offset = md.Offset
|
||||||
|
var initOff = -1
|
||||||
|
if initMD := cs.Manifest.ABI.GetMethod(manifest.MethodInit, 0); initMD != nil {
|
||||||
|
initOff = initMD.Offset
|
||||||
|
}
|
||||||
|
|
||||||
|
// Clear context loaded by 'loadgo', 'loadnef' or 'loaddeployed' to properly handle LoadNEFMethod.
|
||||||
|
// At the same time, preserve previously set gas limit and the set of breakpoints.
|
||||||
|
ic := getInteropContextFromContext(c.App)
|
||||||
|
gasLimit := v.GasLimit
|
||||||
|
breaks := v.Context().BreakPoints() // We ensure that there's a context loaded.
|
||||||
|
ic.ReuseVM(v)
|
||||||
|
v.GasLimit = gasLimit
|
||||||
|
v.LoadNEFMethod(&cs.NEF, &cs.Manifest, util.Uint160{}, cs.Hash, callflag.All, hasRet, offset, initOff, nil)
|
||||||
|
for _, bp := range breaks {
|
||||||
|
v.AddBreakPoint(bp)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
for i := len(params) - 1; i >= 0; i-- {
|
for i := len(params) - 1; i >= 0; i-- {
|
||||||
v.Estack().PushVal(params[i])
|
v.Estack().PushVal(params[i])
|
||||||
}
|
}
|
||||||
if runCurrent {
|
|
||||||
if !v.Ready() {
|
|
||||||
return errors.New("no program loaded")
|
|
||||||
}
|
|
||||||
v.Context().Jump(offset)
|
|
||||||
if initMD := m.ABI.GetMethod(manifest.MethodInit, 0); initMD != nil {
|
|
||||||
v.Call(initMD.Offset)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
runVMWithHandling(c)
|
runVMWithHandling(c)
|
||||||
changePrompt(c.App)
|
changePrompt(c.App)
|
||||||
|
@ -1085,7 +1185,7 @@ func handleStep(c *cli.Context) error {
|
||||||
if len(args) > 0 {
|
if len(args) > 0 {
|
||||||
n, err = strconv.Atoi(args[0])
|
n, err = strconv.Atoi(args[0])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("%w: %s", ErrInvalidParameter, err) //nolint:errorlint // errorlint: non-wrapping format verb for fmt.Errorf. Use `%w` to format errors
|
return fmt.Errorf("%w: %w", ErrInvalidParameter, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
v.AddBreakPointRel(n)
|
v.AddBreakPointRel(n)
|
||||||
|
@ -1336,58 +1436,57 @@ func Parse(args []string) (string, error) {
|
||||||
return "", ErrMissingParameter
|
return "", ErrMissingParameter
|
||||||
}
|
}
|
||||||
arg := args[0]
|
arg := args[0]
|
||||||
buf := bytes.NewBuffer(nil)
|
var buf []byte
|
||||||
if val, err := strconv.ParseInt(arg, 10, 64); err == nil {
|
if val, err := strconv.ParseInt(arg, 10, 64); err == nil {
|
||||||
bs := bigint.ToBytes(big.NewInt(val))
|
bs := bigint.ToBytes(big.NewInt(val))
|
||||||
buf.WriteString(fmt.Sprintf("Integer to Hex\t%s\n", hex.EncodeToString(bs)))
|
buf = fmt.Appendf(buf, "Integer to Hex\t%s\n", hex.EncodeToString(bs))
|
||||||
buf.WriteString(fmt.Sprintf("Integer to Base64\t%s\n", base64.StdEncoding.EncodeToString(bs)))
|
buf = fmt.Appendf(buf, "Integer to Base64\t%s\n", base64.StdEncoding.EncodeToString(bs))
|
||||||
}
|
}
|
||||||
noX := strings.TrimPrefix(arg, "0x")
|
noX := strings.TrimPrefix(arg, "0x")
|
||||||
if rawStr, err := hex.DecodeString(noX); err == nil {
|
if rawStr, err := hex.DecodeString(noX); err == nil {
|
||||||
if val, err := util.Uint160DecodeBytesBE(rawStr); err == nil {
|
if val, err := util.Uint160DecodeBytesBE(rawStr); err == nil {
|
||||||
buf.WriteString(fmt.Sprintf("BE ScriptHash to Address\t%s\n", address.Uint160ToString(val)))
|
buf = fmt.Appendf(buf, "BE ScriptHash to Address\t%s\n", address.Uint160ToString(val))
|
||||||
buf.WriteString(fmt.Sprintf("LE ScriptHash to Address\t%s\n", address.Uint160ToString(val.Reverse())))
|
buf = fmt.Appendf(buf, "LE ScriptHash to Address\t%s\n", address.Uint160ToString(val.Reverse()))
|
||||||
}
|
}
|
||||||
if pub, err := keys.NewPublicKeyFromBytes(rawStr, elliptic.P256()); err == nil {
|
if pub, err := keys.NewPublicKeyFromBytes(rawStr, elliptic.P256()); err == nil {
|
||||||
sh := pub.GetScriptHash()
|
sh := pub.GetScriptHash()
|
||||||
buf.WriteString(fmt.Sprintf("Public key to BE ScriptHash\t%s\n", sh))
|
buf = fmt.Appendf(buf, "Public key to BE ScriptHash\t%s\n", sh)
|
||||||
buf.WriteString(fmt.Sprintf("Public key to LE ScriptHash\t%s\n", sh.Reverse()))
|
buf = fmt.Appendf(buf, "Public key to LE ScriptHash\t%s\n", sh.Reverse())
|
||||||
buf.WriteString(fmt.Sprintf("Public key to Address\t%s\n", address.Uint160ToString(sh)))
|
buf = fmt.Appendf(buf, "Public key to Address\t%s\n", address.Uint160ToString(sh))
|
||||||
}
|
}
|
||||||
buf.WriteString(fmt.Sprintf("Hex to String\t%s\n", fmt.Sprintf("%q", string(rawStr))))
|
buf = fmt.Appendf(buf, "Hex to String\t%s\n", fmt.Sprintf("%q", string(rawStr)))
|
||||||
buf.WriteString(fmt.Sprintf("Hex to Integer\t%s\n", bigint.FromBytes(rawStr)))
|
buf = fmt.Appendf(buf, "Hex to Integer\t%s\n", bigint.FromBytes(rawStr))
|
||||||
buf.WriteString(fmt.Sprintf("Swap Endianness\t%s\n", hex.EncodeToString(slice.CopyReverse(rawStr))))
|
buf = fmt.Appendf(buf, "Swap Endianness\t%s\n", hex.EncodeToString(slice.CopyReverse(rawStr)))
|
||||||
}
|
}
|
||||||
if addr, err := address.StringToUint160(arg); err == nil {
|
if addr, err := address.StringToUint160(arg); err == nil {
|
||||||
buf.WriteString(fmt.Sprintf("Address to BE ScriptHash\t%s\n", addr))
|
buf = fmt.Appendf(buf, "Address to BE ScriptHash\t%s\n", addr)
|
||||||
buf.WriteString(fmt.Sprintf("Address to LE ScriptHash\t%s\n", addr.Reverse()))
|
buf = fmt.Appendf(buf, "Address to LE ScriptHash\t%s\n", addr.Reverse())
|
||||||
buf.WriteString(fmt.Sprintf("Address to Base64 (BE)\t%s\n", base64.StdEncoding.EncodeToString(addr.BytesBE())))
|
buf = fmt.Appendf(buf, "Address to Base64 (BE)\t%s\n", base64.StdEncoding.EncodeToString(addr.BytesBE()))
|
||||||
buf.WriteString(fmt.Sprintf("Address to Base64 (LE)\t%s\n", base64.StdEncoding.EncodeToString(addr.BytesLE())))
|
buf = fmt.Appendf(buf, "Address to Base64 (LE)\t%s\n", base64.StdEncoding.EncodeToString(addr.BytesLE()))
|
||||||
}
|
}
|
||||||
if rawStr, err := base64.StdEncoding.DecodeString(arg); err == nil {
|
if rawStr, err := base64.StdEncoding.DecodeString(arg); err == nil {
|
||||||
buf.WriteString(fmt.Sprintf("Base64 to String\t%s\n", fmt.Sprintf("%q", string(rawStr))))
|
buf = fmt.Appendf(buf, "Base64 to String\t%s\n", fmt.Sprintf("%q", string(rawStr)))
|
||||||
buf.WriteString(fmt.Sprintf("Base64 to BigInteger\t%s\n", bigint.FromBytes(rawStr)))
|
buf = fmt.Appendf(buf, "Base64 to BigInteger\t%s\n", bigint.FromBytes(rawStr))
|
||||||
if u, err := util.Uint160DecodeBytesBE(rawStr); err == nil {
|
if u, err := util.Uint160DecodeBytesBE(rawStr); err == nil {
|
||||||
buf.WriteString(fmt.Sprintf("Base64 to BE ScriptHash\t%s\n", u.StringBE()))
|
buf = fmt.Appendf(buf, "Base64 to BE ScriptHash\t%s\n", u.StringBE())
|
||||||
buf.WriteString(fmt.Sprintf("Base64 to LE ScriptHash\t%s\n", u.StringLE()))
|
buf = fmt.Appendf(buf, "Base64 to LE ScriptHash\t%s\n", u.StringLE())
|
||||||
buf.WriteString(fmt.Sprintf("Base64 to Address (BE)\t%s\n", address.Uint160ToString(u)))
|
buf = fmt.Appendf(buf, "Base64 to Address (BE)\t%s\n", address.Uint160ToString(u))
|
||||||
buf.WriteString(fmt.Sprintf("Base64 to Address (LE)\t%s\n", address.Uint160ToString(u.Reverse())))
|
buf = fmt.Appendf(buf, "Base64 to Address (LE)\t%s\n", address.Uint160ToString(u.Reverse()))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
buf.WriteString(fmt.Sprintf("String to Hex\t%s\n", hex.EncodeToString([]byte(arg))))
|
buf = fmt.Appendf(buf, "String to Hex\t%s\n", hex.EncodeToString([]byte(arg)))
|
||||||
buf.WriteString(fmt.Sprintf("String to Base64\t%s\n", base64.StdEncoding.EncodeToString([]byte(arg))))
|
buf = fmt.Appendf(buf, "String to Base64\t%s\n", base64.StdEncoding.EncodeToString([]byte(arg)))
|
||||||
|
|
||||||
out := buf.Bytes()
|
res := bytes.NewBuffer(nil)
|
||||||
buf = bytes.NewBuffer(nil)
|
w := tabwriter.NewWriter(res, 0, 4, 4, '\t', 0)
|
||||||
w := tabwriter.NewWriter(buf, 0, 4, 4, '\t', 0)
|
if _, err := w.Write(buf); err != nil {
|
||||||
if _, err := w.Write(out); err != nil {
|
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
if err := w.Flush(); err != nil {
|
if err := w.Flush(); err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
return buf.String(), nil
|
return res.String(), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
const logo = `
|
const logo = `
|
||||||
|
|
|
@ -14,13 +14,16 @@ import (
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
|
"sync/atomic"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/chzyer/readline"
|
"github.com/chzyer/readline"
|
||||||
|
"github.com/nspcc-dev/neo-go/cli/cmdargs"
|
||||||
"github.com/nspcc-dev/neo-go/cli/paramcontext"
|
"github.com/nspcc-dev/neo-go/cli/paramcontext"
|
||||||
"github.com/nspcc-dev/neo-go/internal/basicchain"
|
"github.com/nspcc-dev/neo-go/internal/basicchain"
|
||||||
"github.com/nspcc-dev/neo-go/internal/random"
|
"github.com/nspcc-dev/neo-go/internal/random"
|
||||||
|
"github.com/nspcc-dev/neo-go/internal/versionutil"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/compiler"
|
"github.com/nspcc-dev/neo-go/pkg/compiler"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/config"
|
"github.com/nspcc-dev/neo-go/pkg/config"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/config/netmode"
|
"github.com/nspcc-dev/neo-go/pkg/config/netmode"
|
||||||
|
@ -40,9 +43,11 @@ import (
|
||||||
"github.com/nspcc-dev/neo-go/pkg/vm/opcode"
|
"github.com/nspcc-dev/neo-go/pkg/vm/opcode"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
"go.uber.org/atomic"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// Keep contract NEFs consistent between runs.
|
||||||
|
const _ = versionutil.TestVersion
|
||||||
|
|
||||||
type readCloser struct {
|
type readCloser struct {
|
||||||
sync.Mutex
|
sync.Mutex
|
||||||
bytes.Buffer
|
bytes.Buffer
|
||||||
|
@ -88,10 +93,11 @@ func newTestVMCLIWithLogoAndCustomConfig(t *testing.T, printLogo bool, cfg *conf
|
||||||
}
|
}
|
||||||
var c config.Config
|
var c config.Config
|
||||||
if cfg == nil {
|
if cfg == nil {
|
||||||
configPath := "../../config/protocol.unit_testnet.single.yml"
|
configPath := filepath.Join("..", "..", "config", "protocol.unit_testnet.single.yml")
|
||||||
var err error
|
var err error
|
||||||
c, err = config.LoadFile(configPath)
|
c, err = config.LoadFile(configPath, filepath.Join("..", "..", "config"))
|
||||||
require.NoError(t, err, "could not load chain config")
|
require.NoError(t, err, "could not load chain config")
|
||||||
|
require.Equal(t, filepath.Join("..", "..", "testdata", "wallet1_solo.json"), c.ApplicationConfiguration.Consensus.UnlockWallet.Path)
|
||||||
c.ApplicationConfiguration.DBConfiguration.Type = dbconfig.InMemoryDB
|
c.ApplicationConfiguration.DBConfiguration.Type = dbconfig.InMemoryDB
|
||||||
} else {
|
} else {
|
||||||
c = *cfg
|
c = *cfg
|
||||||
|
@ -128,6 +134,10 @@ func newTestVMClIWithState(t *testing.T) *executor {
|
||||||
}
|
}
|
||||||
bc, validators, committee, err := chain.NewMultiWithCustomConfigAndStoreNoCheck(t, customConfig, store)
|
bc, validators, committee, err := chain.NewMultiWithCustomConfigAndStoreNoCheck(t, customConfig, store)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
// Save config for future usage.
|
||||||
|
protoCfg := bc.GetConfig()
|
||||||
|
|
||||||
go bc.Run()
|
go bc.Run()
|
||||||
e := neotest.NewExecutor(t, bc, validators, committee)
|
e := neotest.NewExecutor(t, bc, validators, committee)
|
||||||
basicchain.InitSimple(t, "../../", e)
|
basicchain.InitSimple(t, "../../", e)
|
||||||
|
@ -139,7 +149,9 @@ func newTestVMClIWithState(t *testing.T) *executor {
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
cfg.ApplicationConfiguration.DBConfiguration.Type = dbconfig.LevelDB
|
cfg.ApplicationConfiguration.DBConfiguration.Type = dbconfig.LevelDB
|
||||||
cfg.ApplicationConfiguration.DBConfiguration.LevelDBOptions = opts
|
cfg.ApplicationConfiguration.DBConfiguration.LevelDBOptions = opts
|
||||||
cfg.ProtocolConfiguration.StateRootInHeader = true
|
cfg.ProtocolConfiguration.StateRootInHeader = protoCfg.StateRootInHeader
|
||||||
|
cfg.ProtocolConfiguration.P2PStateExchangeExtensions = protoCfg.P2PStateExchangeExtensions
|
||||||
|
cfg.ProtocolConfiguration.Hardforks = protoCfg.Hardforks
|
||||||
return newTestVMCLIWithLogoAndCustomConfig(t, false, &cfg)
|
return newTestVMCLIWithLogoAndCustomConfig(t, false, &cfg)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -180,9 +192,9 @@ func (e *executor) checkError(t *testing.T, expectedErr error) {
|
||||||
require.True(t, strings.HasPrefix(line, expected), fmt.Errorf("expected `%s`, got `%s`", expected, line))
|
require.True(t, strings.HasPrefix(line, expected), fmt.Errorf("expected `%s`, got `%s`", expected, line))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *executor) checkStack(t *testing.T, items ...interface{}) {
|
func (e *executor) checkStack(t *testing.T, items ...any) {
|
||||||
d := json.NewDecoder(e.out)
|
d := json.NewDecoder(e.out)
|
||||||
var actual interface{}
|
var actual any
|
||||||
require.NoError(t, d.Decode(&actual))
|
require.NoError(t, d.Decode(&actual))
|
||||||
rawActual, err := json.Marshal(actual)
|
rawActual, err := json.Marshal(actual)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
@ -210,7 +222,7 @@ func (e *executor) checkEvents(t *testing.T, isKeywordExpected bool, events ...s
|
||||||
e.checkNextLine(t, "Events:")
|
e.checkNextLine(t, "Events:")
|
||||||
}
|
}
|
||||||
d := json.NewDecoder(e.out)
|
d := json.NewDecoder(e.out)
|
||||||
var actual interface{}
|
var actual any
|
||||||
require.NoError(t, d.Decode(&actual))
|
require.NoError(t, d.Decode(&actual))
|
||||||
rawActual, err := json.Marshal(actual)
|
rawActual, err := json.Marshal(actual)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
@ -249,9 +261,9 @@ func (e *executor) checkChange(t *testing.T, c storageChange) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *executor) checkSlot(t *testing.T, items ...interface{}) {
|
func (e *executor) checkSlot(t *testing.T, items ...any) {
|
||||||
d := json.NewDecoder(e.out)
|
d := json.NewDecoder(e.out)
|
||||||
var actual interface{}
|
var actual any
|
||||||
require.NoError(t, d.Decode(&actual))
|
require.NoError(t, d.Decode(&actual))
|
||||||
rawActual, err := json.Marshal(actual)
|
rawActual, err := json.Marshal(actual)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
@ -280,6 +292,95 @@ func (e *executor) checkSlot(t *testing.T, items ...interface{}) {
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestRun_WithNewVMContextAndBreakpoints(t *testing.T) {
|
||||||
|
t.Run("contract without init", func(t *testing.T) {
|
||||||
|
src := `package kek
|
||||||
|
func Main(a, b int) int {
|
||||||
|
var c = a + b
|
||||||
|
return c + 5
|
||||||
|
}`
|
||||||
|
tmpDir := t.TempDir()
|
||||||
|
filename := prepareLoadgoSrc(t, tmpDir, src)
|
||||||
|
|
||||||
|
e := newTestVMCLI(t)
|
||||||
|
e.runProgWithTimeout(t, 10*time.Second,
|
||||||
|
"loadgo "+filename,
|
||||||
|
"break 8",
|
||||||
|
"run main 3 5",
|
||||||
|
"run",
|
||||||
|
)
|
||||||
|
|
||||||
|
e.checkNextLine(t, "READY: loaded \\d* instructions")
|
||||||
|
e.checkNextLine(t, "breakpoint added at instruction 8")
|
||||||
|
e.checkNextLine(t, "at breakpoint 8 (PUSH5)*")
|
||||||
|
e.checkStack(t, 13)
|
||||||
|
})
|
||||||
|
t.Run("contract with init", func(t *testing.T) {
|
||||||
|
src := `package kek
|
||||||
|
var I = 5
|
||||||
|
func Main(a, b int) int {
|
||||||
|
var c = a + b
|
||||||
|
return c + I
|
||||||
|
}`
|
||||||
|
|
||||||
|
tmpDir := t.TempDir()
|
||||||
|
filename := prepareLoadgoSrc(t, tmpDir, src)
|
||||||
|
|
||||||
|
e := newTestVMCLI(t)
|
||||||
|
e.runProgWithTimeout(t, 10*time.Second,
|
||||||
|
"loadgo "+filename,
|
||||||
|
"break 10",
|
||||||
|
"run main 3 5",
|
||||||
|
"run",
|
||||||
|
)
|
||||||
|
|
||||||
|
e.checkNextLine(t, "READY: loaded \\d* instructions")
|
||||||
|
e.checkNextLine(t, "breakpoint added at instruction 10")
|
||||||
|
e.checkNextLine(t, "at breakpoint 10 (ADD)*")
|
||||||
|
e.checkStack(t, 13)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// prepareLoadgoSrc prepares provided SC source file for loading into VM via `loadgo` command.
|
||||||
|
func prepareLoadgoSrc(t *testing.T, tmpDir, src string) string {
|
||||||
|
filename := filepath.Join(tmpDir, "vmtestcontract.go")
|
||||||
|
require.NoError(t, os.WriteFile(filename, []byte(src), os.ModePerm))
|
||||||
|
filename = "'" + filename + "'"
|
||||||
|
wd, err := os.Getwd()
|
||||||
|
require.NoError(t, err)
|
||||||
|
goMod := []byte(`module test.example/kek
|
||||||
|
require (
|
||||||
|
github.com/nspcc-dev/neo-go/pkg/interop v0.0.0
|
||||||
|
)
|
||||||
|
replace github.com/nspcc-dev/neo-go/pkg/interop => ` + filepath.Join(wd, "../../pkg/interop") + `
|
||||||
|
go 1.20`)
|
||||||
|
require.NoError(t, os.WriteFile(filepath.Join(tmpDir, "go.mod"), goMod, os.ModePerm))
|
||||||
|
return filename
|
||||||
|
}
|
||||||
|
|
||||||
|
// prepareLoadnefSrc compiles provided SC source and prepares NEF and manifest for loading into VM
|
||||||
|
// via `loadnef` command. It returns the name of manifest and NEF files ready to be used in CLI
|
||||||
|
// commands.
|
||||||
|
func prepareLoadnefSrc(t *testing.T, tmpDir, src string) (string, string) {
|
||||||
|
nefFile, di, err := compiler.CompileWithOptions("test.go", strings.NewReader(src), nil)
|
||||||
|
require.NoError(t, err)
|
||||||
|
filename := filepath.Join(tmpDir, "vmtestcontract.nef")
|
||||||
|
rawNef, err := nefFile.Bytes()
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.NoError(t, os.WriteFile(filename, rawNef, os.ModePerm))
|
||||||
|
m, err := di.ConvertToManifest(&compiler.Options{})
|
||||||
|
require.NoError(t, err)
|
||||||
|
manifestFile := filepath.Join(tmpDir, "vmtestcontract.manifest.json")
|
||||||
|
rawManifest, err := json.Marshal(m)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.NoError(t, os.WriteFile(manifestFile, rawManifest, os.ModePerm))
|
||||||
|
|
||||||
|
manifestFile = "'" + manifestFile + "'"
|
||||||
|
filename = "'" + filename + "'"
|
||||||
|
|
||||||
|
return manifestFile, filename
|
||||||
|
}
|
||||||
|
|
||||||
func TestLoad(t *testing.T) {
|
func TestLoad(t *testing.T) {
|
||||||
script := []byte{byte(opcode.PUSH3), byte(opcode.PUSH4), byte(opcode.ADD)}
|
script := []byte{byte(opcode.PUSH3), byte(opcode.PUSH4), byte(opcode.ADD)}
|
||||||
|
|
||||||
|
@ -298,21 +399,27 @@ func TestLoad(t *testing.T) {
|
||||||
"loadhex",
|
"loadhex",
|
||||||
"loadhex notahex",
|
"loadhex notahex",
|
||||||
"loadhex "+hex.EncodeToString(script),
|
"loadhex "+hex.EncodeToString(script),
|
||||||
"loadhex "+hex.EncodeToString(checkWitnessScript)+" "+ownerAddress, // owner:DefaultScope => true
|
"loadhex "+hex.EncodeToString(checkWitnessScript)+" "+cmdargs.CosignersSeparator,
|
||||||
|
"loadhex "+hex.EncodeToString(checkWitnessScript)+" "+"not-a-separator",
|
||||||
|
"loadhex "+hex.EncodeToString(checkWitnessScript)+" "+cmdargs.CosignersSeparator+" "+"not-a-signer",
|
||||||
|
"loadhex "+hex.EncodeToString(checkWitnessScript)+" "+cmdargs.CosignersSeparator+" "+ownerAddress, // owner:DefaultScope => true
|
||||||
"run",
|
"run",
|
||||||
"loadhex "+hex.EncodeToString(checkWitnessScript)+" "+ownerAddress+":None", // owner:None => false
|
"loadhex "+hex.EncodeToString(checkWitnessScript)+" "+cmdargs.CosignersSeparator+" "+ownerAddress+":None", // owner:None => false
|
||||||
"run",
|
"run",
|
||||||
"loadhex "+hex.EncodeToString(checkWitnessScript)+" "+ownerAcc.StringLE(), // ownerLE:DefaultScope => true
|
"loadhex "+hex.EncodeToString(checkWitnessScript)+" "+cmdargs.CosignersSeparator+" "+ownerAcc.StringLE(), // ownerLE:DefaultScope => true
|
||||||
"run",
|
"run",
|
||||||
"loadhex "+hex.EncodeToString(checkWitnessScript)+" 0x"+ownerAcc.StringLE(), // owner0xLE:DefaultScope => true
|
"loadhex "+hex.EncodeToString(checkWitnessScript)+" "+cmdargs.CosignersSeparator+" "+"0x"+ownerAcc.StringLE(), // owner0xLE:DefaultScope => true
|
||||||
"run",
|
"run",
|
||||||
"loadhex "+hex.EncodeToString(checkWitnessScript)+" "+sideAcc.StringLE(), // sideLE:DefaultScope => false
|
"loadhex "+hex.EncodeToString(checkWitnessScript)+" "+cmdargs.CosignersSeparator+" "+sideAcc.StringLE(), // sideLE:DefaultScope => false
|
||||||
"run",
|
"run",
|
||||||
)
|
)
|
||||||
|
|
||||||
e.checkError(t, ErrMissingParameter)
|
e.checkError(t, ErrMissingParameter)
|
||||||
e.checkError(t, ErrInvalidParameter)
|
e.checkError(t, ErrInvalidParameter)
|
||||||
e.checkNextLine(t, "READY: loaded 3 instructions")
|
e.checkNextLine(t, "READY: loaded 3 instructions")
|
||||||
|
e.checkError(t, ErrInvalidParameter)
|
||||||
|
e.checkError(t, ErrInvalidParameter)
|
||||||
|
e.checkError(t, ErrInvalidParameter)
|
||||||
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
||||||
e.checkStack(t, true)
|
e.checkStack(t, true)
|
||||||
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
||||||
|
@ -330,21 +437,27 @@ func TestLoad(t *testing.T) {
|
||||||
"loadbase64",
|
"loadbase64",
|
||||||
"loadbase64 not_a_base64",
|
"loadbase64 not_a_base64",
|
||||||
"loadbase64 "+base64.StdEncoding.EncodeToString(script),
|
"loadbase64 "+base64.StdEncoding.EncodeToString(script),
|
||||||
"loadbase64 "+base64.StdEncoding.EncodeToString(checkWitnessScript)+" "+ownerAddress, // owner:DefaultScope => true
|
"loadbase64 "+base64.StdEncoding.EncodeToString(checkWitnessScript)+" "+cmdargs.CosignersSeparator,
|
||||||
|
"loadbase64 "+base64.StdEncoding.EncodeToString(checkWitnessScript)+" "+"not-a-separator",
|
||||||
|
"loadbase64 "+base64.StdEncoding.EncodeToString(checkWitnessScript)+" "+cmdargs.CosignersSeparator+" not-a-signer",
|
||||||
|
"loadbase64 "+base64.StdEncoding.EncodeToString(checkWitnessScript)+" "+cmdargs.CosignersSeparator+" "+ownerAddress, // owner:DefaultScope => true
|
||||||
"run",
|
"run",
|
||||||
"loadbase64 "+base64.StdEncoding.EncodeToString(checkWitnessScript)+" "+ownerAddress+":None", // owner:None => false
|
"loadbase64 "+base64.StdEncoding.EncodeToString(checkWitnessScript)+" "+cmdargs.CosignersSeparator+" "+ownerAddress+":None", // owner:None => false
|
||||||
"run",
|
"run",
|
||||||
"loadbase64 "+base64.StdEncoding.EncodeToString(checkWitnessScript)+" "+ownerAcc.StringLE(), // ownerLE:DefaultScope => true
|
"loadbase64 "+base64.StdEncoding.EncodeToString(checkWitnessScript)+" "+cmdargs.CosignersSeparator+" "+ownerAcc.StringLE(), // ownerLE:DefaultScope => true
|
||||||
"run",
|
"run",
|
||||||
"loadbase64 "+base64.StdEncoding.EncodeToString(checkWitnessScript)+" 0x"+ownerAcc.StringLE(), // owner0xLE:DefaultScope => true
|
"loadbase64 "+base64.StdEncoding.EncodeToString(checkWitnessScript)+" "+cmdargs.CosignersSeparator+" "+"0x"+ownerAcc.StringLE(), // owner0xLE:DefaultScope => true
|
||||||
"run",
|
"run",
|
||||||
"loadbase64 "+base64.StdEncoding.EncodeToString(checkWitnessScript)+" "+sideAcc.StringLE(), // sideLE:DefaultScope => false
|
"loadbase64 "+base64.StdEncoding.EncodeToString(checkWitnessScript)+" "+cmdargs.CosignersSeparator+" "+sideAcc.StringLE(), // sideLE:DefaultScope => false
|
||||||
"run",
|
"run",
|
||||||
)
|
)
|
||||||
|
|
||||||
e.checkError(t, ErrMissingParameter)
|
e.checkError(t, ErrMissingParameter)
|
||||||
e.checkError(t, ErrInvalidParameter)
|
e.checkError(t, ErrInvalidParameter)
|
||||||
e.checkNextLine(t, "READY: loaded 3 instructions")
|
e.checkNextLine(t, "READY: loaded 3 instructions")
|
||||||
|
e.checkError(t, ErrInvalidParameter)
|
||||||
|
e.checkError(t, ErrInvalidParameter)
|
||||||
|
e.checkError(t, ErrInvalidParameter)
|
||||||
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
||||||
e.checkStack(t, true)
|
e.checkStack(t, true)
|
||||||
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
||||||
|
@ -365,19 +478,13 @@ func TestLoad(t *testing.T) {
|
||||||
return a * b
|
return a * b
|
||||||
}
|
}
|
||||||
}`
|
}`
|
||||||
|
|
||||||
|
t.Run("loadgo", func(t *testing.T) {
|
||||||
tmpDir := t.TempDir()
|
tmpDir := t.TempDir()
|
||||||
|
|
||||||
checkLoadgo := func(t *testing.T, tName, cName, cErrName string) {
|
checkLoadgo := func(t *testing.T, cName, cErrName string) {
|
||||||
t.Run("loadgo "+tName, func(t *testing.T) {
|
filename := prepareLoadgoSrc(t, tmpDir, src)
|
||||||
filename := filepath.Join(tmpDir, cName)
|
|
||||||
require.NoError(t, os.WriteFile(filename, []byte(src), os.ModePerm))
|
|
||||||
filename = "'" + filename + "'"
|
|
||||||
filenameErr := filepath.Join(tmpDir, cErrName)
|
filenameErr := filepath.Join(tmpDir, cErrName)
|
||||||
require.NoError(t, os.WriteFile(filenameErr, []byte(src+"invalid_token"), os.ModePerm))
|
|
||||||
filenameErr = "'" + filenameErr + "'"
|
|
||||||
goMod := []byte(`module test.example/vmcli
|
|
||||||
go 1.17`)
|
|
||||||
require.NoError(t, os.WriteFile(filepath.Join(tmpDir, "go.mod"), goMod, os.ModePerm))
|
|
||||||
|
|
||||||
e := newTestVMCLI(t)
|
e := newTestVMCLI(t)
|
||||||
e.runProgWithTimeout(t, 10*time.Second,
|
e.runProgWithTimeout(t, 10*time.Second,
|
||||||
|
@ -390,28 +497,16 @@ go 1.17`)
|
||||||
e.checkNextLine(t, "Error:")
|
e.checkNextLine(t, "Error:")
|
||||||
e.checkNextLine(t, "READY: loaded \\d* instructions")
|
e.checkNextLine(t, "READY: loaded \\d* instructions")
|
||||||
e.checkStack(t, 8)
|
e.checkStack(t, 8)
|
||||||
|
}
|
||||||
|
|
||||||
|
t.Run("simple", func(t *testing.T) {
|
||||||
|
checkLoadgo(t, "vmtestcontract.go", "vmtestcontract_err.go")
|
||||||
|
})
|
||||||
|
t.Run("utf-8 with spaces", func(t *testing.T) {
|
||||||
|
checkLoadgo(t, "тестовый контракт.go", "тестовый контракт с ошибкой.go")
|
||||||
})
|
})
|
||||||
}
|
|
||||||
|
|
||||||
checkLoadgo(t, "simple", "vmtestcontract.go", "vmtestcontract_err.go")
|
t.Run("check calling flags", func(t *testing.T) {
|
||||||
checkLoadgo(t, "utf-8 with spaces", "тестовый контракт.go", "тестовый контракт с ошибкой.go")
|
|
||||||
|
|
||||||
prepareLoadgoSrc := func(t *testing.T, srcAllowNotify string) string {
|
|
||||||
filename := filepath.Join(tmpDir, "vmtestcontract.go")
|
|
||||||
require.NoError(t, os.WriteFile(filename, []byte(srcAllowNotify), os.ModePerm))
|
|
||||||
filename = "'" + filename + "'"
|
|
||||||
wd, err := os.Getwd()
|
|
||||||
require.NoError(t, err)
|
|
||||||
goMod := []byte(`module test.example/kek
|
|
||||||
require (
|
|
||||||
github.com/nspcc-dev/neo-go/pkg/interop v0.0.0
|
|
||||||
)
|
|
||||||
replace github.com/nspcc-dev/neo-go/pkg/interop => ` + filepath.Join(wd, "../../pkg/interop") + `
|
|
||||||
go 1.17`)
|
|
||||||
require.NoError(t, os.WriteFile(filepath.Join(tmpDir, "go.mod"), goMod, os.ModePerm))
|
|
||||||
return filename
|
|
||||||
}
|
|
||||||
t.Run("loadgo, check calling flags", func(t *testing.T) {
|
|
||||||
srcAllowNotify := `package kek
|
srcAllowNotify := `package kek
|
||||||
import "github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
import "github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
||||||
func Main() int {
|
func Main() int {
|
||||||
|
@ -419,7 +514,7 @@ go 1.17`)
|
||||||
return 1
|
return 1
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
filename := prepareLoadgoSrc(t, srcAllowNotify)
|
filename := prepareLoadgoSrc(t, tmpDir, srcAllowNotify)
|
||||||
|
|
||||||
e := newTestVMCLI(t)
|
e := newTestVMCLI(t)
|
||||||
e.runProg(t,
|
e.runProg(t,
|
||||||
|
@ -428,24 +523,35 @@ go 1.17`)
|
||||||
e.checkNextLine(t, "READY: loaded \\d* instructions")
|
e.checkNextLine(t, "READY: loaded \\d* instructions")
|
||||||
e.checkStack(t, 1)
|
e.checkStack(t, 1)
|
||||||
})
|
})
|
||||||
t.Run("loadgo, check signers", func(t *testing.T) {
|
t.Run("check signers", func(t *testing.T) {
|
||||||
srcCheckWitness := `package kek
|
srcCheckWitness := `package kek
|
||||||
import (
|
import (
|
||||||
"github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
"github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/interop/util"
|
"github.com/nspcc-dev/neo-go/pkg/interop/lib/address"
|
||||||
)
|
)
|
||||||
func Main() bool {
|
func Main() bool {
|
||||||
var owner = util.FromAddress("` + ownerAddress + `")
|
var owner = address.ToHash160("` + ownerAddress + `")
|
||||||
return runtime.CheckWitness(owner)
|
return runtime.CheckWitness(owner)
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
filename := prepareLoadgoSrc(t, srcCheckWitness)
|
filename := prepareLoadgoSrc(t, tmpDir, srcCheckWitness)
|
||||||
|
t.Run("invalid", func(t *testing.T) {
|
||||||
|
e := newTestVMCLI(t)
|
||||||
|
e.runProg(t,
|
||||||
|
"loadgo "+filename+" "+cmdargs.CosignersSeparator,
|
||||||
|
"loadgo "+filename+" "+"not-a-separator",
|
||||||
|
"loadgo "+filename+" "+cmdargs.CosignersSeparator+" not-a-signer",
|
||||||
|
)
|
||||||
|
e.checkError(t, ErrInvalidParameter)
|
||||||
|
e.checkError(t, ErrInvalidParameter)
|
||||||
|
e.checkError(t, ErrInvalidParameter)
|
||||||
|
})
|
||||||
t.Run("address", func(t *testing.T) {
|
t.Run("address", func(t *testing.T) {
|
||||||
e := newTestVMCLI(t)
|
e := newTestVMCLI(t)
|
||||||
e.runProg(t,
|
e.runProg(t,
|
||||||
"loadgo "+filename+" "+ownerAddress, // owner:DefaultScope => true
|
"loadgo "+filename+" "+cmdargs.CosignersSeparator+" "+ownerAddress, // owner:DefaultScope => true
|
||||||
"run main",
|
"run main",
|
||||||
"loadgo "+filename+" "+ownerAddress+":None", // owner:None => false
|
"loadgo "+filename+" "+cmdargs.CosignersSeparator+" "+ownerAddress+":None", // owner:None => false
|
||||||
"run main")
|
"run main")
|
||||||
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
||||||
e.checkStack(t, true)
|
e.checkStack(t, true)
|
||||||
|
@ -455,9 +561,9 @@ go 1.17`)
|
||||||
t.Run("string LE", func(t *testing.T) {
|
t.Run("string LE", func(t *testing.T) {
|
||||||
e := newTestVMCLI(t)
|
e := newTestVMCLI(t)
|
||||||
e.runProg(t,
|
e.runProg(t,
|
||||||
"loadgo "+filename+" "+ownerAcc.StringLE(), // ownerLE:DefaultScope => true
|
"loadgo "+filename+" "+cmdargs.CosignersSeparator+" "+ownerAcc.StringLE(), // ownerLE:DefaultScope => true
|
||||||
"run main",
|
"run main",
|
||||||
"loadgo "+filename+" 0x"+ownerAcc.StringLE(), // owner0xLE:DefaultScope => true
|
"loadgo "+filename+" "+cmdargs.CosignersSeparator+" "+"0x"+ownerAcc.StringLE(), // owner0xLE:DefaultScope => true
|
||||||
"run main")
|
"run main")
|
||||||
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
||||||
e.checkStack(t, true)
|
e.checkStack(t, true)
|
||||||
|
@ -467,43 +573,41 @@ go 1.17`)
|
||||||
t.Run("nonwitnessed signer", func(t *testing.T) {
|
t.Run("nonwitnessed signer", func(t *testing.T) {
|
||||||
e := newTestVMCLI(t)
|
e := newTestVMCLI(t)
|
||||||
e.runProg(t,
|
e.runProg(t,
|
||||||
"loadgo "+filename+" "+sideAcc.StringLE(), // sideLE:DefaultScope => false
|
"loadgo "+filename+" "+cmdargs.CosignersSeparator+" "+sideAcc.StringLE(), // sideLE:DefaultScope => false
|
||||||
"run main")
|
"run main")
|
||||||
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
||||||
e.checkStack(t, false)
|
e.checkStack(t, false)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
})
|
||||||
|
|
||||||
t.Run("loadnef", func(t *testing.T) {
|
t.Run("loadnef", func(t *testing.T) {
|
||||||
config.Version = "0.92.0-test"
|
tmpDir := t.TempDir()
|
||||||
|
|
||||||
nefFile, di, err := compiler.CompileWithOptions("test.go", strings.NewReader(src), nil)
|
manifestFile, nefFile := prepareLoadnefSrc(t, tmpDir, src)
|
||||||
require.NoError(t, err)
|
|
||||||
filename := filepath.Join(tmpDir, "vmtestcontract.nef")
|
|
||||||
rawNef, err := nefFile.Bytes()
|
|
||||||
require.NoError(t, err)
|
|
||||||
require.NoError(t, os.WriteFile(filename, rawNef, os.ModePerm))
|
|
||||||
m, err := di.ConvertToManifest(&compiler.Options{})
|
|
||||||
require.NoError(t, err)
|
|
||||||
manifestFile := filepath.Join(tmpDir, "vmtestcontract.manifest.json")
|
|
||||||
rawManifest, err := json.Marshal(m)
|
|
||||||
require.NoError(t, err)
|
|
||||||
require.NoError(t, os.WriteFile(manifestFile, rawManifest, os.ModePerm))
|
|
||||||
filenameErr := filepath.Join(tmpDir, "vmtestcontract_err.nef")
|
filenameErr := filepath.Join(tmpDir, "vmtestcontract_err.nef")
|
||||||
require.NoError(t, os.WriteFile(filenameErr, append([]byte{1, 2, 3, 4}, rawNef...), os.ModePerm))
|
require.NoError(t, os.WriteFile(filenameErr, []byte{1, 2, 3, 4}, os.ModePerm))
|
||||||
notExists := filepath.Join(tmpDir, "notexists.json")
|
notExists := filepath.Join(tmpDir, "notexists.json")
|
||||||
|
|
||||||
manifestFile = "'" + manifestFile + "'"
|
|
||||||
filename = "'" + filename + "'"
|
|
||||||
filenameErr = "'" + filenameErr + "'"
|
filenameErr = "'" + filenameErr + "'"
|
||||||
|
|
||||||
e := newTestVMCLI(t)
|
e := newTestVMCLI(t)
|
||||||
e.runProg(t,
|
e.runProg(t,
|
||||||
"loadnef",
|
"loadnef",
|
||||||
"loadnef "+filenameErr+" "+manifestFile,
|
"loadnef "+filenameErr+" "+manifestFile,
|
||||||
"loadnef "+filename+" "+notExists,
|
"loadnef "+nefFile+" "+notExists,
|
||||||
"loadnef "+filename+" "+filename,
|
"loadnef "+nefFile+" "+nefFile,
|
||||||
"loadnef "+filename+" "+manifestFile,
|
"loadnef "+nefFile+" "+manifestFile,
|
||||||
"run main add 3 5")
|
"run main add 3 5",
|
||||||
|
"loadnef "+nefFile,
|
||||||
|
"run main add 3 5",
|
||||||
|
"loadnef "+nefFile+" "+cmdargs.CosignersSeparator,
|
||||||
|
"loadnef "+nefFile+" "+manifestFile+" "+cmdargs.CosignersSeparator,
|
||||||
|
"loadnef "+nefFile+" "+manifestFile+" "+"not-a-separator",
|
||||||
|
"loadnef "+nefFile+" "+cmdargs.CosignersSeparator+" "+util.Uint160{1, 2, 3}.StringLE(),
|
||||||
|
"run main add 3 5",
|
||||||
|
"loadnef "+nefFile+" "+manifestFile+" "+cmdargs.CosignersSeparator+" "+util.Uint160{1, 2, 3}.StringLE(),
|
||||||
|
"run main add 3 5",
|
||||||
|
)
|
||||||
|
|
||||||
e.checkError(t, ErrMissingParameter)
|
e.checkError(t, ErrMissingParameter)
|
||||||
e.checkNextLine(t, "Error:")
|
e.checkNextLine(t, "Error:")
|
||||||
|
@ -511,6 +615,82 @@ go 1.17`)
|
||||||
e.checkNextLine(t, "Error:")
|
e.checkNextLine(t, "Error:")
|
||||||
e.checkNextLine(t, "READY: loaded \\d* instructions")
|
e.checkNextLine(t, "READY: loaded \\d* instructions")
|
||||||
e.checkStack(t, 8)
|
e.checkStack(t, 8)
|
||||||
|
e.checkNextLine(t, "READY: loaded \\d* instructions")
|
||||||
|
e.checkStack(t, 8)
|
||||||
|
e.checkNextLine(t, "Error:") // manifest missing, missing signer after --
|
||||||
|
e.checkNextLine(t, "Error:") // manifest present, missing signer after --
|
||||||
|
e.checkNextLine(t, "Error:") // manifest present, invalid separator
|
||||||
|
e.checkNextLine(t, "READY: loaded \\d* instructions") // manifest missing, signer present, OK
|
||||||
|
e.checkStack(t, 8)
|
||||||
|
e.checkNextLine(t, "READY: loaded \\d* instructions") // manifest present, signer present, OK
|
||||||
|
e.checkStack(t, 8)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestLoad_RunWithCALLT(t *testing.T) {
|
||||||
|
// Our smart compiler will generate CALLT instruction for the following StdLib call:
|
||||||
|
src := `package kek
|
||||||
|
import "github.com/nspcc-dev/neo-go/pkg/interop/native/std"
|
||||||
|
func Main() int {
|
||||||
|
return std.Atoi("123", 10)
|
||||||
|
}`
|
||||||
|
|
||||||
|
t.Run("loadgo", func(t *testing.T) {
|
||||||
|
tmp := t.TempDir()
|
||||||
|
filename := prepareLoadgoSrc(t, tmp, src)
|
||||||
|
e := newTestVMCLI(t)
|
||||||
|
e.runProg(t,
|
||||||
|
"loadgo "+filename,
|
||||||
|
"run main",
|
||||||
|
)
|
||||||
|
e.checkNextLine(t, "READY: loaded \\d* instructions")
|
||||||
|
e.checkStack(t, 123)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("loadnef", func(t *testing.T) {
|
||||||
|
tmpDir := t.TempDir()
|
||||||
|
manifestFile, nefFile := prepareLoadnefSrc(t, tmpDir, src)
|
||||||
|
|
||||||
|
e := newTestVMCLI(t)
|
||||||
|
e.runProg(t,
|
||||||
|
"loadnef "+nefFile+" "+manifestFile,
|
||||||
|
"run main",
|
||||||
|
)
|
||||||
|
e.checkNextLine(t, "READY: loaded \\d* instructions")
|
||||||
|
e.checkStack(t, 123)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("loaddeployed", func(t *testing.T) {
|
||||||
|
// We'll use `Runtime example` example contract which has a call to native Management
|
||||||
|
// inside performed via CALLT instruction (`destroy` method).
|
||||||
|
e := newTestVMClIWithState(t)
|
||||||
|
var (
|
||||||
|
cH util.Uint160
|
||||||
|
cName = "Runtime example"
|
||||||
|
bc = e.cli.chain
|
||||||
|
)
|
||||||
|
for i := int32(1); ; i++ {
|
||||||
|
h, err := bc.GetContractScriptHash(i)
|
||||||
|
if err != nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
cs := bc.GetContractState(h)
|
||||||
|
if cs == nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if cs.Manifest.Name == cName {
|
||||||
|
cH = cs.Hash
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
require.NotEmpty(t, cH, fmt.Sprintf("failed to locate `%s` example contract with CALLT usage in the simple chain", cName))
|
||||||
|
e.runProg(t,
|
||||||
|
"loaddeployed "+cH.StringLE()+" -- NbrUYaZgyhSkNoRo9ugRyEMdUZxrhkNaWB:Global", // the contract's owner got from the contract's code.
|
||||||
|
"run destroy",
|
||||||
|
"exit",
|
||||||
|
)
|
||||||
|
e.checkNextLine(t, "READY: loaded \\d* instructions")
|
||||||
|
e.checkStack(t) // Nothing on stack, successful execution.
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -532,7 +712,7 @@ func TestRunWithDifferentArguments(t *testing.T) {
|
||||||
func GetString(arg string) string {
|
func GetString(arg string) string {
|
||||||
return arg
|
return arg
|
||||||
}
|
}
|
||||||
func GetArr(arg []interface{}) []interface{}{
|
func GetArr(arg []any) []any{
|
||||||
return arg
|
return arg
|
||||||
}`
|
}`
|
||||||
|
|
||||||
|
@ -924,7 +1104,7 @@ func TestRunWithHistoricState(t *testing.T) {
|
||||||
e.checkNextLine(t, "READY: loaded 36 instructions")
|
e.checkNextLine(t, "READY: loaded 36 instructions")
|
||||||
e.checkStack(t, []byte{1})
|
e.checkStack(t, []byte{1})
|
||||||
e.checkNextLine(t, "READY: loaded 36 instructions")
|
e.checkNextLine(t, "READY: loaded 36 instructions")
|
||||||
e.checkNextLineExact(t, "Error: at instruction 31 (SYSCALL): System.Contract.Call failed: called contract 73a23e915b66ae406866787f4a6c1c517dc981e2 not found: key not found\n")
|
e.checkNextLineExact(t, "Error: at instruction 31 (SYSCALL): System.Contract.Call failed: called contract 0f825b050eb8ce9eaa82993e90615025ab798016 not found: key not found\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestEvents(t *testing.T) {
|
func TestEvents(t *testing.T) {
|
||||||
|
@ -933,7 +1113,7 @@ func TestEvents(t *testing.T) {
|
||||||
script := io.NewBufBinWriter()
|
script := io.NewBufBinWriter()
|
||||||
h, err := e.cli.chain.GetContractScriptHash(2) // examples/runtime/runtime.go
|
h, err := e.cli.chain.GetContractScriptHash(2) // examples/runtime/runtime.go
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
emit.AppCall(script.BinWriter, h, "notify", callflag.All, []interface{}{true, 5})
|
emit.AppCall(script.BinWriter, h, "notify", callflag.All, []any{true, 5})
|
||||||
e.runProg(t,
|
e.runProg(t,
|
||||||
"loadhex "+hex.EncodeToString(script.Bytes()),
|
"loadhex "+hex.EncodeToString(script.Bytes()),
|
||||||
"run",
|
"run",
|
||||||
|
@ -1153,24 +1333,27 @@ func TestLoaddeployed(t *testing.T) {
|
||||||
"run get 1",
|
"run get 1",
|
||||||
"loaddeployed --gas 420000 "+h.StringLE(), // gas-limited
|
"loaddeployed --gas 420000 "+h.StringLE(), // gas-limited
|
||||||
"run get 1",
|
"run get 1",
|
||||||
"loaddeployed 0x"+h.StringLE(), // hash LE with 0x prefix
|
"loaddeployed "+"0x"+h.StringLE(), // hash LE with 0x prefix
|
||||||
"run get 1",
|
"run get 1",
|
||||||
"loaddeployed 1", // contract ID
|
"loaddeployed 1", // contract ID
|
||||||
"run get 1",
|
"run get 1",
|
||||||
"loaddeployed --historic 2 1", // historic state, check that hash is properly set
|
"loaddeployed --historic 2 1", // historic state, check that hash is properly set
|
||||||
"run get 1",
|
"run get 1",
|
||||||
// Check signers parsing:
|
// Check signers parsing:
|
||||||
"loaddeployed 2 "+ownerAddress, // check witness (owner:DefautScope => true)
|
"loaddeployed 2 "+cmdargs.CosignersSeparator,
|
||||||
|
"loaddeployed 2 "+"not-a-separator",
|
||||||
|
"loaddeployed 2 "+cmdargs.CosignersSeparator+" not-a-signer",
|
||||||
|
"loaddeployed 2 "+cmdargs.CosignersSeparator+" "+ownerAddress, // check witness (owner:DefautScope => true)
|
||||||
"run checkWitness",
|
"run checkWitness",
|
||||||
"loaddeployed 2 "+ownerAddress+":None", // check witness (owner:None => false)
|
"loaddeployed 2 "+cmdargs.CosignersSeparator+" "+ownerAddress+":None", // check witness (owner:None => false)
|
||||||
"run checkWitness",
|
"run checkWitness",
|
||||||
"loaddeployed 2 "+ownerAddress+":CalledByEntry", // check witness (owner:CalledByEntry => true)
|
"loaddeployed 2 "+cmdargs.CosignersSeparator+" "+ownerAddress+":CalledByEntry", // check witness (owner:CalledByEntry => true)
|
||||||
"run checkWitness",
|
"run checkWitness",
|
||||||
"loaddeployed 2 "+ownerAcc.StringLE()+":CalledByEntry", // check witness (ownerLE:CalledByEntry => true)
|
"loaddeployed 2 "+cmdargs.CosignersSeparator+" "+ownerAcc.StringLE()+":CalledByEntry", // check witness (ownerLE:CalledByEntry => true)
|
||||||
"run checkWitness",
|
"run checkWitness",
|
||||||
"loaddeployed 2 0x"+ownerAcc.StringLE()+":CalledByEntry", // check witness (owner0xLE:CalledByEntry => true)
|
"loaddeployed 2 "+cmdargs.CosignersSeparator+" "+"0x"+ownerAcc.StringLE()+":CalledByEntry", // check witness (owner0xLE:CalledByEntry => true)
|
||||||
"run checkWitness",
|
"run checkWitness",
|
||||||
"loaddeployed 2 "+sideAcc.StringLE()+":Global", // check witness (sideLE:Global => false)
|
"loaddeployed 2 "+cmdargs.CosignersSeparator+" "+sideAcc.StringLE()+":Global", // check witness (sideLE:Global => false)
|
||||||
"run checkWitness",
|
"run checkWitness",
|
||||||
"loaddeployed", // missing argument
|
"loaddeployed", // missing argument
|
||||||
"exit",
|
"exit",
|
||||||
|
@ -1186,6 +1369,9 @@ func TestLoaddeployed(t *testing.T) {
|
||||||
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
e.checkNextLine(t, "READY: loaded \\d+ instructions")
|
||||||
e.checkStack(t, []byte{1})
|
e.checkStack(t, []byte{1})
|
||||||
// Check signers parsing:
|
// Check signers parsing:
|
||||||
|
e.checkError(t, ErrInvalidParameter)
|
||||||
|
e.checkError(t, ErrInvalidParameter)
|
||||||
|
e.checkError(t, ErrInvalidParameter)
|
||||||
e.checkNextLine(t, "READY: loaded \\d+ instructions") // check witness of owner:DefaultScope
|
e.checkNextLine(t, "READY: loaded \\d+ instructions") // check witness of owner:DefaultScope
|
||||||
e.checkStack(t, true)
|
e.checkStack(t, true)
|
||||||
e.checkNextLine(t, "READY: loaded \\d+ instructions") // check witness of owner:None
|
e.checkNextLine(t, "READY: loaded \\d+ instructions") // check witness of owner:None
|
||||||
|
|
|
@ -13,7 +13,7 @@ import (
|
||||||
|
|
||||||
// NewCommands returns 'vm' command.
|
// NewCommands returns 'vm' command.
|
||||||
func NewCommands() []cli.Command {
|
func NewCommands() []cli.Command {
|
||||||
cfgFlags := []cli.Flag{options.Config}
|
cfgFlags := []cli.Flag{options.Config, options.ConfigFile, options.RelativePath}
|
||||||
cfgFlags = append(cfgFlags, options.Network...)
|
cfgFlags = append(cfgFlags, options.Network...)
|
||||||
return []cli.Command{{
|
return []cli.Command{{
|
||||||
Name: "vm",
|
Name: "vm",
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
package wallet_test
|
package wallet_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/hex"
|
|
||||||
"math/big"
|
"math/big"
|
||||||
"strconv"
|
"strconv"
|
||||||
"testing"
|
"testing"
|
||||||
|
@ -18,7 +17,7 @@ func TestRegisterCandidate(t *testing.T) {
|
||||||
|
|
||||||
validatorAddress := testcli.ValidatorPriv.Address()
|
validatorAddress := testcli.ValidatorPriv.Address()
|
||||||
validatorPublic := testcli.ValidatorPriv.PublicKey()
|
validatorPublic := testcli.ValidatorPriv.PublicKey()
|
||||||
validatorHex := hex.EncodeToString(validatorPublic.Bytes())
|
validatorHex := validatorPublic.StringCompressed()
|
||||||
|
|
||||||
e.In.WriteString("one\r")
|
e.In.WriteString("one\r")
|
||||||
e.Run(t, "neo-go", "wallet", "nep17", "multitransfer",
|
e.Run(t, "neo-go", "wallet", "nep17", "multitransfer",
|
||||||
|
@ -159,4 +158,61 @@ func TestRegisterCandidate(t *testing.T) {
|
||||||
e.RunWithError(t, "neo-go", "query", "voter", "--rpc-endpoint", "http://"+e.RPC.Addresses()[0], validatorAddress, validatorAddress)
|
e.RunWithError(t, "neo-go", "query", "voter", "--rpc-endpoint", "http://"+e.RPC.Addresses()[0], validatorAddress, validatorAddress)
|
||||||
e.RunWithError(t, "neo-go", "query", "committee", "--rpc-endpoint", "http://"+e.RPC.Addresses()[0], "something")
|
e.RunWithError(t, "neo-go", "query", "committee", "--rpc-endpoint", "http://"+e.RPC.Addresses()[0], "something")
|
||||||
e.RunWithError(t, "neo-go", "query", "candidates", "--rpc-endpoint", "http://"+e.RPC.Addresses()[0], "something")
|
e.RunWithError(t, "neo-go", "query", "candidates", "--rpc-endpoint", "http://"+e.RPC.Addresses()[0], "something")
|
||||||
|
|
||||||
|
t.Run("VoteUnvote await", func(t *testing.T) {
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.Run(t, "neo-go", "wallet", "candidate", "register",
|
||||||
|
"--rpc-endpoint", "http://"+e.RPC.Addresses()[0],
|
||||||
|
"--wallet", testcli.ValidatorWallet,
|
||||||
|
"--address", validatorAddress,
|
||||||
|
"--force", "--await")
|
||||||
|
e.CheckAwaitableTxPersisted(t)
|
||||||
|
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.Run(t, "neo-go", "wallet", "candidate", "vote",
|
||||||
|
"--rpc-endpoint", "http://"+e.RPC.Addresses()[0],
|
||||||
|
"--wallet", testcli.ValidatorWallet,
|
||||||
|
"--address", validatorAddress,
|
||||||
|
"--candidate", validatorHex,
|
||||||
|
"--force",
|
||||||
|
"--await")
|
||||||
|
|
||||||
|
e.CheckAwaitableTxPersisted(t)
|
||||||
|
b, _ := e.Chain.GetGoverningTokenBalance(testcli.ValidatorPriv.GetScriptHash())
|
||||||
|
|
||||||
|
// unvote
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.Run(t, "neo-go", "wallet", "candidate", "vote",
|
||||||
|
"--rpc-endpoint", "http://"+e.RPC.Addresses()[0],
|
||||||
|
"--wallet", testcli.ValidatorWallet,
|
||||||
|
"--address", validatorAddress,
|
||||||
|
"--force", "--await")
|
||||||
|
_, index := e.CheckAwaitableTxPersisted(t)
|
||||||
|
|
||||||
|
vs, err = e.Chain.GetEnrollments()
|
||||||
|
require.Equal(t, 1, len(vs))
|
||||||
|
require.Equal(t, validatorPublic, vs[0].Key)
|
||||||
|
require.Equal(t, big.NewInt(0), vs[0].Votes)
|
||||||
|
|
||||||
|
// check state
|
||||||
|
e.Run(t, "neo-go", "query", "voter",
|
||||||
|
"--rpc-endpoint", "http://"+e.RPC.Addresses()[0],
|
||||||
|
validatorAddress)
|
||||||
|
e.CheckNextLine(t, "^\\s*Voted:\\s+"+"null") // no vote.
|
||||||
|
e.CheckNextLine(t, "^\\s*Amount\\s*:\\s*"+b.String()+"$")
|
||||||
|
e.CheckNextLine(t, "^\\s*Block\\s*:\\s*"+strconv.FormatUint(uint64(index), 10))
|
||||||
|
e.CheckEOF(t)
|
||||||
|
})
|
||||||
|
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.Run(t, "neo-go", "wallet", "candidate", "unregister",
|
||||||
|
"--rpc-endpoint", "http://"+e.RPC.Addresses()[0],
|
||||||
|
"--wallet", testcli.ValidatorWallet,
|
||||||
|
"--address", validatorAddress,
|
||||||
|
"--force",
|
||||||
|
"--await")
|
||||||
|
e.CheckAwaitableTxPersisted(t)
|
||||||
|
|
||||||
|
vs, err = e.Chain.GetEnrollments()
|
||||||
|
require.Equal(t, 0, len(vs))
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,6 +8,7 @@ import (
|
||||||
"errors"
|
"errors"
|
||||||
"os"
|
"os"
|
||||||
|
|
||||||
|
"github.com/nspcc-dev/neo-go/cli/options"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
||||||
|
@ -46,7 +47,7 @@ func newWalletV2FromFile(path string, configPath string) (*walletV2, *string, er
|
||||||
}
|
}
|
||||||
var pass *string
|
var pass *string
|
||||||
if len(configPath) != 0 {
|
if len(configPath) != 0 {
|
||||||
cfg, err := ReadWalletConfig(configPath)
|
cfg, err := options.ReadWalletConfig(configPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, err
|
return nil, nil, err
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,12 +1,12 @@
|
||||||
package wallet
|
package wallet
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"crypto/elliptic"
|
"crypto/elliptic"
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/util/slice"
|
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -36,27 +36,27 @@ func TestParseMultisigContract(t *testing.T) {
|
||||||
testParseMultisigContract(t, s, 1, pub)
|
testParseMultisigContract(t, s, 1, pub)
|
||||||
})
|
})
|
||||||
t.Run("bad, no check multisig", func(t *testing.T) {
|
t.Run("bad, no check multisig", func(t *testing.T) {
|
||||||
sBad := slice.Copy(s)
|
sBad := bytes.Clone(s)
|
||||||
sBad[len(sBad)-1] ^= 0xFF
|
sBad[len(sBad)-1] ^= 0xFF
|
||||||
testParseMultisigContract(t, sBad, 0)
|
testParseMultisigContract(t, sBad, 0)
|
||||||
})
|
})
|
||||||
t.Run("bad, invalid number of keys", func(t *testing.T) {
|
t.Run("bad, invalid number of keys", func(t *testing.T) {
|
||||||
sBad := slice.Copy(s)
|
sBad := bytes.Clone(s)
|
||||||
sBad[len(sBad)-2] = opPush1 + 1
|
sBad[len(sBad)-2] = opPush1 + 1
|
||||||
testParseMultisigContract(t, sBad, 0)
|
testParseMultisigContract(t, sBad, 0)
|
||||||
})
|
})
|
||||||
t.Run("bad, invalid first instruction", func(t *testing.T) {
|
t.Run("bad, invalid first instruction", func(t *testing.T) {
|
||||||
sBad := slice.Copy(s)
|
sBad := bytes.Clone(s)
|
||||||
sBad[0] = 0xFF
|
sBad[0] = 0xFF
|
||||||
testParseMultisigContract(t, sBad, 0)
|
testParseMultisigContract(t, sBad, 0)
|
||||||
})
|
})
|
||||||
t.Run("bad, invalid public key", func(t *testing.T) {
|
t.Run("bad, invalid public key", func(t *testing.T) {
|
||||||
sBad := slice.Copy(s)
|
sBad := bytes.Clone(s)
|
||||||
sBad[2] = 0xFF
|
sBad[2] = 0xFF
|
||||||
testParseMultisigContract(t, sBad, 0)
|
testParseMultisigContract(t, sBad, 0)
|
||||||
})
|
})
|
||||||
t.Run("bad, many sigs", func(t *testing.T) {
|
t.Run("bad, many sigs", func(t *testing.T) {
|
||||||
sBad := slice.Copy(s)
|
sBad := bytes.Clone(s)
|
||||||
sBad[0] = opPush1 + 1
|
sBad[0] = opPush1 + 1
|
||||||
testParseMultisigContract(t, sBad, 0)
|
testParseMultisigContract(t, sBad, 0)
|
||||||
})
|
})
|
||||||
|
|
|
@ -8,7 +8,10 @@ import (
|
||||||
"github.com/nspcc-dev/neo-go/cli/flags"
|
"github.com/nspcc-dev/neo-go/cli/flags"
|
||||||
"github.com/nspcc-dev/neo-go/cli/options"
|
"github.com/nspcc-dev/neo-go/cli/options"
|
||||||
"github.com/nspcc-dev/neo-go/cli/paramcontext"
|
"github.com/nspcc-dev/neo-go/cli/paramcontext"
|
||||||
|
"github.com/nspcc-dev/neo-go/cli/txctx"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/waiter"
|
||||||
"github.com/urfave/cli"
|
"github.com/urfave/cli"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -17,15 +20,11 @@ func signStoredTransaction(ctx *cli.Context) error {
|
||||||
out = ctx.String("out")
|
out = ctx.String("out")
|
||||||
rpcNode = ctx.String(options.RPCEndpointFlag)
|
rpcNode = ctx.String(options.RPCEndpointFlag)
|
||||||
addrFlag = ctx.Generic("address").(*flags.Address)
|
addrFlag = ctx.Generic("address").(*flags.Address)
|
||||||
|
aer *state.AppExecResult
|
||||||
)
|
)
|
||||||
if err := cmdargs.EnsureNone(ctx); err != nil {
|
if err := cmdargs.EnsureNone(ctx); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
wall, pass, err := readWallet(ctx)
|
|
||||||
if err != nil {
|
|
||||||
return cli.NewExitError(err, 1)
|
|
||||||
}
|
|
||||||
defer wall.Close()
|
|
||||||
|
|
||||||
pc, err := paramcontext.Read(ctx.String("in"))
|
pc, err := paramcontext.Read(ctx.String("in"))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -35,9 +34,7 @@ func signStoredTransaction(ctx *cli.Context) error {
|
||||||
if !addrFlag.IsSet {
|
if !addrFlag.IsSet {
|
||||||
return cli.NewExitError("address was not provided", 1)
|
return cli.NewExitError("address was not provided", 1)
|
||||||
}
|
}
|
||||||
|
acc, _, err := options.GetAccFromContext(ctx)
|
||||||
var ch = addrFlag.Uint160()
|
|
||||||
acc, err := getDecryptedAccount(wall, ch, pass)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(err, 1)
|
return cli.NewExitError(err, 1)
|
||||||
}
|
}
|
||||||
|
@ -47,20 +44,13 @@ func signStoredTransaction(ctx *cli.Context) error {
|
||||||
return cli.NewExitError("verifiable item is not a transaction", 1)
|
return cli.NewExitError("verifiable item is not a transaction", 1)
|
||||||
}
|
}
|
||||||
|
|
||||||
signerFound := false
|
if !tx.HasSigner(acc.ScriptHash()) {
|
||||||
for i := range tx.Signers {
|
|
||||||
if tx.Signers[i].Account == ch {
|
|
||||||
signerFound = true
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !signerFound {
|
|
||||||
return cli.NewExitError("tx signers don't contain provided account", 1)
|
return cli.NewExitError("tx signers don't contain provided account", 1)
|
||||||
}
|
}
|
||||||
|
|
||||||
if acc.CanSign() {
|
if acc.CanSign() {
|
||||||
sign := acc.SignHashable(pc.Network, pc.Verifiable)
|
sign := acc.SignHashable(pc.Network, pc.Verifiable)
|
||||||
if err := pc.AddSignature(ch, acc.Contract, acc.PublicKey(), sign); err != nil {
|
if err := pc.AddSignature(acc.ScriptHash(), acc.Contract, acc.PublicKey(), sign); err != nil {
|
||||||
return cli.NewExitError(fmt.Errorf("can't add signature: %w", err), 1)
|
return cli.NewExitError(fmt.Errorf("can't add signature: %w", err), 1)
|
||||||
}
|
}
|
||||||
} else if rpcNode == "" {
|
} else if rpcNode == "" {
|
||||||
|
@ -98,10 +88,15 @@ func signStoredTransaction(ctx *cli.Context) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(fmt.Errorf("failed to submit transaction to RPC node: %w", err), 1)
|
return cli.NewExitError(fmt.Errorf("failed to submit transaction to RPC node: %w", err), 1)
|
||||||
}
|
}
|
||||||
fmt.Fprintln(ctx.App.Writer, res.StringLE())
|
if ctx.Bool("await") {
|
||||||
return nil
|
version, err := c.GetVersion()
|
||||||
|
aer, err = waiter.New(c, version).Wait(res, tx.ValidUntilBlock, err)
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(fmt.Errorf("failed to await transaction %s: %w", res.StringLE(), err), 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Fprintln(ctx.App.Writer, tx.Hash().StringLE())
|
txctx.DumpTransactionInfo(ctx.App.Writer, tx.Hash(), aer)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
package wallet_test
|
package wallet_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/hex"
|
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"math/big"
|
"math/big"
|
||||||
|
@ -45,9 +44,9 @@ func TestSignMultisigTx(t *testing.T) {
|
||||||
"--wallet", w,
|
"--wallet", w,
|
||||||
"--wif", wif,
|
"--wif", wif,
|
||||||
"--min", "2",
|
"--min", "2",
|
||||||
hex.EncodeToString(pubs[0].Bytes()),
|
pubs[0].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[1].Bytes()),
|
pubs[1].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[2].Bytes()))
|
pubs[2].StringCompressed())
|
||||||
}
|
}
|
||||||
addAccount(wallet1Path, privs[0].WIF())
|
addAccount(wallet1Path, privs[0].WIF())
|
||||||
addAccount(wallet2Path, privs[1].WIF())
|
addAccount(wallet2Path, privs[1].WIF())
|
||||||
|
|
|
@ -101,7 +101,7 @@ func newNEP11Commands() []cli.Command {
|
||||||
{
|
{
|
||||||
Name: "transfer",
|
Name: "transfer",
|
||||||
Usage: "transfer NEP-11 tokens",
|
Usage: "transfer NEP-11 tokens",
|
||||||
UsageText: "transfer -w wallet [--wallet-config path] --rpc-endpoint <node> --timeout <time> --from <addr> --to <addr> --token <hash-or-name> --id <token-id> [--amount string] [data] [-- <cosigner1:Scope> [<cosigner2> [...]]]",
|
UsageText: "transfer -w wallet [--wallet-config path] --rpc-endpoint <node> --timeout <time> --from <addr> --to <addr> --token <hash-or-name> --id <token-id> [--amount string] [--await] [data] [-- <cosigner1:Scope> [<cosigner2> [...]]]",
|
||||||
Action: transferNEP11,
|
Action: transferNEP11,
|
||||||
Flags: transferFlags,
|
Flags: transferFlags,
|
||||||
Description: `Transfers specified NEP-11 token with optional cosigners list attached to
|
Description: `Transfers specified NEP-11 token with optional cosigners list attached to
|
||||||
|
@ -110,7 +110,8 @@ func newNEP11Commands() []cli.Command {
|
||||||
'contract testinvokefunction' documentation for the details
|
'contract testinvokefunction' documentation for the details
|
||||||
about cosigners syntax. If no cosigners are given then the
|
about cosigners syntax. If no cosigners are given then the
|
||||||
sender with CalledByEntry scope will be used as the only
|
sender with CalledByEntry scope will be used as the only
|
||||||
signer.
|
signer. If --await flag is set then the command will wait
|
||||||
|
for the transaction to be included in a block.
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|
|
@ -30,6 +30,14 @@ import (
|
||||||
"github.com/urfave/cli"
|
"github.com/urfave/cli"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// transferTarget represents target address, token amount and data for transfer.
|
||||||
|
type transferTarget struct {
|
||||||
|
Token util.Uint160
|
||||||
|
Address util.Uint160
|
||||||
|
Amount int64
|
||||||
|
Data any
|
||||||
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
tokenFlag = cli.StringFlag{
|
tokenFlag = cli.StringFlag{
|
||||||
Name: "token",
|
Name: "token",
|
||||||
|
@ -62,6 +70,7 @@ var (
|
||||||
txctx.GasFlag,
|
txctx.GasFlag,
|
||||||
txctx.SysGasFlag,
|
txctx.SysGasFlag,
|
||||||
txctx.ForceFlag,
|
txctx.ForceFlag,
|
||||||
|
txctx.AwaitFlag,
|
||||||
cli.StringFlag{
|
cli.StringFlag{
|
||||||
Name: "amount",
|
Name: "amount",
|
||||||
Usage: "Amount of asset to send",
|
Usage: "Amount of asset to send",
|
||||||
|
@ -75,6 +84,7 @@ var (
|
||||||
txctx.GasFlag,
|
txctx.GasFlag,
|
||||||
txctx.SysGasFlag,
|
txctx.SysGasFlag,
|
||||||
txctx.ForceFlag,
|
txctx.ForceFlag,
|
||||||
|
txctx.AwaitFlag,
|
||||||
}, options.RPC...)
|
}, options.RPC...)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -139,20 +149,22 @@ func newNEP17Commands() []cli.Command {
|
||||||
{
|
{
|
||||||
Name: "transfer",
|
Name: "transfer",
|
||||||
Usage: "transfer NEP-17 tokens",
|
Usage: "transfer NEP-17 tokens",
|
||||||
UsageText: "transfer -w wallet [--wallet-config path] --rpc-endpoint <node> --timeout <time> --from <addr> --to <addr> --token <hash-or-name> --amount string [data] [-- <cosigner1:Scope> [<cosigner2> [...]]]",
|
UsageText: "transfer -w wallet [--wallet-config path] [--await] --rpc-endpoint <node> --timeout <time> --from <addr> --to <addr> --token <hash-or-name> --amount string [data] [-- <cosigner1:Scope> [<cosigner2> [...]]]",
|
||||||
Action: transferNEP17,
|
Action: transferNEP17,
|
||||||
Flags: transferFlags,
|
Flags: transferFlags,
|
||||||
Description: `Transfers specified NEP-17 token amount with optional 'data' parameter and cosigners
|
Description: `Transfers specified NEP-17 token amount with optional 'data' parameter and cosigners
|
||||||
list attached to the transfer. See 'contract testinvokefunction' documentation
|
list attached to the transfer. See 'contract testinvokefunction' documentation
|
||||||
for the details about 'data' parameter and cosigners syntax. If no 'data' is
|
for the details about 'data' parameter and cosigners syntax. If no 'data' is
|
||||||
given then default nil value will be used. If no cosigners are given then the
|
given then default nil value will be used. If no cosigners are given then the
|
||||||
sender with CalledByEntry scope will be used as the only signer.
|
sender with CalledByEntry scope will be used as the only signer. When --await
|
||||||
|
flag is used, the command waits for the transaction to be included in a block
|
||||||
|
before exiting.
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "multitransfer",
|
Name: "multitransfer",
|
||||||
Usage: "transfer NEP-17 tokens to multiple recipients",
|
Usage: "transfer NEP-17 tokens to multiple recipients",
|
||||||
UsageText: `multitransfer -w wallet [--wallet-config path] --rpc-endpoint <node> --timeout <time> --from <addr>` +
|
UsageText: `multitransfer -w wallet [--wallet-config path] [--await] --rpc-endpoint <node> --timeout <time> --from <addr>` +
|
||||||
` <token1>:<addr1>:<amount1> [<token2>:<addr2>:<amount2> [...]] [-- <cosigner1:Scope> [<cosigner2> [...]]]`,
|
` <token1>:<addr1>:<amount1> [<token2>:<addr2>:<amount2> [...]] [-- <cosigner1:Scope> [<cosigner2> [...]]]`,
|
||||||
Action: multiTransferNEP17,
|
Action: multiTransferNEP17,
|
||||||
Flags: multiTransferFlags,
|
Flags: multiTransferFlags,
|
||||||
|
@ -514,7 +526,7 @@ func multiTransferNEP17(ctx *cli.Context) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(err, 1)
|
return cli.NewExitError(err, 1)
|
||||||
}
|
}
|
||||||
acc, err := getDecryptedAccount(wall, from, pass)
|
acc, err := options.GetUnlockedAccount(wall, from, pass)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(err, 1)
|
return cli.NewExitError(err, 1)
|
||||||
}
|
}
|
||||||
|
@ -522,25 +534,36 @@ func multiTransferNEP17(ctx *cli.Context) error {
|
||||||
gctx, cancel := options.GetTimeoutContext(ctx)
|
gctx, cancel := options.GetTimeoutContext(ctx)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
c, err := options.GetRPCClient(gctx, ctx)
|
|
||||||
if err != nil {
|
|
||||||
return cli.NewExitError(err, 1)
|
|
||||||
}
|
|
||||||
|
|
||||||
if ctx.NArg() == 0 {
|
if ctx.NArg() == 0 {
|
||||||
return cli.NewExitError("empty recipients list", 1)
|
return cli.NewExitError("empty recipients list", 1)
|
||||||
}
|
}
|
||||||
var (
|
var (
|
||||||
recipients []rpcclient.TransferTarget
|
recipients []transferTarget
|
||||||
cosignersOffset = ctx.NArg()
|
cosignersSepPos = ctx.NArg() // `--` position.
|
||||||
)
|
)
|
||||||
cache := make(map[string]*wallet.Token)
|
|
||||||
for i := 0; i < ctx.NArg(); i++ {
|
for i := 0; i < ctx.NArg(); i++ {
|
||||||
arg := ctx.Args().Get(i)
|
arg := ctx.Args().Get(i)
|
||||||
if arg == cmdargs.CosignersSeparator {
|
if arg == cmdargs.CosignersSeparator {
|
||||||
cosignersOffset = i + 1
|
cosignersSepPos = i
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
cosigners, extErr := cmdargs.GetSignersFromContext(ctx, cosignersSepPos+1)
|
||||||
|
if extErr != nil {
|
||||||
|
return extErr
|
||||||
|
}
|
||||||
|
signersAccounts, err := cmdargs.GetSignersAccounts(acc, wall, cosigners, transaction.CalledByEntry)
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(fmt.Errorf("invalid signers: %w", err), 1)
|
||||||
|
}
|
||||||
|
c, act, exitErr := options.GetRPCWithActor(gctx, ctx, signersAccounts)
|
||||||
|
if exitErr != nil {
|
||||||
|
return exitErr
|
||||||
|
}
|
||||||
|
|
||||||
|
cache := make(map[string]*wallet.Token)
|
||||||
|
for i := 0; i < cosignersSepPos; i++ {
|
||||||
|
arg := ctx.Args().Get(i)
|
||||||
ss := strings.SplitN(arg, ":", 3)
|
ss := strings.SplitN(arg, ":", 3)
|
||||||
if len(ss) != 3 {
|
if len(ss) != 3 {
|
||||||
return cli.NewExitError("send format must be '<token>:<addr>:<amount>", 1)
|
return cli.NewExitError("send format must be '<token>:<addr>:<amount>", 1)
|
||||||
|
@ -564,7 +587,7 @@ func multiTransferNEP17(ctx *cli.Context) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(fmt.Errorf("invalid amount: %w", err), 1)
|
return cli.NewExitError(fmt.Errorf("invalid amount: %w", err), 1)
|
||||||
}
|
}
|
||||||
recipients = append(recipients, rpcclient.TransferTarget{
|
recipients = append(recipients, transferTarget{
|
||||||
Token: token.Hash,
|
Token: token.Hash,
|
||||||
Address: addr,
|
Address: addr,
|
||||||
Amount: amount.Int64(),
|
Amount: amount.Int64(),
|
||||||
|
@ -572,19 +595,6 @@ func multiTransferNEP17(ctx *cli.Context) error {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
cosigners, extErr := cmdargs.GetSignersFromContext(ctx, cosignersOffset)
|
|
||||||
if extErr != nil {
|
|
||||||
return extErr
|
|
||||||
}
|
|
||||||
signersAccounts, err := cmdargs.GetSignersAccounts(acc, wall, cosigners, transaction.CalledByEntry)
|
|
||||||
if err != nil {
|
|
||||||
return cli.NewExitError(fmt.Errorf("invalid signers: %w", err), 1)
|
|
||||||
}
|
|
||||||
act, err := actor.New(c, signersAccounts)
|
|
||||||
if err != nil {
|
|
||||||
return cli.NewExitError(fmt.Errorf("failed to create RPC actor: %w", err), 1)
|
|
||||||
}
|
|
||||||
|
|
||||||
tx, err := makeMultiTransferNEP17(act, recipients)
|
tx, err := makeMultiTransferNEP17(act, recipients)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(fmt.Errorf("can't make transaction: %w", err), 1)
|
return cli.NewExitError(fmt.Errorf("can't make transaction: %w", err), 1)
|
||||||
|
@ -610,7 +620,7 @@ func transferNEP(ctx *cli.Context, standard string) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(err, 1)
|
return cli.NewExitError(err, 1)
|
||||||
}
|
}
|
||||||
acc, err := getDecryptedAccount(wall, from, pass)
|
acc, err := options.GetUnlockedAccount(wall, from, pass)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(err, 1)
|
return cli.NewExitError(err, 1)
|
||||||
}
|
}
|
||||||
|
@ -618,9 +628,22 @@ func transferNEP(ctx *cli.Context, standard string) error {
|
||||||
gctx, cancel := options.GetTimeoutContext(ctx)
|
gctx, cancel := options.GetTimeoutContext(ctx)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
c, err := options.GetRPCClient(gctx, ctx)
|
cosignersOffset, data, extErr := cmdargs.GetDataFromContext(ctx)
|
||||||
|
if extErr != nil {
|
||||||
|
return extErr
|
||||||
|
}
|
||||||
|
cosigners, extErr := cmdargs.GetSignersFromContext(ctx, cosignersOffset)
|
||||||
|
if extErr != nil {
|
||||||
|
return extErr
|
||||||
|
}
|
||||||
|
signersAccounts, err := cmdargs.GetSignersAccounts(acc, wall, cosigners, transaction.CalledByEntry)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(err, 1)
|
return cli.NewExitError(fmt.Errorf("invalid signers: %w", err), 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
c, act, exitErr := options.GetRPCWithActor(gctx, ctx, signersAccounts)
|
||||||
|
if exitErr != nil {
|
||||||
|
return exitErr
|
||||||
}
|
}
|
||||||
|
|
||||||
toFlag := ctx.Generic("to").(*flags.Address)
|
toFlag := ctx.Generic("to").(*flags.Address)
|
||||||
|
@ -636,24 +659,6 @@ func transferNEP(ctx *cli.Context, standard string) error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
cosignersOffset, data, extErr := cmdargs.GetDataFromContext(ctx)
|
|
||||||
if extErr != nil {
|
|
||||||
return extErr
|
|
||||||
}
|
|
||||||
|
|
||||||
cosigners, extErr := cmdargs.GetSignersFromContext(ctx, cosignersOffset)
|
|
||||||
if extErr != nil {
|
|
||||||
return extErr
|
|
||||||
}
|
|
||||||
signersAccounts, err := cmdargs.GetSignersAccounts(acc, wall, cosigners, transaction.CalledByEntry)
|
|
||||||
if err != nil {
|
|
||||||
return cli.NewExitError(fmt.Errorf("invalid signers: %w", err), 1)
|
|
||||||
}
|
|
||||||
act, err := actor.New(c, signersAccounts)
|
|
||||||
if err != nil {
|
|
||||||
return cli.NewExitError(fmt.Errorf("failed to create RPC actor: %w", err), 1)
|
|
||||||
}
|
|
||||||
|
|
||||||
amountArg := ctx.String("amount")
|
amountArg := ctx.String("amount")
|
||||||
amount, err := fixedn.FromString(amountArg, int(token.Decimals))
|
amount, err := fixedn.FromString(amountArg, int(token.Decimals))
|
||||||
// It's OK for NEP-11 transfer to not have amount set.
|
// It's OK for NEP-11 transfer to not have amount set.
|
||||||
|
@ -690,7 +695,7 @@ func transferNEP(ctx *cli.Context, standard string) error {
|
||||||
return txctx.SignAndSend(ctx, act, acc, tx)
|
return txctx.SignAndSend(ctx, act, acc, tx)
|
||||||
}
|
}
|
||||||
|
|
||||||
func makeMultiTransferNEP17(act *actor.Actor, recipients []rpcclient.TransferTarget) (*transaction.Transaction, error) {
|
func makeMultiTransferNEP17(act *actor.Actor, recipients []transferTarget) (*transaction.Transaction, error) {
|
||||||
scr := smartcontract.NewBuilder()
|
scr := smartcontract.NewBuilder()
|
||||||
for i := range recipients {
|
for i := range recipients {
|
||||||
scr.InvokeWithAssert(recipients[i].Token, "transfer", act.Sender(),
|
scr.InvokeWithAssert(recipients[i].Token, "transfer", act.Sender(),
|
||||||
|
|
|
@ -5,13 +5,10 @@ import (
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/cli/cmdargs"
|
"github.com/nspcc-dev/neo-go/cli/cmdargs"
|
||||||
"github.com/nspcc-dev/neo-go/cli/flags"
|
"github.com/nspcc-dev/neo-go/cli/flags"
|
||||||
"github.com/nspcc-dev/neo-go/cli/input"
|
|
||||||
"github.com/nspcc-dev/neo-go/cli/options"
|
"github.com/nspcc-dev/neo-go/cli/options"
|
||||||
"github.com/nspcc-dev/neo-go/cli/txctx"
|
"github.com/nspcc-dev/neo-go/cli/txctx"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/rpcclient/actor"
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/rpcclient/neo"
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/neo"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/util"
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/wallet"
|
"github.com/nspcc-dev/neo-go/pkg/wallet"
|
||||||
|
@ -23,7 +20,7 @@ func newValidatorCommands() []cli.Command {
|
||||||
{
|
{
|
||||||
Name: "register",
|
Name: "register",
|
||||||
Usage: "register as a new candidate",
|
Usage: "register as a new candidate",
|
||||||
UsageText: "register -w <path> -r <rpc> -a <addr> [-g gas] [-e sysgas] [--out file] [--force]",
|
UsageText: "register -w <path> -r <rpc> -a <addr> [-g gas] [-e sysgas] [--out file] [--force] [--await]",
|
||||||
Action: handleRegister,
|
Action: handleRegister,
|
||||||
Flags: append([]cli.Flag{
|
Flags: append([]cli.Flag{
|
||||||
walletPathFlag,
|
walletPathFlag,
|
||||||
|
@ -32,6 +29,7 @@ func newValidatorCommands() []cli.Command {
|
||||||
txctx.SysGasFlag,
|
txctx.SysGasFlag,
|
||||||
txctx.OutFlag,
|
txctx.OutFlag,
|
||||||
txctx.ForceFlag,
|
txctx.ForceFlag,
|
||||||
|
txctx.AwaitFlag,
|
||||||
flags.AddressFlag{
|
flags.AddressFlag{
|
||||||
Name: "address, a",
|
Name: "address, a",
|
||||||
Usage: "Address to register",
|
Usage: "Address to register",
|
||||||
|
@ -41,7 +39,7 @@ func newValidatorCommands() []cli.Command {
|
||||||
{
|
{
|
||||||
Name: "unregister",
|
Name: "unregister",
|
||||||
Usage: "unregister self as a candidate",
|
Usage: "unregister self as a candidate",
|
||||||
UsageText: "unregister -w <path> -r <rpc> -a <addr> [-g gas] [-e sysgas] [--out file] [--force]",
|
UsageText: "unregister -w <path> -r <rpc> -a <addr> [-g gas] [-e sysgas] [--out file] [--force] [--await]",
|
||||||
Action: handleUnregister,
|
Action: handleUnregister,
|
||||||
Flags: append([]cli.Flag{
|
Flags: append([]cli.Flag{
|
||||||
walletPathFlag,
|
walletPathFlag,
|
||||||
|
@ -50,6 +48,7 @@ func newValidatorCommands() []cli.Command {
|
||||||
txctx.SysGasFlag,
|
txctx.SysGasFlag,
|
||||||
txctx.OutFlag,
|
txctx.OutFlag,
|
||||||
txctx.ForceFlag,
|
txctx.ForceFlag,
|
||||||
|
txctx.AwaitFlag,
|
||||||
flags.AddressFlag{
|
flags.AddressFlag{
|
||||||
Name: "address, a",
|
Name: "address, a",
|
||||||
Usage: "Address to unregister",
|
Usage: "Address to unregister",
|
||||||
|
@ -59,9 +58,10 @@ func newValidatorCommands() []cli.Command {
|
||||||
{
|
{
|
||||||
Name: "vote",
|
Name: "vote",
|
||||||
Usage: "vote for a validator",
|
Usage: "vote for a validator",
|
||||||
UsageText: "vote -w <path> -r <rpc> [-s <timeout>] [-g gas] [-e sysgas] -a <addr> [-c <public key>] [--out file] [--force]",
|
UsageText: "vote -w <path> -r <rpc> [-s <timeout>] [-g gas] [-e sysgas] -a <addr> [-c <public key>] [--out file] [--force] [--await]",
|
||||||
Description: `Votes for a validator by calling "vote" method of a NEO native
|
Description: `Votes for a validator by calling "vote" method of a NEO native
|
||||||
contract. Do not provide candidate argument to perform unvoting.
|
contract. Do not provide candidate argument to perform unvoting. If --await flag is
|
||||||
|
included, the command waits for the transaction to be included in a block before exiting.
|
||||||
`,
|
`,
|
||||||
Action: handleVote,
|
Action: handleVote,
|
||||||
Flags: append([]cli.Flag{
|
Flags: append([]cli.Flag{
|
||||||
|
@ -71,6 +71,7 @@ func newValidatorCommands() []cli.Command {
|
||||||
txctx.SysGasFlag,
|
txctx.SysGasFlag,
|
||||||
txctx.OutFlag,
|
txctx.OutFlag,
|
||||||
txctx.ForceFlag,
|
txctx.ForceFlag,
|
||||||
|
txctx.AwaitFlag,
|
||||||
flags.AddressFlag{
|
flags.AddressFlag{
|
||||||
Name: "address, a",
|
Name: "address, a",
|
||||||
Usage: "Address to vote from",
|
Usage: "Address to vote from",
|
||||||
|
@ -111,7 +112,7 @@ func handleNeoAction(ctx *cli.Context, mkTx func(*neo.Contract, util.Uint160, *w
|
||||||
return cli.NewExitError("address was not provided", 1)
|
return cli.NewExitError("address was not provided", 1)
|
||||||
}
|
}
|
||||||
addr := addrFlag.Uint160()
|
addr := addrFlag.Uint160()
|
||||||
acc, err := getDecryptedAccount(wall, addr, pass)
|
acc, err := options.GetUnlockedAccount(wall, addr, pass)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(err, 1)
|
return cli.NewExitError(err, 1)
|
||||||
}
|
}
|
||||||
|
@ -119,13 +120,13 @@ func handleNeoAction(ctx *cli.Context, mkTx func(*neo.Contract, util.Uint160, *w
|
||||||
gctx, cancel := options.GetTimeoutContext(ctx)
|
gctx, cancel := options.GetTimeoutContext(ctx)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
c, err := options.GetRPCClient(gctx, ctx)
|
signers, err := cmdargs.GetSignersAccounts(acc, wall, nil, transaction.CalledByEntry)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(err, 1)
|
return cli.NewExitError(fmt.Errorf("invalid signers: %w", err), 1)
|
||||||
}
|
}
|
||||||
act, err := actor.NewSimple(c, acc)
|
_, act, exitErr := options.GetRPCWithActor(gctx, ctx, signers)
|
||||||
if err != nil {
|
if exitErr != nil {
|
||||||
return cli.NewExitError(fmt.Errorf("RPC actor issue: %w", err), 1)
|
return exitErr
|
||||||
}
|
}
|
||||||
|
|
||||||
contract := neo.New(act)
|
contract := neo.New(act)
|
||||||
|
@ -153,32 +154,3 @@ func handleVote(ctx *cli.Context) error {
|
||||||
return contract.VoteUnsigned(addr, pub)
|
return contract.VoteUnsigned(addr, pub)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// getDecryptedAccount tries to get and unlock the specified account if it has a
|
|
||||||
// key inside (otherwise it's returned as is, without an ability to sign). If
|
|
||||||
// password is nil, it will be requested via terminal.
|
|
||||||
func getDecryptedAccount(wall *wallet.Wallet, addr util.Uint160, password *string) (*wallet.Account, error) {
|
|
||||||
acc := wall.GetAccount(addr)
|
|
||||||
if acc == nil {
|
|
||||||
return nil, fmt.Errorf("can't find account for the address: %s", address.Uint160ToString(addr))
|
|
||||||
}
|
|
||||||
|
|
||||||
// No private key available, nothing to decrypt, but it's still a useful account for many purposes.
|
|
||||||
if acc.EncryptedWIF == "" {
|
|
||||||
return acc, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if password == nil {
|
|
||||||
pass, err := input.ReadPassword(EnterPasswordPrompt)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println("Error reading password", err)
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
password = &pass
|
|
||||||
}
|
|
||||||
err := acc.Decrypt(*password, wall.Scrypt)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return acc, nil
|
|
||||||
}
|
|
||||||
|
|
|
@ -15,7 +15,6 @@ import (
|
||||||
"github.com/nspcc-dev/neo-go/cli/input"
|
"github.com/nspcc-dev/neo-go/cli/input"
|
||||||
"github.com/nspcc-dev/neo-go/cli/options"
|
"github.com/nspcc-dev/neo-go/cli/options"
|
||||||
"github.com/nspcc-dev/neo-go/cli/txctx"
|
"github.com/nspcc-dev/neo-go/cli/txctx"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/config"
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
||||||
|
@ -26,7 +25,6 @@ import (
|
||||||
"github.com/nspcc-dev/neo-go/pkg/vm"
|
"github.com/nspcc-dev/neo-go/pkg/vm"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/wallet"
|
"github.com/nspcc-dev/neo-go/pkg/wallet"
|
||||||
"github.com/urfave/cli"
|
"github.com/urfave/cli"
|
||||||
"gopkg.in/yaml.v3"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -88,6 +86,7 @@ func NewCommands() []cli.Command {
|
||||||
txctx.SysGasFlag,
|
txctx.SysGasFlag,
|
||||||
txctx.OutFlag,
|
txctx.OutFlag,
|
||||||
txctx.ForceFlag,
|
txctx.ForceFlag,
|
||||||
|
txctx.AwaitFlag,
|
||||||
flags.AddressFlag{
|
flags.AddressFlag{
|
||||||
Name: "address, a",
|
Name: "address, a",
|
||||||
Usage: "Address to claim GAS for",
|
Usage: "Address to claim GAS for",
|
||||||
|
@ -98,6 +97,7 @@ func NewCommands() []cli.Command {
|
||||||
walletPathFlag,
|
walletPathFlag,
|
||||||
walletConfigFlag,
|
walletConfigFlag,
|
||||||
txctx.OutFlag,
|
txctx.OutFlag,
|
||||||
|
txctx.AwaitFlag,
|
||||||
inFlag,
|
inFlag,
|
||||||
flags.AddressFlag{
|
flags.AddressFlag{
|
||||||
Name: "address, a",
|
Name: "address, a",
|
||||||
|
@ -112,7 +112,7 @@ func NewCommands() []cli.Command {
|
||||||
{
|
{
|
||||||
Name: "claim",
|
Name: "claim",
|
||||||
Usage: "claim GAS",
|
Usage: "claim GAS",
|
||||||
UsageText: "neo-go wallet claim -w wallet [--wallet-config path] [-g gas] [-e sysgas] -a address -r endpoint [-s timeout] [--out file] [--force]",
|
UsageText: "neo-go wallet claim -w wallet [--wallet-config path] [-g gas] [-e sysgas] -a address -r endpoint [-s timeout] [--out file] [--force] [--await]",
|
||||||
Action: claimGas,
|
Action: claimGas,
|
||||||
Flags: claimFlags,
|
Flags: claimFlags,
|
||||||
},
|
},
|
||||||
|
@ -236,8 +236,15 @@ func NewCommands() []cli.Command {
|
||||||
{
|
{
|
||||||
Name: "import-multisig",
|
Name: "import-multisig",
|
||||||
Usage: "import multisig contract",
|
Usage: "import multisig contract",
|
||||||
UsageText: "import-multisig -w wallet [--wallet-config path] --wif <wif> [--name <account_name>] --min <n>" +
|
UsageText: "import-multisig -w wallet [--wallet-config path] [--wif <wif>] [--name <account_name>] --min <m>" +
|
||||||
" [<pubkey1> [<pubkey2> [...]]]",
|
" [<pubkey1> [<pubkey2> [...]]]",
|
||||||
|
Description: `Imports a standard multisignature contract with "m out of n" signatures required where "m" is
|
||||||
|
specified by --min flag and "n" is the length of provided set of public keys. If
|
||||||
|
--wif flag is provided, it's used to create an account with the given name (or
|
||||||
|
without a name if --name flag is not provided). Otherwise, the command tries to
|
||||||
|
find an account with one of the given public keys and convert it to multisig. If
|
||||||
|
no suitable account is found and no --wif flag is specified, an error is returned.
|
||||||
|
`,
|
||||||
Action: importMultisig,
|
Action: importMultisig,
|
||||||
Flags: []cli.Flag{
|
Flags: []cli.Flag{
|
||||||
walletPathFlag,
|
walletPathFlag,
|
||||||
|
@ -290,13 +297,15 @@ func NewCommands() []cli.Command {
|
||||||
{
|
{
|
||||||
Name: "sign",
|
Name: "sign",
|
||||||
Usage: "cosign transaction with multisig/contract/additional account",
|
Usage: "cosign transaction with multisig/contract/additional account",
|
||||||
UsageText: "sign -w wallet [--wallet-config path] --address <address> --in <file.in> [--out <file.out>] [-r <endpoint>]",
|
UsageText: "sign -w wallet [--wallet-config path] --address <address> --in <file.in> [--out <file.out>] [-r <endpoint>] [--await]",
|
||||||
Description: `Signs the given (in file.in) context (which must be a transaction
|
Description: `Signs the given (in file.in) context (which must be a transaction
|
||||||
signing context) for the given address using the given wallet. This command can
|
signing context) for the given address using the given wallet. This command can
|
||||||
output the resulting JSON (with additional signature added) right to the console
|
output the resulting JSON (with additional signature added) right to the console
|
||||||
(if no file.out and no RPC endpoint specified) or into a file (which can be the
|
(if no file.out and no RPC endpoint specified) or into a file (which can be the
|
||||||
same as input one). If an RPC endpoint is given it'll also try to construct a
|
same as input one). If an RPC endpoint is given it'll also try to construct a
|
||||||
complete transaction and send it via RPC (printing its hash if everything is OK).
|
complete transaction and send it via RPC (printing its hash if everything is OK).
|
||||||
|
If the --await (with a given RPC endpoint) flag is included, the command waits
|
||||||
|
for the transaction to be included in a block before exiting.
|
||||||
`,
|
`,
|
||||||
Action: signStoredTransaction,
|
Action: signStoredTransaction,
|
||||||
Flags: signFlags,
|
Flags: signFlags,
|
||||||
|
@ -519,6 +528,12 @@ loop:
|
||||||
}
|
}
|
||||||
|
|
||||||
func importMultisig(ctx *cli.Context) error {
|
func importMultisig(ctx *cli.Context) error {
|
||||||
|
var (
|
||||||
|
label *string
|
||||||
|
acc *wallet.Account
|
||||||
|
accPub *keys.PublicKey
|
||||||
|
)
|
||||||
|
|
||||||
wall, pass, err := openWallet(ctx, true)
|
wall, pass, err := openWallet(ctx, true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(err, 1)
|
return cli.NewExitError(err, 1)
|
||||||
|
@ -540,12 +555,45 @@ func importMultisig(ctx *cli.Context) error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var label *string
|
|
||||||
if ctx.IsSet("name") {
|
if ctx.IsSet("name") {
|
||||||
l := ctx.String("name")
|
l := ctx.String("name")
|
||||||
label = &l
|
label = &l
|
||||||
}
|
}
|
||||||
acc, err := newAccountFromWIF(ctx.App.Writer, ctx.String("wif"), wall.Scrypt, label, pass)
|
|
||||||
|
loop:
|
||||||
|
for _, pub := range pubs {
|
||||||
|
for _, wallAcc := range wall.Accounts {
|
||||||
|
if wallAcc.ScriptHash().Equals(pub.GetScriptHash()) {
|
||||||
|
if acc != nil {
|
||||||
|
// Multiple matching accounts found, fallback to WIF-based conversion.
|
||||||
|
acc = nil
|
||||||
|
break loop
|
||||||
|
}
|
||||||
|
acc = new(wallet.Account)
|
||||||
|
*acc = *wallAcc
|
||||||
|
accPub = pub
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if acc != nil {
|
||||||
|
err = acc.ConvertMultisigEncrypted(accPub, m, pubs)
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(err, 1)
|
||||||
|
}
|
||||||
|
if label != nil {
|
||||||
|
acc.Label = *label
|
||||||
|
}
|
||||||
|
if err := addAccountAndSave(wall, acc); err != nil {
|
||||||
|
return cli.NewExitError(err, 1)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if !ctx.IsSet("wif") {
|
||||||
|
return cli.NewExitError(errors.New("none of the provided public keys correspond to an existing key in the wallet or multiple matching accounts found in the wallet, and no WIF is provided"), 1)
|
||||||
|
}
|
||||||
|
acc, err = newAccountFromWIF(ctx.App.Writer, ctx.String("wif"), wall.Scrypt, label, pass)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(err, 1)
|
return cli.NewExitError(err, 1)
|
||||||
}
|
}
|
||||||
|
@ -603,7 +651,8 @@ func importDeployed(ctx *cli.Context) error {
|
||||||
return cli.NewExitError("contract has no `verify` method with boolean return", 1)
|
return cli.NewExitError("contract has no `verify` method with boolean return", 1)
|
||||||
}
|
}
|
||||||
acc.Address = address.Uint160ToString(cs.Hash)
|
acc.Address = address.Uint160ToString(cs.Hash)
|
||||||
acc.Contract.Script = cs.NEF.Script
|
// Explicitly overwrite single signature script of the provided WIF since the contract is known to be deployed.
|
||||||
|
acc.Contract.Script = nil
|
||||||
acc.Contract.Parameters = acc.Contract.Parameters[:0]
|
acc.Contract.Parameters = acc.Contract.Parameters[:0]
|
||||||
for _, p := range md.Parameters {
|
for _, p := range md.Parameters {
|
||||||
acc.Contract.Parameters = append(acc.Contract.Parameters, wallet.ContractParam{
|
acc.Contract.Parameters = append(acc.Contract.Parameters, wallet.ContractParam{
|
||||||
|
@ -822,7 +871,7 @@ func createWallet(ctx *cli.Context) error {
|
||||||
}
|
}
|
||||||
var pass *string
|
var pass *string
|
||||||
if len(configPath) != 0 {
|
if len(configPath) != 0 {
|
||||||
cfg, err := ReadWalletConfig(configPath)
|
cfg, err := options.ReadWalletConfig(configPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cli.NewExitError(err, 1)
|
return cli.NewExitError(err, 1)
|
||||||
}
|
}
|
||||||
|
@ -900,14 +949,14 @@ func createAccount(wall *wallet.Wallet, pass *string) error {
|
||||||
func openWallet(ctx *cli.Context, canUseWalletConfig bool) (*wallet.Wallet, *string, error) {
|
func openWallet(ctx *cli.Context, canUseWalletConfig bool) (*wallet.Wallet, *string, error) {
|
||||||
path, pass, err := getWalletPathAndPass(ctx, canUseWalletConfig)
|
path, pass, err := getWalletPathAndPass(ctx, canUseWalletConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, err
|
return nil, nil, cli.NewExitError(fmt.Errorf("failed to get wallet path or password: %w", err), 1)
|
||||||
}
|
}
|
||||||
if path == "-" {
|
if path == "-" {
|
||||||
return nil, nil, errNoStdin
|
return nil, nil, errNoStdin
|
||||||
}
|
}
|
||||||
w, err := wallet.NewWalletFromFile(path)
|
w, err := wallet.NewWalletFromFile(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, err
|
return nil, nil, cli.NewExitError(fmt.Errorf("failed to read wallet: %w", err), 1)
|
||||||
}
|
}
|
||||||
return w, pass, nil
|
return w, pass, nil
|
||||||
}
|
}
|
||||||
|
@ -946,7 +995,7 @@ func getWalletPathAndPass(ctx *cli.Context, canUseWalletConfig bool) (string, *s
|
||||||
}
|
}
|
||||||
var pass *string
|
var pass *string
|
||||||
if len(configPath) != 0 {
|
if len(configPath) != 0 {
|
||||||
cfg, err := ReadWalletConfig(configPath)
|
cfg, err := options.ReadWalletConfig(configPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", nil, err
|
return "", nil, err
|
||||||
}
|
}
|
||||||
|
@ -956,27 +1005,6 @@ func getWalletPathAndPass(ctx *cli.Context, canUseWalletConfig bool) (string, *s
|
||||||
return path, pass, nil
|
return path, pass, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func ReadWalletConfig(configPath string) (*config.Wallet, error) {
|
|
||||||
file, err := os.Open(configPath)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
defer file.Close()
|
|
||||||
|
|
||||||
configData, err := os.ReadFile(configPath)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("unable to read wallet config: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
cfg := &config.Wallet{}
|
|
||||||
|
|
||||||
err = yaml.Unmarshal(configData, &cfg)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to unmarshal wallet config YAML: %w", err)
|
|
||||||
}
|
|
||||||
return cfg, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func newAccountFromWIF(w io.Writer, wif string, scrypt keys.ScryptParams, label *string, pass *string) (*wallet.Account, error) {
|
func newAccountFromWIF(w io.Writer, wif string, scrypt keys.ScryptParams, label *string, pass *string) (*wallet.Account, error) {
|
||||||
var (
|
var (
|
||||||
phrase, name string
|
phrase, name string
|
||||||
|
@ -985,7 +1013,12 @@ func newAccountFromWIF(w io.Writer, wif string, scrypt keys.ScryptParams, label
|
||||||
if pass != nil {
|
if pass != nil {
|
||||||
phrase = *pass
|
phrase = *pass
|
||||||
}
|
}
|
||||||
if label != nil {
|
if label == nil {
|
||||||
|
name, err = readAccountName()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to read account label: %w", err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
name = *label
|
name = *label
|
||||||
}
|
}
|
||||||
// note: NEP2 strings always have length of 58 even though
|
// note: NEP2 strings always have length of 58 even though
|
||||||
|
@ -1024,12 +1057,6 @@ func newAccountFromWIF(w io.Writer, wif string, scrypt keys.ScryptParams, label
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Fprintln(w, "Provided WIF was unencrypted. Wallet can contain only encrypted keys.")
|
fmt.Fprintln(w, "Provided WIF was unencrypted. Wallet can contain only encrypted keys.")
|
||||||
if label == nil {
|
|
||||||
name, err = readAccountName()
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to read account label: %w", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if pass == nil {
|
if pass == nil {
|
||||||
phrase, err = readNewPassword()
|
phrase, err = readNewPassword()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
package wallet_test
|
package wallet_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/hex"
|
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"math/big"
|
"math/big"
|
||||||
"os"
|
"os"
|
||||||
|
@ -356,9 +355,26 @@ func TestWalletInit(t *testing.T) {
|
||||||
t.Run("InvalidPassword", func(t *testing.T) {
|
t.Run("InvalidPassword", func(t *testing.T) {
|
||||||
e.In.WriteString("password1\r")
|
e.In.WriteString("password1\r")
|
||||||
e.RunWithError(t, "neo-go", "wallet", "import", "--wallet", walletPath,
|
e.RunWithError(t, "neo-go", "wallet", "import", "--wallet", walletPath,
|
||||||
"--wif", acc.EncryptedWIF)
|
"--wif", acc.EncryptedWIF, "--name", "acc1")
|
||||||
})
|
})
|
||||||
|
|
||||||
|
e.In.WriteString("somepass\r")
|
||||||
|
e.Run(t, "neo-go", "wallet", "import", "--wallet", walletPath,
|
||||||
|
"--wif", acc.EncryptedWIF, "--name", "acc1")
|
||||||
|
|
||||||
|
w, err := wallet.NewWalletFromFile(walletPath)
|
||||||
|
require.NoError(t, err)
|
||||||
|
actual := w.GetAccount(acc.PrivateKey().GetScriptHash())
|
||||||
|
require.NotNil(t, actual)
|
||||||
|
require.Equal(t, "acc1", actual.Label)
|
||||||
|
require.NoError(t, actual.Decrypt("somepass", w.Scrypt))
|
||||||
|
})
|
||||||
|
t.Run("EncryptedWIF with name specified via input", func(t *testing.T) {
|
||||||
|
acc, err := wallet.NewAccount()
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.NoError(t, acc.Encrypt("somepass", keys.NEP2ScryptParams()))
|
||||||
|
|
||||||
|
e.In.WriteString("acc2\r")
|
||||||
e.In.WriteString("somepass\r")
|
e.In.WriteString("somepass\r")
|
||||||
e.Run(t, "neo-go", "wallet", "import", "--wallet", walletPath,
|
e.Run(t, "neo-go", "wallet", "import", "--wallet", walletPath,
|
||||||
"--wif", acc.EncryptedWIF)
|
"--wif", acc.EncryptedWIF)
|
||||||
|
@ -367,6 +383,7 @@ func TestWalletInit(t *testing.T) {
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
actual := w.GetAccount(acc.PrivateKey().GetScriptHash())
|
actual := w.GetAccount(acc.PrivateKey().GetScriptHash())
|
||||||
require.NotNil(t, actual)
|
require.NotNil(t, actual)
|
||||||
|
require.Equal(t, "acc2", actual.Label)
|
||||||
require.NoError(t, actual.Decrypt("somepass", w.Scrypt))
|
require.NoError(t, actual.Decrypt("somepass", w.Scrypt))
|
||||||
})
|
})
|
||||||
t.Run("EncryptedWIF with wallet config", func(t *testing.T) {
|
t.Run("EncryptedWIF with wallet config", func(t *testing.T) {
|
||||||
|
@ -388,12 +405,13 @@ func TestWalletInit(t *testing.T) {
|
||||||
e.In.WriteString(pass + "\r")
|
e.In.WriteString(pass + "\r")
|
||||||
}
|
}
|
||||||
e.Run(t, "neo-go", "wallet", "import", "--wallet-config", configPath,
|
e.Run(t, "neo-go", "wallet", "import", "--wallet-config", configPath,
|
||||||
"--wif", acc.EncryptedWIF)
|
"--wif", acc.EncryptedWIF, "--name", "acc3"+configPass)
|
||||||
|
|
||||||
w, err := wallet.NewWalletFromFile(walletPath)
|
w, err := wallet.NewWalletFromFile(walletPath)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
actual := w.GetAccount(acc.PrivateKey().GetScriptHash())
|
actual := w.GetAccount(acc.PrivateKey().GetScriptHash())
|
||||||
require.NotNil(t, actual)
|
require.NotNil(t, actual)
|
||||||
|
require.Equal(t, "acc3"+configPass, actual.Label)
|
||||||
require.NoError(t, actual.Decrypt(pass, w.Scrypt))
|
require.NoError(t, actual.Decrypt(pass, w.Scrypt))
|
||||||
}
|
}
|
||||||
t.Run("config password mismatch", func(t *testing.T) {
|
t.Run("config password mismatch", func(t *testing.T) {
|
||||||
|
@ -418,16 +436,16 @@ func TestWalletInit(t *testing.T) {
|
||||||
"--wallet", walletPath,
|
"--wallet", walletPath,
|
||||||
"--min", "2"}
|
"--min", "2"}
|
||||||
t.Run("invalid pub encoding", func(t *testing.T) {
|
t.Run("invalid pub encoding", func(t *testing.T) {
|
||||||
e.RunWithError(t, append(cmd, hex.EncodeToString(pubs[1].Bytes()),
|
e.RunWithError(t, append(cmd, pubs[1].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[1].Bytes()),
|
pubs[1].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[2].Bytes()),
|
pubs[2].StringCompressed(),
|
||||||
"not-a-pub")...)
|
"not-a-pub")...)
|
||||||
})
|
})
|
||||||
t.Run("missing WIF", func(t *testing.T) {
|
t.Run("missing WIF", func(t *testing.T) {
|
||||||
e.RunWithError(t, append(cmd, hex.EncodeToString(pubs[0].Bytes()),
|
e.RunWithError(t, append(cmd, pubs[0].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[1].Bytes()),
|
pubs[1].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[2].Bytes()),
|
pubs[2].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[3].Bytes()))...)
|
pubs[3].StringCompressed())...)
|
||||||
})
|
})
|
||||||
cmd = append(cmd, "--wif", privs[0].WIF())
|
cmd = append(cmd, "--wif", privs[0].WIF())
|
||||||
t.Run("InvalidPublicKeys", func(t *testing.T) {
|
t.Run("InvalidPublicKeys", func(t *testing.T) {
|
||||||
|
@ -436,18 +454,18 @@ func TestWalletInit(t *testing.T) {
|
||||||
e.In.WriteString("multipass\r")
|
e.In.WriteString("multipass\r")
|
||||||
defer e.In.Reset()
|
defer e.In.Reset()
|
||||||
|
|
||||||
e.RunWithError(t, append(cmd, hex.EncodeToString(pubs[1].Bytes()),
|
e.RunWithError(t, append(cmd, pubs[1].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[1].Bytes()),
|
pubs[1].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[2].Bytes()),
|
pubs[2].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[3].Bytes()))...)
|
pubs[3].StringCompressed())...)
|
||||||
})
|
})
|
||||||
e.In.WriteString("multiacc\r")
|
e.In.WriteString("multiacc\r")
|
||||||
e.In.WriteString("multipass\r")
|
e.In.WriteString("multipass\r")
|
||||||
e.In.WriteString("multipass\r")
|
e.In.WriteString("multipass\r")
|
||||||
e.Run(t, append(cmd, hex.EncodeToString(pubs[0].Bytes()),
|
e.Run(t, append(cmd, pubs[0].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[1].Bytes()),
|
pubs[1].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[2].Bytes()),
|
pubs[2].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[3].Bytes()))...)
|
pubs[3].StringCompressed())...)
|
||||||
|
|
||||||
script, err := smartcontract.CreateMultiSigRedeemScript(2, pubs)
|
script, err := smartcontract.CreateMultiSigRedeemScript(2, pubs)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
@ -463,10 +481,62 @@ func TestWalletInit(t *testing.T) {
|
||||||
e.In.WriteString("multiacc\r")
|
e.In.WriteString("multiacc\r")
|
||||||
e.In.WriteString("multipass\r")
|
e.In.WriteString("multipass\r")
|
||||||
e.In.WriteString("multipass\r")
|
e.In.WriteString("multipass\r")
|
||||||
e.RunWithError(t, append(cmd, hex.EncodeToString(pubs[0].Bytes()),
|
e.RunWithError(t, append(cmd, pubs[0].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[1].Bytes()),
|
pubs[1].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[2].Bytes()),
|
pubs[2].StringCompressed(),
|
||||||
hex.EncodeToString(pubs[3].Bytes()))...)
|
pubs[3].StringCompressed())...)
|
||||||
|
})
|
||||||
|
|
||||||
|
privs, pubs = testcli.GenerateKeys(t, 3)
|
||||||
|
script, err = smartcontract.CreateMultiSigRedeemScript(2, pubs)
|
||||||
|
require.NoError(t, err)
|
||||||
|
// Create a wallet and import a standard account
|
||||||
|
e.Run(t, "neo-go", "wallet", "init", "--wallet", walletPath)
|
||||||
|
e.In.WriteString("standardacc\rstdpass\rstdpass\r")
|
||||||
|
e.Run(t, "neo-go", "wallet", "import",
|
||||||
|
"--wallet", walletPath,
|
||||||
|
"--wif", privs[0].WIF())
|
||||||
|
w, err = wallet.NewWalletFromFile(walletPath)
|
||||||
|
require.NoError(t, err)
|
||||||
|
actual = w.GetAccount(privs[0].GetScriptHash())
|
||||||
|
require.NotNil(t, actual)
|
||||||
|
require.NotEqual(t, actual.Contract.Script, script)
|
||||||
|
|
||||||
|
// Test when a public key of an already imported account is present
|
||||||
|
t.Run("existing account public key, no WIF", func(t *testing.T) {
|
||||||
|
e.Run(t, "neo-go", "wallet", "import-multisig",
|
||||||
|
"--wallet", walletPath,
|
||||||
|
"--min", "2",
|
||||||
|
pubs[0].StringCompressed(), // Public key of the already imported account
|
||||||
|
pubs[1].StringCompressed(),
|
||||||
|
pubs[2].StringCompressed())
|
||||||
|
|
||||||
|
w, err := wallet.NewWalletFromFile(walletPath)
|
||||||
|
require.NoError(t, err)
|
||||||
|
actual := w.GetAccount(hash.Hash160(script))
|
||||||
|
require.NotNil(t, actual)
|
||||||
|
require.Equal(t, actual.Contract.Script, script)
|
||||||
|
require.NoError(t, actual.Decrypt("stdpass", w.Scrypt))
|
||||||
|
require.NotEqual(t, actual.Address, w.GetAccount(privs[0].GetScriptHash()).Address)
|
||||||
|
})
|
||||||
|
|
||||||
|
// Test when no public key of an already imported account is present, and no WIF is provided
|
||||||
|
t.Run("no existing account public key, no WIF", func(t *testing.T) {
|
||||||
|
_, pubsNew := testcli.GenerateKeys(t, 3)
|
||||||
|
scriptNew, err := smartcontract.CreateMultiSigRedeemScript(2, pubsNew)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
e.RunWithError(t, "neo-go", "wallet", "import-multisig",
|
||||||
|
"--wallet", walletPath,
|
||||||
|
"--min", "2",
|
||||||
|
pubsNew[0].StringCompressed(),
|
||||||
|
pubsNew[1].StringCompressed(),
|
||||||
|
pubsNew[2].StringCompressed())
|
||||||
|
|
||||||
|
w, err := wallet.NewWalletFromFile(walletPath)
|
||||||
|
require.NoError(t, err)
|
||||||
|
actual := w.GetAccount(hash.Hash160(scriptNew))
|
||||||
|
require.Nil(t, actual)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
@ -586,6 +656,47 @@ func TestWalletClaimGas(t *testing.T) {
|
||||||
} else {
|
} else {
|
||||||
require.Equal(t, 1, balanceAfter.Cmp(balanceBefore))
|
require.Equal(t, 1, balanceAfter.Cmp(balanceBefore))
|
||||||
}
|
}
|
||||||
|
t.Run("await", func(t *testing.T) {
|
||||||
|
args := []string{
|
||||||
|
"neo-go", "wallet", "nep17", "multitransfer",
|
||||||
|
"--rpc-endpoint", "http://" + e.RPC.Addresses()[0],
|
||||||
|
"--wallet", testcli.ValidatorWallet,
|
||||||
|
"--from", testcli.ValidatorAddr, "--await",
|
||||||
|
"--force",
|
||||||
|
"NEO:" + testcli.TestWalletAccount + ":1000",
|
||||||
|
"GAS:" + testcli.TestWalletAccount + ":1000", // for tx send
|
||||||
|
}
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.Run(t, args...)
|
||||||
|
e.CheckAwaitableTxPersisted(t)
|
||||||
|
|
||||||
|
h, err := address.StringToUint160(testcli.TestWalletAccount)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
balanceBefore := e.Chain.GetUtilityTokenBalance(h)
|
||||||
|
claimHeight := e.Chain.BlockHeight() + 1
|
||||||
|
cl, err := e.Chain.CalculateClaimable(h, claimHeight)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.True(t, cl.Sign() > 0)
|
||||||
|
|
||||||
|
e.In.WriteString("testpass\r")
|
||||||
|
e.Run(t, "neo-go", "wallet", "claim",
|
||||||
|
"--rpc-endpoint", "http://"+e.RPC.Addresses()[0],
|
||||||
|
"--wallet", testcli.TestWalletPath,
|
||||||
|
"--address", testcli.TestWalletAccount,
|
||||||
|
"--force", "--await")
|
||||||
|
tx, height = e.CheckAwaitableTxPersisted(t)
|
||||||
|
balanceBefore.Sub(balanceBefore, big.NewInt(tx.NetworkFee+tx.SystemFee))
|
||||||
|
balanceBefore.Add(balanceBefore, cl)
|
||||||
|
|
||||||
|
balanceAfter = e.Chain.GetUtilityTokenBalance(h)
|
||||||
|
// height can be bigger than claimHeight especially when tests are executed with -race.
|
||||||
|
if height == claimHeight {
|
||||||
|
require.Equal(t, 0, balanceAfter.Cmp(balanceBefore))
|
||||||
|
} else {
|
||||||
|
require.Equal(t, 1, balanceAfter.Cmp(balanceBefore))
|
||||||
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestWalletImportDeployed(t *testing.T) {
|
func TestWalletImportDeployed(t *testing.T) {
|
||||||
|
@ -803,6 +914,31 @@ func TestOfflineSigning(t *testing.T) {
|
||||||
"--in", txPath)
|
"--in", txPath)
|
||||||
})
|
})
|
||||||
e.CheckTxPersisted(t)
|
e.CheckTxPersisted(t)
|
||||||
|
|
||||||
|
t.Run("await 1/1 multisig", func(t *testing.T) {
|
||||||
|
args := []string{"neo-go", "wallet", "nep17", "transfer",
|
||||||
|
"--rpc-endpoint", "http://" + e.RPC.Addresses()[0],
|
||||||
|
"--wallet", walletPath,
|
||||||
|
"--from", testcli.ValidatorAddr,
|
||||||
|
"--to", w.Accounts[0].Address,
|
||||||
|
"--token", "NEO",
|
||||||
|
"--amount", "1",
|
||||||
|
"--force",
|
||||||
|
}
|
||||||
|
e.Run(t, append(args, "--out", txPath)...)
|
||||||
|
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.Run(t, "neo-go", "wallet", "sign",
|
||||||
|
"--wallet", testcli.ValidatorWallet, "--address", testcli.ValidatorAddr,
|
||||||
|
"--in", txPath, "--out", txPath)
|
||||||
|
|
||||||
|
e.Run(t, "neo-go", "wallet", "sign",
|
||||||
|
"--rpc-endpoint", "http://"+e.RPC.Addresses()[0],
|
||||||
|
"--wallet", walletPath, "--address", testcli.ValidatorAddr,
|
||||||
|
"--in", txPath, "--await")
|
||||||
|
e.CheckAwaitableTxPersisted(t)
|
||||||
|
})
|
||||||
|
|
||||||
t.Run("simple signature", func(t *testing.T) {
|
t.Run("simple signature", func(t *testing.T) {
|
||||||
simpleAddr := w.Accounts[0].Address
|
simpleAddr := w.Accounts[0].Address
|
||||||
args := []string{"neo-go", "wallet", "nep17", "transfer",
|
args := []string{"neo-go", "wallet", "nep17", "transfer",
|
||||||
|
@ -834,6 +970,31 @@ func TestOfflineSigning(t *testing.T) {
|
||||||
"--rpc-endpoint", "http://"+e.RPC.Addresses()[0],
|
"--rpc-endpoint", "http://"+e.RPC.Addresses()[0],
|
||||||
txPath)
|
txPath)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
t.Run("await simple signature", func(t *testing.T) {
|
||||||
|
simpleAddr := w.Accounts[0].Address
|
||||||
|
args := []string{"neo-go", "wallet", "nep17", "transfer",
|
||||||
|
"--rpc-endpoint", "http://" + e.RPC.Addresses()[0],
|
||||||
|
"--wallet", walletPath,
|
||||||
|
"--from", simpleAddr,
|
||||||
|
"--to", testcli.ValidatorAddr,
|
||||||
|
"--token", "NEO",
|
||||||
|
"--amount", "1",
|
||||||
|
"--force",
|
||||||
|
}
|
||||||
|
|
||||||
|
e.Run(t, append(args, "--out", txPath)...)
|
||||||
|
|
||||||
|
e.In.WriteString("one\r")
|
||||||
|
e.Run(t, "neo-go", "wallet", "sign",
|
||||||
|
"--wallet", testcli.ValidatorWallet, "--address", simpleAddr,
|
||||||
|
"--in", txPath, "--out", txPath)
|
||||||
|
|
||||||
|
e.Run(t, "neo-go", "util", "sendtx",
|
||||||
|
"--rpc-endpoint", "http://"+e.RPC.Addresses()[0],
|
||||||
|
txPath, "--await")
|
||||||
|
e.CheckAwaitableTxPersisted(t)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestWalletDump(t *testing.T) {
|
func TestWalletDump(t *testing.T) {
|
||||||
|
|
|
@ -22,19 +22,11 @@ ProtocolConfiguration:
|
||||||
- morph6.fs.neo.org:40333
|
- morph6.fs.neo.org:40333
|
||||||
- morph7.fs.neo.org:40333
|
- morph7.fs.neo.org:40333
|
||||||
VerifyTransactions: true
|
VerifyTransactions: true
|
||||||
P2PSigExtensions: false
|
P2PSigExtensions: true
|
||||||
Hardforks:
|
Hardforks:
|
||||||
Aspidochelone: 3000000
|
Aspidochelone: 3000000
|
||||||
NativeActivations:
|
Basilisk: 4500000
|
||||||
ContractManagement: [0]
|
Cockatrice: 5800000
|
||||||
StdLib: [0]
|
|
||||||
CryptoLib: [0]
|
|
||||||
LedgerContract: [0]
|
|
||||||
NeoToken: [0]
|
|
||||||
GasToken: [0]
|
|
||||||
PolicyContract: [0]
|
|
||||||
RoleManagement: [0]
|
|
||||||
OracleContract: [0]
|
|
||||||
|
|
||||||
ApplicationConfiguration:
|
ApplicationConfiguration:
|
||||||
SkipBlockVerification: false
|
SkipBlockVerification: false
|
||||||
|
|
|
@ -37,16 +37,8 @@ ProtocolConfiguration:
|
||||||
P2PSigExtensions: false
|
P2PSigExtensions: false
|
||||||
Hardforks:
|
Hardforks:
|
||||||
Aspidochelone: 1730000
|
Aspidochelone: 1730000
|
||||||
NativeActivations:
|
Basilisk: 4120000
|
||||||
ContractManagement: [0]
|
Cockatrice: 5450000
|
||||||
StdLib: [0]
|
|
||||||
CryptoLib: [0]
|
|
||||||
LedgerContract: [0]
|
|
||||||
NeoToken: [0]
|
|
||||||
GasToken: [0]
|
|
||||||
PolicyContract: [0]
|
|
||||||
RoleManagement: [0]
|
|
||||||
OracleContract: [0]
|
|
||||||
|
|
||||||
ApplicationConfiguration:
|
ApplicationConfiguration:
|
||||||
SkipBlockVerification: false
|
SkipBlockVerification: false
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue