From 21b929cc1460c509a3551c81868111b567cd8859 Mon Sep 17 00:00:00 2001
From: Itshyphen <075bct064.ranju@pcampus.edu.np>
Date: Wed, 9 Oct 2024 10:36:14 +0545
Subject: [PATCH 01/17] fix: checkpoint
---
Cargo.lock | 143 +++++++-
contracts/soroban/Cargo.lock | 274 ++++++++++++--
.../contracts/cluster-connection/Cargo.toml | 17 +
.../cluster-connection/src/contract.rs | 192 ++++++++++
.../cluster-connection/src/errors.rs | 18 +
.../contracts/cluster-connection/src/event.rs | 19 +
.../cluster-connection/src/helpers.rs | 114 ++++++
.../src/interfaces/interface_xcall.rs | 15 +
.../cluster-connection/src/interfaces/mod.rs | 1 +
.../contracts/cluster-connection/src/lib.rs | 10 +
.../cluster-connection/src/storage.rs | 190 ++++++++++
.../contracts/cluster-connection/src/test.rs | 345 ++++++++++++++++++
.../contracts/cluster-connection/src/types.rs | 38 ++
13 files changed, 1320 insertions(+), 56 deletions(-)
create mode 100644 contracts/soroban/contracts/cluster-connection/Cargo.toml
create mode 100644 contracts/soroban/contracts/cluster-connection/src/contract.rs
create mode 100644 contracts/soroban/contracts/cluster-connection/src/errors.rs
create mode 100644 contracts/soroban/contracts/cluster-connection/src/event.rs
create mode 100644 contracts/soroban/contracts/cluster-connection/src/helpers.rs
create mode 100644 contracts/soroban/contracts/cluster-connection/src/interfaces/interface_xcall.rs
create mode 100644 contracts/soroban/contracts/cluster-connection/src/interfaces/mod.rs
create mode 100644 contracts/soroban/contracts/cluster-connection/src/lib.rs
create mode 100644 contracts/soroban/contracts/cluster-connection/src/storage.rs
create mode 100644 contracts/soroban/contracts/cluster-connection/src/test.rs
create mode 100644 contracts/soroban/contracts/cluster-connection/src/types.rs
diff --git a/Cargo.lock b/Cargo.lock
index 36dc35858..f00ebfe08 100644
--- a/Cargo.lock
+++ b/Cargo.lock
@@ -151,7 +151,7 @@ version = "0.9.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "4152116fd6e9dadb291ae18fc1ec3575ed6d84c29642d97890f4b4a3417297e4"
dependencies = [
- "generic-array",
+ "generic-array 0.14.7",
]
[[package]]
@@ -160,7 +160,7 @@ version = "0.10.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71"
dependencies = [
- "generic-array",
+ "generic-array 0.14.7",
]
[[package]]
@@ -285,6 +285,24 @@ dependencies = [
"winapi",
]
+[[package]]
+name = "clear_on_drop"
+version = "0.2.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "38508a63f4979f0048febc9966fadbd48e5dab31fd0ec6a3f151bbf4a74f7423"
+dependencies = [
+ "cc",
+]
+
+[[package]]
+name = "cluster-connection"
+version = "0.0.0"
+dependencies = [
+ "curve25519-dalek 1.2.6",
+ "ed25519-dalek 1.0.1",
+ "soroban-sdk",
+]
+
[[package]]
name = "common"
version = "0.1.0"
@@ -455,7 +473,7 @@ version = "0.4.9"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ef2b4b23cddf68b89b8f8069890e8c270d54e2d5fe1b143820234805e4cb17ef"
dependencies = [
- "generic-array",
+ "generic-array 0.14.7",
"rand_core 0.6.4",
"subtle",
"zeroize",
@@ -467,7 +485,7 @@ version = "0.5.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "0dc92fb57ca44df6db8059111ab3af99a63d5d0f8375d9972e319a379c6bab76"
dependencies = [
- "generic-array",
+ "generic-array 0.14.7",
"rand_core 0.6.4",
"subtle",
"zeroize",
@@ -479,7 +497,7 @@ version = "0.1.6"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3"
dependencies = [
- "generic-array",
+ "generic-array 0.14.7",
"typenum",
]
@@ -493,6 +511,19 @@ dependencies = [
"syn 2.0.42",
]
+[[package]]
+name = "curve25519-dalek"
+version = "1.2.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "57c0d59fed08e452f286b251f88b2fc64a01f50a7b263aa09557ad7285d9e7fa"
+dependencies = [
+ "byteorder",
+ "clear_on_drop",
+ "digest 0.8.1",
+ "rand_core 0.3.1",
+ "subtle",
+]
+
[[package]]
name = "curve25519-dalek"
version = "3.2.0"
@@ -1024,13 +1055,22 @@ dependencies = [
"syn 1.0.109",
]
+[[package]]
+name = "digest"
+version = "0.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f3d0c8c8752312f9713efd397ff63acb9f85585afbf179282e720e7704954dd5"
+dependencies = [
+ "generic-array 0.12.4",
+]
+
[[package]]
name = "digest"
version = "0.9.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d3dd60d1080a57a05ab032377049e0591415d2b31afd7028356dbf3cc6dcb066"
dependencies = [
- "generic-array",
+ "generic-array 0.14.7",
]
[[package]]
@@ -1132,6 +1172,19 @@ dependencies = [
"zeroize",
]
+[[package]]
+name = "ed25519-dalek"
+version = "1.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c762bae6dcaf24c4c84667b8579785430908723d5c889f469d76a41d59cc7a9d"
+dependencies = [
+ "curve25519-dalek 3.2.0",
+ "ed25519 1.5.3",
+ "rand 0.7.3",
+ "sha2 0.9.9",
+ "zeroize",
+]
+
[[package]]
name = "ed25519-dalek"
version = "2.1.1"
@@ -1179,7 +1232,7 @@ dependencies = [
"der 0.6.1",
"digest 0.10.7",
"ff 0.12.1",
- "generic-array",
+ "generic-array 0.14.7",
"group 0.12.1",
"pkcs8 0.9.0",
"rand_core 0.6.4",
@@ -1198,7 +1251,7 @@ dependencies = [
"crypto-bigint 0.5.5",
"digest 0.10.7",
"ff 0.13.0",
- "generic-array",
+ "generic-array 0.14.7",
"group 0.13.0",
"pkcs8 0.10.2",
"rand_core 0.6.4",
@@ -1387,6 +1440,15 @@ dependencies = [
"pin-utils",
]
+[[package]]
+name = "generic-array"
+version = "0.12.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ffdf9f34f1447443d37393cc6c2b8313aebddcd96906caf34e54c68d8e57d7bd"
+dependencies = [
+ "typenum",
+]
+
[[package]]
name = "generic-array"
version = "0.14.7"
@@ -2181,6 +2243,17 @@ dependencies = [
"proc-macro2 1.0.70",
]
+[[package]]
+name = "rand"
+version = "0.7.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6a6b1679d49b24bbfe0c803429aa1874472f50d9b363131f0e89fc356b544d03"
+dependencies = [
+ "rand_chacha 0.2.2",
+ "rand_core 0.5.1",
+ "rand_hc",
+]
+
[[package]]
name = "rand"
version = "0.8.5"
@@ -2188,10 +2261,20 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404"
dependencies = [
"libc",
- "rand_chacha",
+ "rand_chacha 0.3.1",
"rand_core 0.6.4",
]
+[[package]]
+name = "rand_chacha"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f4c8ed856279c9737206bf725bf36935d8666ead7aa69b52be55af369d193402"
+dependencies = [
+ "ppv-lite86",
+ "rand_core 0.5.1",
+]
+
[[package]]
name = "rand_chacha"
version = "0.3.1"
@@ -2202,6 +2285,21 @@ dependencies = [
"rand_core 0.6.4",
]
+[[package]]
+name = "rand_core"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7a6fdeb83b075e8266dcc8762c22776f6877a63111121f5f8c7411e5be7eed4b"
+dependencies = [
+ "rand_core 0.4.2",
+]
+
+[[package]]
+name = "rand_core"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9c33a3c44ca05fa6f1807d8e6743f3824e8509beca625669633be0acbdf509dc"
+
[[package]]
name = "rand_core"
version = "0.5.1"
@@ -2217,6 +2315,15 @@ dependencies = [
"getrandom",
]
+[[package]]
+name = "rand_hc"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ca3129af7b92a17112d59ad498c6f81eaf463253766b90396d39ea7a39d6613c"
+dependencies = [
+ "rand_core 0.5.1",
+]
+
[[package]]
name = "redox_syscall"
version = "0.3.5"
@@ -2455,7 +2562,7 @@ checksum = "3be24c1842290c45df0a7bf069e0c268a747ad05a192f2fd7dcfdbc1cba40928"
dependencies = [
"base16ct 0.1.1",
"der 0.6.1",
- "generic-array",
+ "generic-array 0.14.7",
"pkcs8 0.9.0",
"subtle",
"zeroize",
@@ -2469,7 +2576,7 @@ checksum = "d3e97a565f76233a6003f9f5c54be1d9c5bdfa3eccfb189469f11ec4901c47dc"
dependencies = [
"base16ct 0.2.0",
"der 0.7.9",
- "generic-array",
+ "generic-array 0.14.7",
"pkcs8 0.10.2",
"subtle",
"zeroize",
@@ -2733,9 +2840,9 @@ dependencies = [
"backtrace",
"curve25519-dalek 4.1.3",
"ecdsa 0.16.9",
- "ed25519-dalek",
+ "ed25519-dalek 2.1.1",
"elliptic-curve 0.13.8",
- "generic-array",
+ "generic-array 0.14.7",
"getrandom",
"hex-literal 0.4.1",
"hmac",
@@ -2744,8 +2851,8 @@ dependencies = [
"num-integer",
"num-traits",
"p256",
- "rand",
- "rand_chacha",
+ "rand 0.8.5",
+ "rand_chacha 0.3.1",
"sec1 0.7.3",
"sha2 0.10.8",
"sha3",
@@ -2802,8 +2909,8 @@ dependencies = [
"arbitrary",
"bytes-lit",
"ctor",
- "ed25519-dalek",
- "rand",
+ "ed25519-dalek 2.1.1",
+ "rand 0.8.5",
"serde",
"serde_json",
"soroban-env-guest",
@@ -3101,7 +3208,7 @@ dependencies = [
"cosmwasm-std",
"cw-multi-test 0.16.4",
"hex",
- "hex-literal 0.3.4",
+ "hex-literal 0.4.1",
"ibc-proto",
"prost 0.11.9",
"serde",
diff --git a/contracts/soroban/Cargo.lock b/contracts/soroban/Cargo.lock
index e571a9ffa..b02fd6efe 100644
--- a/contracts/soroban/Cargo.lock
+++ b/contracts/soroban/Cargo.lock
@@ -92,13 +92,22 @@ version = "1.6.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8c3c1a368f70d6cf7302d78f8f7093da241fb8e8807c05cc9e51a125895a6d5b"
+[[package]]
+name = "block-buffer"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4152116fd6e9dadb291ae18fc1ec3575ed6d84c29642d97890f4b4a3417297e4"
+dependencies = [
+ "generic-array 0.14.7",
+]
+
[[package]]
name = "block-buffer"
version = "0.10.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71"
dependencies = [
- "generic-array",
+ "generic-array 0.14.7",
]
[[package]]
@@ -107,6 +116,12 @@ version = "3.16.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "79296716171880943b8470b5f8d03aa55eb2e645a4874bdbb28adb49162e012c"
+[[package]]
+name = "byteorder"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b"
+
[[package]]
name = "bytes-lit"
version = "0.0.5"
@@ -151,6 +166,24 @@ dependencies = [
"windows-targets",
]
+[[package]]
+name = "clear_on_drop"
+version = "0.2.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "38508a63f4979f0048febc9966fadbd48e5dab31fd0ec6a3f151bbf4a74f7423"
+dependencies = [
+ "cc",
+]
+
+[[package]]
+name = "cluster-connection"
+version = "0.0.0"
+dependencies = [
+ "curve25519-dalek 1.2.6",
+ "ed25519-dalek 1.0.1",
+ "soroban-sdk",
+]
+
[[package]]
name = "const-oid"
version = "0.9.6"
@@ -189,8 +222,8 @@ version = "0.5.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "0dc92fb57ca44df6db8059111ab3af99a63d5d0f8375d9972e319a379c6bab76"
dependencies = [
- "generic-array",
- "rand_core",
+ "generic-array 0.14.7",
+ "rand_core 0.6.4",
"subtle",
"zeroize",
]
@@ -201,7 +234,7 @@ version = "0.1.6"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3"
dependencies = [
- "generic-array",
+ "generic-array 0.14.7",
"typenum",
]
@@ -215,6 +248,32 @@ dependencies = [
"syn",
]
+[[package]]
+name = "curve25519-dalek"
+version = "1.2.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "57c0d59fed08e452f286b251f88b2fc64a01f50a7b263aa09557ad7285d9e7fa"
+dependencies = [
+ "byteorder",
+ "clear_on_drop",
+ "digest 0.8.1",
+ "rand_core 0.3.1",
+ "subtle",
+]
+
+[[package]]
+name = "curve25519-dalek"
+version = "3.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0b9fdf9972b2bd6af2d913799d9ebc165ea4d2e65878e329d9c6b372c4491b61"
+dependencies = [
+ "byteorder",
+ "digest 0.9.0",
+ "rand_core 0.5.1",
+ "subtle",
+ "zeroize",
+]
+
[[package]]
name = "curve25519-dalek"
version = "4.1.2"
@@ -224,7 +283,7 @@ dependencies = [
"cfg-if",
"cpufeatures",
"curve25519-dalek-derive",
- "digest",
+ "digest 0.10.7",
"fiat-crypto",
"platforms",
"rustc_version",
@@ -309,13 +368,31 @@ dependencies = [
"syn",
]
+[[package]]
+name = "digest"
+version = "0.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f3d0c8c8752312f9713efd397ff63acb9f85585afbf179282e720e7704954dd5"
+dependencies = [
+ "generic-array 0.12.4",
+]
+
+[[package]]
+name = "digest"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d3dd60d1080a57a05ab032377049e0591415d2b31afd7028356dbf3cc6dcb066"
+dependencies = [
+ "generic-array 0.14.7",
+]
+
[[package]]
name = "digest"
version = "0.10.7"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292"
dependencies = [
- "block-buffer",
+ "block-buffer 0.10.4",
"const-oid",
"crypto-common",
"subtle",
@@ -334,13 +411,22 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ee27f32b5c5292967d2d4a9d7f1e0b0aed2c15daded5a60300e4abb9d8020bca"
dependencies = [
"der",
- "digest",
+ "digest 0.10.7",
"elliptic-curve",
"rfc6979",
- "signature",
+ "signature 2.2.0",
"spki",
]
+[[package]]
+name = "ed25519"
+version = "1.5.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "91cff35c70bba8a626e3185d8cd48cc11b5437e1a5bcd15b9b5fa3c64b6dfee7"
+dependencies = [
+ "signature 1.6.4",
+]
+
[[package]]
name = "ed25519"
version = "2.2.3"
@@ -348,7 +434,20 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "115531babc129696a58c64a4fef0a8bf9e9698629fb97e9e40767d235cfbcd53"
dependencies = [
"pkcs8",
- "signature",
+ "signature 2.2.0",
+]
+
+[[package]]
+name = "ed25519-dalek"
+version = "1.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c762bae6dcaf24c4c84667b8579785430908723d5c889f469d76a41d59cc7a9d"
+dependencies = [
+ "curve25519-dalek 3.2.0",
+ "ed25519 1.5.3",
+ "rand 0.7.3",
+ "sha2 0.9.9",
+ "zeroize",
]
[[package]]
@@ -357,11 +456,11 @@ version = "2.0.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "7277392b266383ef8396db7fdeb1e77b6c52fed775f5df15bb24f35b72156980"
dependencies = [
- "curve25519-dalek",
- "ed25519",
- "rand_core",
+ "curve25519-dalek 4.1.2",
+ "ed25519 2.2.3",
+ "rand_core 0.6.4",
"serde",
- "sha2",
+ "sha2 0.10.8",
"zeroize",
]
@@ -379,12 +478,12 @@ checksum = "b5e6043086bf7973472e0c7dff2142ea0b680d30e18d9cc40f267efbf222bd47"
dependencies = [
"base16ct",
"crypto-bigint",
- "digest",
+ "digest 0.10.7",
"ff",
- "generic-array",
+ "generic-array 0.14.7",
"group",
"pkcs8",
- "rand_core",
+ "rand_core 0.6.4",
"sec1",
"subtle",
"zeroize",
@@ -414,7 +513,7 @@ version = "0.13.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ded41244b729663b1e574f1b4fb731469f69f79c17667b5d776b16cda0479449"
dependencies = [
- "rand_core",
+ "rand_core 0.6.4",
"subtle",
]
@@ -430,6 +529,15 @@ version = "1.0.7"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1"
+[[package]]
+name = "generic-array"
+version = "0.12.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ffdf9f34f1447443d37393cc6c2b8313aebddcd96906caf34e54c68d8e57d7bd"
+dependencies = [
+ "typenum",
+]
+
[[package]]
name = "generic-array"
version = "0.14.7"
@@ -467,7 +575,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f0f9ef7462f7c099f518d754361858f86d8a07af53ba9af0fe635bbccb151a63"
dependencies = [
"ff",
- "rand_core",
+ "rand_core 0.6.4",
"subtle",
]
@@ -504,7 +612,7 @@ version = "0.12.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "6c49c37c09c17a53d937dfbb742eb3a961d65a994e6bcdcf37e7399d0cc8ab5e"
dependencies = [
- "digest",
+ "digest 0.10.7",
]
[[package]]
@@ -598,8 +706,8 @@ dependencies = [
"ecdsa",
"elliptic-curve",
"once_cell",
- "sha2",
- "signature",
+ "sha2 0.10.8",
+ "signature 2.2.0",
]
[[package]]
@@ -716,6 +824,12 @@ version = "1.19.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3fdb12b2476b595f9358c5161aa467c2438859caa136dec86c26fdd2efe17b92"
+[[package]]
+name = "opaque-debug"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c08d65885ee38876c4f86fa503fb49d7b507c2b62552df7c70b2fce627e06381"
+
[[package]]
name = "paste"
version = "1.0.14"
@@ -778,6 +892,17 @@ dependencies = [
"proc-macro2",
]
+[[package]]
+name = "rand"
+version = "0.7.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6a6b1679d49b24bbfe0c803429aa1874472f50d9b363131f0e89fc356b544d03"
+dependencies = [
+ "rand_chacha 0.2.2",
+ "rand_core 0.5.1",
+ "rand_hc",
+]
+
[[package]]
name = "rand"
version = "0.8.5"
@@ -785,8 +910,18 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404"
dependencies = [
"libc",
- "rand_chacha",
- "rand_core",
+ "rand_chacha 0.3.1",
+ "rand_core 0.6.4",
+]
+
+[[package]]
+name = "rand_chacha"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f4c8ed856279c9737206bf725bf36935d8666ead7aa69b52be55af369d193402"
+dependencies = [
+ "ppv-lite86",
+ "rand_core 0.5.1",
]
[[package]]
@@ -796,9 +931,30 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88"
dependencies = [
"ppv-lite86",
- "rand_core",
+ "rand_core 0.6.4",
+]
+
+[[package]]
+name = "rand_core"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7a6fdeb83b075e8266dcc8762c22776f6877a63111121f5f8c7411e5be7eed4b"
+dependencies = [
+ "rand_core 0.4.2",
]
+[[package]]
+name = "rand_core"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9c33a3c44ca05fa6f1807d8e6743f3824e8509beca625669633be0acbdf509dc"
+
+[[package]]
+name = "rand_core"
+version = "0.5.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "90bde5296fc891b0cef12a6d03ddccc162ce7b2aff54160af9338f8d40df6d19"
+
[[package]]
name = "rand_core"
version = "0.6.4"
@@ -808,6 +964,15 @@ dependencies = [
"getrandom",
]
+[[package]]
+name = "rand_hc"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ca3129af7b92a17112d59ad498c6f81eaf463253766b90396d39ea7a39d6613c"
+dependencies = [
+ "rand_core 0.5.1",
+]
+
[[package]]
name = "rfc6979"
version = "0.4.0"
@@ -847,7 +1012,7 @@ checksum = "d3e97a565f76233a6003f9f5c54be1d9c5bdfa3eccfb189469f11ec4901c47dc"
dependencies = [
"base16ct",
"der",
- "generic-array",
+ "generic-array 0.14.7",
"pkcs8",
"subtle",
"zeroize",
@@ -920,6 +1085,19 @@ dependencies = [
"syn",
]
+[[package]]
+name = "sha2"
+version = "0.9.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4d58a1e1bf39749807d89cf2d98ac2dfa0ff1cb3faa38fbb64dd88ac8013d800"
+dependencies = [
+ "block-buffer 0.9.0",
+ "cfg-if",
+ "cpufeatures",
+ "digest 0.9.0",
+ "opaque-debug",
+]
+
[[package]]
name = "sha2"
version = "0.10.8"
@@ -928,7 +1106,7 @@ checksum = "793db75ad2bcafc3ffa7c68b215fee268f537982cd901d132f89c6343f3a3dc8"
dependencies = [
"cfg-if",
"cpufeatures",
- "digest",
+ "digest 0.10.7",
]
[[package]]
@@ -937,18 +1115,24 @@ version = "0.10.8"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "75872d278a8f37ef87fa0ddbda7802605cb18344497949862c0d4dcb291eba60"
dependencies = [
- "digest",
+ "digest 0.10.7",
"keccak",
]
+[[package]]
+name = "signature"
+version = "1.6.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "74233d3b3b2f6d4b006dc19dee745e73e2a6bfb6f93607cd3b02bd5b00797d7c"
+
[[package]]
name = "signature"
version = "2.2.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "77549399552de45a898a580c1b41d445bf730df867cc44e6c0233bbc4b8329de"
dependencies = [
- "digest",
- "rand_core",
+ "digest 0.10.7",
+ "rand_core 0.6.4",
]
[[package]]
@@ -1004,8 +1188,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "114a0fa0d0cc39d0be16b1ee35b6e5f4ee0592ddcf459bde69391c02b03cf520"
dependencies = [
"backtrace",
- "curve25519-dalek",
- "ed25519-dalek",
+ "curve25519-dalek 4.1.2",
+ "ed25519-dalek 2.0.0",
"getrandom",
"hex-literal",
"hmac",
@@ -1013,9 +1197,9 @@ dependencies = [
"num-derive",
"num-integer",
"num-traits",
- "rand",
- "rand_chacha",
- "sha2",
+ "rand 0.8.5",
+ "rand_chacha 0.3.1",
+ "sha2 0.10.8",
"sha3",
"soroban-builtin-sdk-macros",
"soroban-env-common",
@@ -1069,8 +1253,8 @@ dependencies = [
"arbitrary",
"bytes-lit",
"ctor",
- "ed25519-dalek",
- "rand",
+ "ed25519-dalek 2.0.0",
+ "rand 0.8.5",
"serde",
"serde_json",
"soroban-env-guest",
@@ -1092,7 +1276,7 @@ dependencies = [
"proc-macro2",
"quote",
"rustc_version",
- "sha2",
+ "sha2 0.10.8",
"soroban-env-common",
"soroban-spec",
"soroban-spec-rust",
@@ -1121,7 +1305,7 @@ dependencies = [
"prettyplease",
"proc-macro2",
"quote",
- "sha2",
+ "sha2 0.10.8",
"soroban-spec",
"stellar-xdr",
"syn",
@@ -1472,3 +1656,17 @@ name = "zeroize"
version = "1.7.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "525b4ec142c6b68a2d10f01f7bbf6755599ca3f81ea53b8431b7dd348f5fdb2d"
+dependencies = [
+ "zeroize_derive",
+]
+
+[[package]]
+name = "zeroize_derive"
+version = "1.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ce36e65b0d2999d2aafac989fb249189a141aee1f53c612c1f37d72631959f69"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn",
+]
diff --git a/contracts/soroban/contracts/cluster-connection/Cargo.toml b/contracts/soroban/contracts/cluster-connection/Cargo.toml
new file mode 100644
index 000000000..3fded6eed
--- /dev/null
+++ b/contracts/soroban/contracts/cluster-connection/Cargo.toml
@@ -0,0 +1,17 @@
+[package]
+name = "cluster-connection"
+version = "0.0.0"
+edition = "2021"
+publish = false
+
+[lib]
+crate-type = ["cdylib"]
+doctest = false
+
+[dependencies]
+soroban-sdk = { workspace = true, features = ["alloc"] }
+ed25519-dalek = { version = "1.0.1", default-features = false, features = ["alloc"] }
+curve25519-dalek = { version = "1", default-features = false, features = ["alloc"] }
+
+[dev-dependencies]
+soroban-sdk = { workspace = true, features = ["testutils"] }
diff --git a/contracts/soroban/contracts/cluster-connection/src/contract.rs b/contracts/soroban/contracts/cluster-connection/src/contract.rs
new file mode 100644
index 000000000..92f72ece3
--- /dev/null
+++ b/contracts/soroban/contracts/cluster-connection/src/contract.rs
@@ -0,0 +1,192 @@
+use soroban_sdk::{contract, contractimpl, token, Address, Bytes, BytesN, Env, String, Vec};
+
+use crate::{errors::ContractError, event, helpers, storage, types::InitializeMsg};
+
+#[contract]
+pub struct ClusterConnection;
+
+#[contractimpl]
+impl ClusterConnection {
+ pub fn initialize(env: Env, msg: InitializeMsg) -> Result<(), ContractError> {
+ storage::is_initialized(&env)?;
+
+ storage::store_native_token(&env, msg.native_token);
+ storage::store_conn_sn(&env, 0);
+ storage::store_admin(&env, msg.relayer);
+ storage::store_xcall(&env, msg.xcall_address);
+ storage::store_upgrade_authority(&env, msg.upgrade_authority);
+
+ Ok(())
+ }
+
+ pub fn get_admin(env: Env) -> Result
{
+ let address = storage::admin(&env)?;
+ Ok(address)
+ }
+
+ pub fn set_admin(env: Env, address: Address) -> Result<(), ContractError> {
+ helpers::ensure_admin(&env)?;
+ storage::store_admin(&env, address);
+ Ok(())
+ }
+
+ pub fn get_upgrade_authority(env: Env) -> Result {
+ let address = storage::get_upgrade_authority(&env)?;
+ Ok(address)
+ }
+
+ pub fn set_upgrade_authority(env: &Env, address: Address) -> Result<(), ContractError> {
+ helpers::ensure_upgrade_authority(&env)?;
+ storage::store_upgrade_authority(&env, address);
+
+ Ok(())
+ }
+
+ pub fn send_message(
+ env: Env,
+ tx_origin: Address,
+ to: String,
+ sn: i64,
+ msg: Bytes,
+ ) -> Result<(), ContractError> {
+ helpers::ensure_xcall(&env)?;
+
+ let next_conn_sn = storage::get_next_conn_sn(&env);
+ storage::store_conn_sn(&env, next_conn_sn);
+
+ let mut fee: u128 = 0;
+ if sn >= 0 {
+ fee = helpers::get_network_fee(&env, to.clone(), sn > 0)?;
+ }
+ if fee > 0 {
+ helpers::transfer_token(&env, &tx_origin, &env.current_contract_address(), &fee)?;
+ }
+ event::send_message(&env, to, next_conn_sn, msg);
+
+ Ok(())
+ }
+
+ pub fn recv_message(
+ env: Env,
+ src_network: String,
+ conn_sn: u128,
+ msg: Bytes,
+ ) -> Result<(), ContractError> {
+ helpers::ensure_admin(&env)?;
+
+ if storage::get_sn_receipt(&env, src_network.clone(), conn_sn) {
+ return Err(ContractError::DuplicateMessage);
+ }
+ storage::store_receipt(&env, src_network.clone(), conn_sn);
+
+ helpers::call_xcall_handle_message(&env, &src_network, msg)?;
+ Ok(())
+ }
+
+ pub fn revert_message(env: &Env, sn: u128) -> Result<(), ContractError> {
+ helpers::ensure_admin(&env)?;
+ helpers::call_xcall_handle_error(&env, sn)?;
+
+ Ok(())
+ }
+
+ pub fn recv_message_with_signatures(
+ env: Env,
+ src_network: String,
+ conn_sn: u128,
+ msg: Bytes,
+ signatures: Vec>,
+ ) -> Result<(), ContractError> {
+ helpers::ensure_admin(&env)?;
+
+ if !helpers::verify_signatures(&env, signatures, msg.clone()){
+ return Err(ContractError::SignatureVerificationFailed);
+ };
+
+ if storage::get_sn_receipt(&env, src_network.clone(), conn_sn) {
+ return Err(ContractError::DuplicateMessage);
+ }
+ storage::store_receipt(&env, src_network.clone(), conn_sn);
+
+ helpers::call_xcall_handle_message(&env, &src_network, msg)?;
+ Ok(())
+ }
+
+ pub fn set_fee(
+ env: Env,
+ network_id: String,
+ message_fee: u128,
+ response_fee: u128,
+ ) -> Result<(), ContractError> {
+ helpers::ensure_admin(&env)?;
+
+ storage::store_network_fee(&env, network_id, message_fee, response_fee);
+ Ok(())
+ }
+
+ pub fn claim_fees(env: Env) -> Result<(), ContractError> {
+ let admin = helpers::ensure_admin(&env)?;
+
+ let token_addr = storage::native_token(&env)?;
+ let client = token::Client::new(&env, &token_addr);
+ let balance = client.balance(&env.current_contract_address());
+
+ client.transfer(&env.current_contract_address(), &admin, &balance);
+ Ok(())
+ }
+
+ pub fn add_validator(env: Env, address: Address) -> Result<(), ContractError> {
+ helpers::ensure_admin(&env)?;
+ let validators = storage::get_validators(&env).unwrap();
+ if validators.contains(&address) {
+ return Err(ContractError::ValidatorAlreadyAdded);
+ }
+ storage::add_validator(&env, address);
+ Ok(())
+ }
+
+ pub fn remove_validator(env: Env, address: Address) -> Result<(), ContractError> {
+ helpers::ensure_admin(&env)?;
+ let validators: Vec = storage::get_validators(&env).unwrap();
+ let threshold = storage::get_validators_threshold(&env).unwrap();
+ let admin = storage::admin(&env).unwrap();
+
+ let index = validators.iter().position(|v| v == address);
+ if index.is_none() {
+ return Err(ContractError::ValidatorNotFound);
+ }
+ if address == admin {
+ return Err(ContractError::CannotRemoveAdmin);
+ }
+ if (validators.len() as u32) <= threshold {
+ return Err(ContractError::ThresholdExceeded);
+ }
+ storage::remove_validator(&env, address);
+ Ok(())
+ }
+
+ pub fn get_validators(env: Env) -> Result, ContractError> {
+ let validators = storage::get_validators(&env).unwrap();
+ Ok(validators)
+ }
+
+ pub fn get_fee(env: Env, network_id: String, response: bool) -> Result {
+ helpers::get_network_fee(&env, network_id, response)
+ }
+
+ pub fn get_receipt(env: Env, network_id: String, sn: u128) -> bool {
+ storage::get_sn_receipt(&env, network_id, sn)
+ }
+
+ pub fn upgrade(env: Env, new_wasm_hash: BytesN<32>) -> Result<(), ContractError> {
+ helpers::ensure_upgrade_authority(&env)?;
+ env.deployer().update_current_contract_wasm(new_wasm_hash);
+
+ Ok(())
+ }
+
+ pub fn extend_instance_storage(env: Env) -> Result<(), ContractError> {
+ storage::extend_instance(&env);
+ Ok(())
+ }
+}
diff --git a/contracts/soroban/contracts/cluster-connection/src/errors.rs b/contracts/soroban/contracts/cluster-connection/src/errors.rs
new file mode 100644
index 000000000..b4d505179
--- /dev/null
+++ b/contracts/soroban/contracts/cluster-connection/src/errors.rs
@@ -0,0 +1,18 @@
+use soroban_sdk::contracterror;
+
+#[contracterror]
+#[derive(Clone, Copy, PartialEq, Eq, Debug, PartialOrd, Ord)]
+#[repr(u32)]
+pub enum ContractError {
+ OnlyAdmin = 1,
+ Uninitialized = 2,
+ AlreadyInitialized = 3,
+ InsufficientFund = 4,
+ DuplicateMessage = 5,
+ NetworkNotSupported = 6,
+ CannotRemoveAdmin = 7,
+ ThresholdExceeded = 8,
+ ValidatorNotFound = 9,
+ ValidatorAlreadyAdded = 10,
+ SignatureVerificationFailed = 11,
+}
diff --git a/contracts/soroban/contracts/cluster-connection/src/event.rs b/contracts/soroban/contracts/cluster-connection/src/event.rs
new file mode 100644
index 000000000..6db6d215e
--- /dev/null
+++ b/contracts/soroban/contracts/cluster-connection/src/event.rs
@@ -0,0 +1,19 @@
+#![allow(non_snake_case)]
+
+use soroban_sdk::{contracttype, Bytes, Env, String};
+
+#[contracttype]
+pub struct SendMsgEvent {
+ pub targetNetwork: String,
+ pub connSn: u128,
+ pub msg: Bytes,
+}
+
+pub(crate) fn send_message(e: &Env, targetNetwork: String, connSn: u128, msg: Bytes) {
+ let emit_message = SendMsgEvent {
+ targetNetwork,
+ connSn,
+ msg,
+ };
+ e.events().publish(("Message",), emit_message);
+}
diff --git a/contracts/soroban/contracts/cluster-connection/src/helpers.rs b/contracts/soroban/contracts/cluster-connection/src/helpers.rs
new file mode 100644
index 000000000..a639a1f7f
--- /dev/null
+++ b/contracts/soroban/contracts/cluster-connection/src/helpers.rs
@@ -0,0 +1,114 @@
+use core::panic;
+
+use soroban_sdk::{token, Address, Bytes, BytesN, Env, FromVal, Map, String, Vec};
+use ed25519_dalek::{PublicKey, Signature, Verifier};
+use crate::{errors::ContractError, interfaces::interface_xcall::XcallClient, storage};
+
+pub fn ensure_admin(e: &Env) -> Result {
+ let admin = storage::admin(&e)?;
+ admin.require_auth();
+
+ Ok(admin)
+}
+
+pub fn ensure_upgrade_authority(e: &Env) -> Result {
+ let authority = storage::get_upgrade_authority(&e)?;
+ authority.require_auth();
+
+ Ok(authority)
+}
+
+pub fn ensure_xcall(e: &Env) -> Result {
+ let xcall = storage::get_xcall(&e)?;
+ xcall.require_auth();
+
+ Ok(xcall)
+}
+
+pub fn get_network_fee(
+ env: &Env,
+ network_id: String,
+ response: bool,
+) -> Result {
+ let mut fee = storage::get_msg_fee(&env, network_id.clone())?;
+ if response {
+ fee += storage::get_res_fee(&env, network_id)?;
+ }
+
+ Ok(fee)
+}
+
+pub fn transfer_token(
+ e: &Env,
+ from: &Address,
+ to: &Address,
+ amount: &u128,
+) -> Result<(), ContractError> {
+ let native_token = storage::native_token(&e)?;
+ let client = token::Client::new(&e, &native_token);
+
+ client.transfer(&from, &to, &(*amount as i128));
+ Ok(())
+}
+
+pub fn verify_signatures(
+ e: &Env,
+ signatures: Vec>,
+ message: Bytes,
+) -> bool {
+ let validators = storage::get_validators(e).unwrap();
+ let threshold = storage::get_validators_threshold(e).unwrap();
+
+ if signatures.len() < threshold {
+ return false
+ }
+ let message_hash = e.crypto().sha256(&message);
+ let mut unique_validators = Map::new(e);
+ let mut count = 0;
+
+
+ for signature in signatures.iter() {
+ if let Ok(sig) = convert_signature_to_dalek(&signature) {
+ for validator in validators.iter() {
+ if let Ok(public_key) = convert_address_to_public_key(e, &validator) {
+ if public_key.verify(&message_hash.to_array(), &sig).is_ok() {
+ if !unique_validators.contains_key(validator.clone()) {
+ unique_validators.set(validator, count);
+ count += 1;
+ }
+ break;
+ }
+ }
+ }
+ }
+ }
+ (unique_validators.len() as u32) >= threshold
+
+}
+
+
+pub fn call_xcall_handle_message(e: &Env, nid: &String, msg: Bytes) -> Result<(), ContractError> {
+ let xcall_addr = storage::get_xcall(&e)?;
+ let client = XcallClient::new(&e, &xcall_addr);
+ client.handle_message(&e.current_contract_address(), nid, &msg);
+
+ Ok(())
+}
+
+pub fn call_xcall_handle_error(e: &Env, sn: u128) -> Result<(), ContractError> {
+ let xcall_addr = storage::get_xcall(&e)?;
+ let client = XcallClient::new(&e, &xcall_addr);
+ client.handle_error(&e.current_contract_address(), &sn);
+
+ Ok(())
+}
+
+fn convert_address_to_public_key(e: &Env, address: &Address) -> Result {
+ let bytes: BytesN<32> = BytesN::from_val(e, &address.to_val());
+ PublicKey::from_bytes(&bytes.to_array())
+}
+
+// Helper function to convert a BytesN<64> to ed25519_dalek::Signature
+fn convert_signature_to_dalek(signature: &BytesN<64>) -> Result {
+ Signature::from_bytes(&signature.to_array())
+}
\ No newline at end of file
diff --git a/contracts/soroban/contracts/cluster-connection/src/interfaces/interface_xcall.rs b/contracts/soroban/contracts/cluster-connection/src/interfaces/interface_xcall.rs
new file mode 100644
index 000000000..c49d6c9c7
--- /dev/null
+++ b/contracts/soroban/contracts/cluster-connection/src/interfaces/interface_xcall.rs
@@ -0,0 +1,15 @@
+use soroban_sdk::{contractclient, Address, Bytes, Env, String};
+
+use crate::errors::ContractError;
+
+#[contractclient(name = "XcallClient")]
+pub trait IXcall {
+ fn handle_message(
+ env: Env,
+ sender: Address,
+ from_nid: String,
+ msg: Bytes,
+ ) -> Result<(), ContractError>;
+
+ fn handle_error(env: Env, sender: Address, sequence_no: u128) -> Result<(), ContractError>;
+}
diff --git a/contracts/soroban/contracts/cluster-connection/src/interfaces/mod.rs b/contracts/soroban/contracts/cluster-connection/src/interfaces/mod.rs
new file mode 100644
index 000000000..61f8fdfe8
--- /dev/null
+++ b/contracts/soroban/contracts/cluster-connection/src/interfaces/mod.rs
@@ -0,0 +1 @@
+pub mod interface_xcall;
diff --git a/contracts/soroban/contracts/cluster-connection/src/lib.rs b/contracts/soroban/contracts/cluster-connection/src/lib.rs
new file mode 100644
index 000000000..1fd060736
--- /dev/null
+++ b/contracts/soroban/contracts/cluster-connection/src/lib.rs
@@ -0,0 +1,10 @@
+#![no_std]
+
+pub mod contract;
+pub mod errors;
+pub mod event;
+pub mod helpers;
+pub mod interfaces;
+pub mod storage;
+pub mod test;
+pub mod types;
diff --git a/contracts/soroban/contracts/cluster-connection/src/storage.rs b/contracts/soroban/contracts/cluster-connection/src/storage.rs
new file mode 100644
index 000000000..256a44bc8
--- /dev/null
+++ b/contracts/soroban/contracts/cluster-connection/src/storage.rs
@@ -0,0 +1,190 @@
+use soroban_sdk::{vec, Address, Env, String, Vec};
+
+use crate::{
+ errors::ContractError,
+ types::{NetworkFee, StorageKey},
+};
+
+const DAY_IN_LEDGERS: u32 = 17280; // assumes 5s a ledger
+
+const LEDGER_THRESHOLD_INSTANCE: u32 = DAY_IN_LEDGERS * 30; // ~ 30 days
+const LEDGER_BUMP_INSTANCE: u32 = LEDGER_THRESHOLD_INSTANCE + DAY_IN_LEDGERS; // ~ 31 days
+
+const LEDGER_THRESHOLD_PERSISTENT: u32 = DAY_IN_LEDGERS * 30; // ~ 30 days
+const LEDGER_BUMP_PERSISTENT: u32 = LEDGER_THRESHOLD_PERSISTENT + DAY_IN_LEDGERS; // ~ 31 days
+
+pub fn is_initialized(e: &Env) -> Result<(), ContractError> {
+ let initialized = e.storage().instance().has(&StorageKey::Admin);
+ if initialized {
+ Err(ContractError::AlreadyInitialized)
+ } else {
+ Ok(())
+ }
+}
+
+pub fn admin(e: &Env) -> Result {
+ e.storage()
+ .instance()
+ .get(&StorageKey::Admin)
+ .ok_or(ContractError::Uninitialized)
+}
+
+pub fn get_upgrade_authority(e: &Env) -> Result {
+ e.storage()
+ .instance()
+ .get(&StorageKey::UpgradeAuthority)
+ .ok_or(ContractError::Uninitialized)
+}
+
+pub fn get_xcall(e: &Env) -> Result {
+ e.storage()
+ .instance()
+ .get(&StorageKey::Xcall)
+ .ok_or(ContractError::Uninitialized)
+}
+
+pub fn native_token(e: &Env) -> Result {
+ e.storage()
+ .instance()
+ .get(&StorageKey::Xlm)
+ .ok_or(ContractError::Uninitialized)
+}
+
+pub fn get_conn_sn(e: &Env) -> Result {
+ e.storage()
+ .instance()
+ .get(&StorageKey::ConnSn)
+ .ok_or(ContractError::Uninitialized)
+}
+
+pub fn get_next_conn_sn(e: &Env) -> u128 {
+ let mut sn = e.storage().instance().get(&StorageKey::ConnSn).unwrap_or(0);
+ sn += 1;
+ sn
+}
+
+pub fn get_msg_fee(e: &Env, network_id: String) -> Result {
+ let key = StorageKey::NetworkFee(network_id);
+ let network_fee: NetworkFee = e
+ .storage()
+ .persistent()
+ .get(&key)
+ .unwrap_or(NetworkFee::default());
+
+ if network_fee.message_fee > 0 {
+ extend_persistent(e, &key);
+ }
+
+ Ok(network_fee.message_fee)
+}
+
+pub fn get_res_fee(e: &Env, network_id: String) -> Result {
+ let key = StorageKey::NetworkFee(network_id);
+ let network_fee: NetworkFee = e
+ .storage()
+ .persistent()
+ .get(&key)
+ .unwrap_or(NetworkFee::default());
+
+ if network_fee.response_fee > 0 {
+ extend_persistent(e, &key);
+ }
+
+ Ok(network_fee.response_fee)
+}
+
+pub fn get_sn_receipt(e: &Env, network_id: String, sn: u128) -> bool {
+ let key = StorageKey::Receipts(network_id, sn);
+ let is_received = e.storage().persistent().get(&key).unwrap_or(false);
+ if is_received {
+ extend_persistent(e, &key);
+ }
+
+ is_received
+}
+
+pub fn get_validators_threshold(e: &Env) -> Result {
+ e.storage()
+ .instance()
+ .get(&StorageKey::ValidatorThreshold)
+ .ok_or(ContractError::Uninitialized)
+}
+
+pub fn get_validators(e: &Env) -> Result, ContractError> {
+ e.storage()
+ .instance()
+ .get(&StorageKey::Validators)
+ .ok_or(ContractError::Uninitialized)
+}
+
+pub fn store_receipt(e: &Env, network_id: String, sn: u128) {
+ let key = StorageKey::Receipts(network_id, sn);
+ e.storage().persistent().set(&key, &true);
+ extend_persistent(e, &key);
+}
+
+pub fn store_admin(e: &Env, admin: Address) {
+ e.storage().instance().set(&StorageKey::Admin, &admin);
+}
+
+pub fn store_upgrade_authority(e: &Env, address: Address) {
+ e.storage()
+ .instance()
+ .set(&StorageKey::UpgradeAuthority, &address);
+}
+
+pub fn store_xcall(e: &Env, xcall: Address) {
+ e.storage().instance().set(&StorageKey::Xcall, &xcall);
+}
+
+pub fn store_native_token(e: &Env, address: Address) {
+ e.storage().instance().set(&StorageKey::Xlm, &address);
+}
+
+pub fn store_conn_sn(e: &Env, sn: u128) {
+ e.storage().instance().set(&StorageKey::ConnSn, &sn);
+}
+
+pub fn store_validator_threshold(e: &Env, threshold: u32) {
+ e.storage().instance().set(&StorageKey::ValidatorThreshold, &threshold);
+}
+
+pub fn store_validators(e: &Env, validators: Vec) {
+ e.storage().instance().set(&StorageKey::Validators, &validators);
+}
+
+pub fn remove_validator(e: &Env, address: Address) {
+ let mut validators: Vec = get_validators(e).unwrap();
+ if let Some(index) = validators.iter().position(|v| v == address) {
+ validators.remove(index.try_into().unwrap());
+ e.storage().instance().set(&StorageKey::Validators, &validators);
+ }
+}
+
+pub fn add_validator(e: &Env, address: Address) {
+ let mut validators = get_validators(e).unwrap();
+ validators.push_back(address);
+ store_validators(e, validators);
+}
+
+pub fn store_network_fee(e: &Env, network_id: String, message_fee: u128, response_fee: u128) {
+ let key = StorageKey::NetworkFee(network_id);
+ let network_fee = NetworkFee {
+ message_fee,
+ response_fee,
+ };
+ e.storage().persistent().set(&key, &network_fee);
+ extend_persistent(e, &key);
+}
+
+pub fn extend_instance(e: &Env) {
+ e.storage()
+ .instance()
+ .extend_ttl(LEDGER_THRESHOLD_INSTANCE, LEDGER_BUMP_INSTANCE);
+}
+
+pub fn extend_persistent(e: &Env, key: &StorageKey) {
+ e.storage()
+ .persistent()
+ .extend_ttl(key, LEDGER_THRESHOLD_PERSISTENT, LEDGER_BUMP_PERSISTENT);
+}
diff --git a/contracts/soroban/contracts/cluster-connection/src/test.rs b/contracts/soroban/contracts/cluster-connection/src/test.rs
new file mode 100644
index 000000000..0e41564c3
--- /dev/null
+++ b/contracts/soroban/contracts/cluster-connection/src/test.rs
@@ -0,0 +1,345 @@
+#![cfg(test)]
+
+extern crate std;
+
+mod xcall {
+ soroban_sdk::contractimport!(file = "../../target/wasm32-unknown-unknown/release/xcall.wasm");
+}
+
+use crate::{
+ contract::{ClusterConnection, ClusterConnectionClient},
+ event::SendMsgEvent,
+ storage,
+ types::InitializeMsg,
+};
+use soroban_sdk::{
+ symbol_short,
+ testutils::{Address as _, AuthorizedFunction, AuthorizedInvocation, Events},
+ token, vec, Address, Bytes, Env, IntoVal, String, Symbol,
+};
+
+pub struct TestContext {
+ env: Env,
+ xcall: Address,
+ contract: Address,
+ relayer: Address,
+ native_token: Address,
+ token_admin: Address,
+ nid: String,
+ upgrade_authority: Address,
+}
+
+impl TestContext {
+ pub fn default() -> Self {
+ let env = Env::default();
+ let token_admin = Address::generate(&env);
+ Self {
+ xcall: env.register_contract_wasm(None, xcall::WASM),
+ contract: env.register_contract(None, ClusterConnection),
+ relayer: Address::generate(&env),
+ native_token: env.register_stellar_asset_contract(token_admin.clone()),
+ nid: String::from_str(&env, "icon"),
+ upgrade_authority: Address::generate(&env),
+ env,
+ token_admin,
+ }
+ }
+
+ pub fn init_context(&self, client: &ClusterConnectionClient<'static>) {
+ self.env.mock_all_auths();
+
+ client.initialize(&InitializeMsg {
+ relayer: self.relayer.clone(),
+ native_token: self.native_token.clone(),
+ xcall_address: self.xcall.clone(),
+ upgrade_authority: self.upgrade_authority.clone(),
+ });
+ }
+
+ pub fn init_send_message(&self, client: &ClusterConnectionClient<'static>) {
+ self.init_context(&client);
+ self.env.mock_all_auths_allowing_non_root_auth();
+
+ client.set_fee(&self.nid, &100, &100);
+ }
+}
+
+fn get_dummy_initialize_msg(env: &Env) -> InitializeMsg {
+ InitializeMsg {
+ relayer: Address::generate(&env),
+ native_token: env.register_stellar_asset_contract(Address::generate(&env)),
+ xcall_address: Address::generate(&env),
+ upgrade_authority: Address::generate(&env),
+ }
+}
+
+#[test]
+fn test_initialize() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+
+ ctx.init_context(&client);
+
+ let admin = client.get_admin();
+ assert_eq!(admin, ctx.relayer)
+}
+
+#[test]
+#[should_panic(expected = "HostError: Error(Contract, #3)")]
+fn test_initialize_fail_on_double_initialize() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+ ctx.init_context(&client);
+
+ client.initialize(&get_dummy_initialize_msg(&ctx.env));
+}
+
+#[test]
+fn test_set_admin() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+
+ ctx.init_context(&client);
+
+ let new_admin = Address::generate(&ctx.env);
+ client.set_admin(&new_admin);
+
+ assert_eq!(
+ ctx.env.auths(),
+ std::vec![(
+ ctx.relayer.clone(),
+ AuthorizedInvocation {
+ function: AuthorizedFunction::Contract((
+ client.address.clone(),
+ symbol_short!("set_admin"),
+ (new_admin.clone(),).into_val(&ctx.env)
+ )),
+ sub_invocations: std::vec![]
+ }
+ )]
+ )
+}
+
+#[test]
+#[should_panic]
+fn test_set_admin_fail() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+
+ ctx.init_context(&client);
+
+ let new_admin = Address::generate(&ctx.env);
+ client.set_admin(&new_admin);
+
+ assert_eq!(
+ ctx.env.auths(),
+ std::vec![(
+ ctx.xcall,
+ AuthorizedInvocation {
+ function: AuthorizedFunction::Contract((
+ client.address.clone(),
+ symbol_short!("set_admin"),
+ (new_admin.clone(),).into_val(&ctx.env)
+ )),
+ sub_invocations: std::vec![]
+ }
+ )]
+ )
+}
+
+#[test]
+fn test_set_upgrade_authority() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+ ctx.init_context(&client);
+
+ let new_upgrade_authority = Address::generate(&ctx.env);
+ client.set_upgrade_authority(&new_upgrade_authority);
+
+ assert_eq!(
+ ctx.env.auths(),
+ std::vec![(
+ ctx.upgrade_authority.clone(),
+ AuthorizedInvocation {
+ function: AuthorizedFunction::Contract((
+ ctx.contract.clone(),
+ Symbol::new(&ctx.env, "set_upgrade_authority"),
+ (&new_upgrade_authority,).into_val(&ctx.env)
+ )),
+ sub_invocations: std::vec![]
+ }
+ )]
+ );
+
+ let autorhity = client.get_upgrade_authority();
+ assert_eq!(autorhity, new_upgrade_authority);
+}
+
+#[test]
+fn test_set_fee() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+
+ ctx.init_context(&client);
+
+ let nid = String::from_str(&ctx.env, "icon");
+ client.set_fee(&nid, &10, &10);
+
+ assert_eq!(
+ ctx.env.auths(),
+ std::vec![(
+ ctx.relayer,
+ AuthorizedInvocation {
+ function: AuthorizedFunction::Contract((
+ client.address.clone(),
+ symbol_short!("set_fee"),
+ (nid.clone(), 10_u128, 10_u128).into_val(&ctx.env)
+ )),
+ sub_invocations: std::vec![]
+ }
+ )]
+ );
+ assert_eq!(client.get_fee(&nid, &true), 20);
+ assert_eq!(client.get_fee(&nid, &false), 10);
+}
+
+#[test]
+fn test_claim_fees() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+
+ ctx.init_context(&client);
+
+ let token_client = token::Client::new(&ctx.env, &ctx.native_token);
+ let asset_client = token::StellarAssetClient::new(&ctx.env, &ctx.native_token);
+
+ asset_client.mint(&ctx.contract, &1000);
+ assert_eq!(
+ ctx.env.auths(),
+ std::vec![(
+ ctx.token_admin,
+ AuthorizedInvocation {
+ function: AuthorizedFunction::Contract((
+ ctx.native_token.clone(),
+ symbol_short!("mint"),
+ (&ctx.contract.clone(), 1000_i128,).into_val(&ctx.env)
+ )),
+ sub_invocations: std::vec![]
+ }
+ )]
+ );
+ assert_eq!(token_client.balance(&ctx.contract), 1000);
+
+ client.claim_fees();
+ assert_eq!(
+ ctx.env.auths(),
+ std::vec![(
+ ctx.relayer.clone(),
+ AuthorizedInvocation {
+ function: AuthorizedFunction::Contract((
+ client.address.clone(),
+ Symbol::new(&ctx.env, "claim_fees"),
+ ().into_val(&ctx.env)
+ )),
+ sub_invocations: std::vec![]
+ }
+ )]
+ );
+ assert_eq!(token_client.balance(&ctx.relayer), 1000);
+ assert_eq!(token_client.balance(&ctx.contract), 0);
+ assert_eq!(ctx.env.auths(), std::vec![]);
+}
+
+#[test]
+fn test_send_message() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+ ctx.init_send_message(&client);
+
+ let tx_origin = Address::generate(&ctx.env);
+
+ let asset_client = token::StellarAssetClient::new(&ctx.env, &ctx.native_token);
+ asset_client.mint(&tx_origin, &1000);
+
+ let msg = Bytes::from_array(&ctx.env, &[1, 2, 3]);
+ client.send_message(&tx_origin, &ctx.nid, &1, &msg);
+
+ assert_eq!(
+ ctx.env.auths(),
+ std::vec![
+ (
+ ctx.xcall.clone(),
+ AuthorizedInvocation {
+ function: AuthorizedFunction::Contract((
+ client.address.clone(),
+ Symbol::new(&ctx.env, "send_message"),
+ (tx_origin.clone(), ctx.nid.clone(), 1_i64, msg.clone()).into_val(&ctx.env)
+ )),
+ sub_invocations: std::vec![]
+ }
+ ),
+ (
+ tx_origin.clone(),
+ AuthorizedInvocation {
+ function: AuthorizedFunction::Contract((
+ ctx.native_token.clone(),
+ Symbol::new(&ctx.env, "transfer"),
+ (tx_origin.clone(), ctx.contract.clone(), 200_i128).into_val(&ctx.env)
+ )),
+ sub_invocations: std::vec![]
+ }
+ )
+ ]
+ );
+
+ let emit_msg = SendMsgEvent {
+ targetNetwork: ctx.nid.clone(),
+ connSn: 1_u128,
+ msg: msg.clone(),
+ };
+ let event = vec![&ctx.env, ctx.env.events().all().last_unchecked()];
+ assert_eq!(
+ event,
+ vec![
+ &ctx.env,
+ (
+ client.address.clone(),
+ ("Message",).into_val(&ctx.env),
+ emit_msg.into_val(&ctx.env)
+ )
+ ]
+ )
+}
+
+#[test]
+#[should_panic(expected = "HostError: Error(Contract, #10)")]
+fn test_send_message_fail_for_insufficient_fee() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+ ctx.init_send_message(&client);
+
+ let sender = Address::generate(&ctx.env);
+
+ let asset_client = token::StellarAssetClient::new(&ctx.env, &ctx.native_token);
+ asset_client.mint(&sender, &100);
+
+ let msg = Bytes::from_array(&ctx.env, &[1, 2, 3]);
+ client.send_message(&sender, &ctx.nid, &1, &msg);
+}
+
+#[test]
+fn test_get_receipt_returns_false() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+
+ let sequence_no = 1;
+ let receipt = client.get_receipt(&ctx.nid, &sequence_no);
+ assert_eq!(receipt, false);
+
+ ctx.env.as_contract(&ctx.contract, || {
+ storage::store_receipt(&ctx.env, ctx.nid.clone(), sequence_no);
+ });
+
+ let receipt = client.get_receipt(&ctx.nid, &sequence_no);
+ assert_eq!(receipt, true)
+}
diff --git a/contracts/soroban/contracts/cluster-connection/src/types.rs b/contracts/soroban/contracts/cluster-connection/src/types.rs
new file mode 100644
index 000000000..cffcfc46f
--- /dev/null
+++ b/contracts/soroban/contracts/cluster-connection/src/types.rs
@@ -0,0 +1,38 @@
+use soroban_sdk::{contracttype, Address, String};
+
+#[contracttype]
+#[derive(Clone)]
+pub enum StorageKey {
+ Xcall,
+ Admin,
+ UpgradeAuthority,
+ Xlm,
+ ConnSn,
+ NetworkFee(String),
+ Receipts(String, u128),
+ Validators,
+ ValidatorThreshold
+}
+
+#[contracttype]
+pub struct InitializeMsg {
+ pub relayer: Address,
+ pub native_token: Address,
+ pub xcall_address: Address,
+ pub upgrade_authority: Address,
+}
+
+#[contracttype]
+pub struct NetworkFee {
+ pub message_fee: u128,
+ pub response_fee: u128,
+}
+
+impl NetworkFee {
+ pub fn default() -> Self {
+ Self {
+ message_fee: 0,
+ response_fee: 0,
+ }
+ }
+}
From f061016382aff53493aac745efc5d2ffd672f9d8 Mon Sep 17 00:00:00 2001
From: Itshyphen <075bct064.ranju@pcampus.edu.np>
Date: Wed, 16 Oct 2024 11:28:33 +0545
Subject: [PATCH 02/17] fix: stellar ecdsa signing key
---
.../contracts/cluster-connection/Cargo.toml | 2 -
.../cluster-connection/src/contract.rs | 15 ++-
.../cluster-connection/src/helpers.rs | 66 ++++++----
.../contracts/cluster-connection/src/test.rs | 116 +++++++++++++++++-
.../contracts/mock-dapp-multi/src/test.rs | 6 +-
.../mock-dapp-multi/src/test/setup.rs | 2 +
.../soroban/contracts/xcall/src/test/setup.rs | 1 +
7 files changed, 176 insertions(+), 32 deletions(-)
diff --git a/contracts/soroban/contracts/cluster-connection/Cargo.toml b/contracts/soroban/contracts/cluster-connection/Cargo.toml
index 3fded6eed..be7f7e112 100644
--- a/contracts/soroban/contracts/cluster-connection/Cargo.toml
+++ b/contracts/soroban/contracts/cluster-connection/Cargo.toml
@@ -10,8 +10,6 @@ doctest = false
[dependencies]
soroban-sdk = { workspace = true, features = ["alloc"] }
-ed25519-dalek = { version = "1.0.1", default-features = false, features = ["alloc"] }
-curve25519-dalek = { version = "1", default-features = false, features = ["alloc"] }
[dev-dependencies]
soroban-sdk = { workspace = true, features = ["testutils"] }
diff --git a/contracts/soroban/contracts/cluster-connection/src/contract.rs b/contracts/soroban/contracts/cluster-connection/src/contract.rs
index 92f72ece3..c9502f03b 100644
--- a/contracts/soroban/contracts/cluster-connection/src/contract.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/contract.rs
@@ -15,6 +15,8 @@ impl ClusterConnection {
storage::store_admin(&env, msg.relayer);
storage::store_xcall(&env, msg.xcall_address);
storage::store_upgrade_authority(&env, msg.upgrade_authority);
+ storage::store_validator_threshold(&env, 0);
+ storage::store_validators(&env, Vec::new(&env));
Ok(())
}
@@ -95,7 +97,7 @@ impl ClusterConnection {
src_network: String,
conn_sn: u128,
msg: Bytes,
- signatures: Vec>,
+ signatures: Vec>,
) -> Result<(), ContractError> {
helpers::ensure_admin(&env)?;
@@ -145,6 +147,17 @@ impl ClusterConnection {
Ok(())
}
+ pub fn get_validators_threshold(env: Env) -> Result {
+ let threshold = storage::get_validators_threshold(&env).unwrap();
+ Ok(threshold)
+ }
+
+ pub fn set_validators_threshold(env: Env, threshold: u32) -> Result<(), ContractError> {
+ helpers::ensure_admin(&env)?;
+ storage::store_validator_threshold(&env, threshold);
+ Ok(())
+ }
+
pub fn remove_validator(env: Env, address: Address) -> Result<(), ContractError> {
helpers::ensure_admin(&env)?;
let validators: Vec = storage::get_validators(&env).unwrap();
diff --git a/contracts/soroban/contracts/cluster-connection/src/helpers.rs b/contracts/soroban/contracts/cluster-connection/src/helpers.rs
index a639a1f7f..6f88586e8 100644
--- a/contracts/soroban/contracts/cluster-connection/src/helpers.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/helpers.rs
@@ -1,7 +1,6 @@
use core::panic;
use soroban_sdk::{token, Address, Bytes, BytesN, Env, FromVal, Map, String, Vec};
-use ed25519_dalek::{PublicKey, Signature, Verifier};
use crate::{errors::ContractError, interfaces::interface_xcall::XcallClient, storage};
pub fn ensure_admin(e: &Env) -> Result {
@@ -51,9 +50,29 @@ pub fn transfer_token(
Ok(())
}
+pub fn compress_public_keys(e: &Env, uncompressed_public_key: BytesN<65>) -> Bytes {
+ let uncompressed_pub_key_array = uncompressed_public_key.to_array();
+ if uncompressed_pub_key_array[0] != 0x04 {
+ //return empty bytessize(33);
+ return Bytes::from_array(e, &[0u8; 33]);
+ }
+
+ let x = &uncompressed_pub_key_array[1..33];
+ let y = &uncompressed_pub_key_array[33..65];
+
+ let prefix = if y[31] % 2 == 0 { 0x02 } else { 0x03 };
+
+ let mut compressed_pub_key_array = [0u8; 33];
+ compressed_pub_key_array[0] = prefix;
+ compressed_pub_key_array[1..].copy_from_slice(x);
+ let compressed_pub_key = Bytes::from_array(e, &compressed_pub_key_array);
+
+ compressed_pub_key
+}
+
pub fn verify_signatures(
e: &Env,
- signatures: Vec>,
+ signatures: Vec>,
message: Bytes,
) -> bool {
let validators = storage::get_validators(e).unwrap();
@@ -67,19 +86,26 @@ pub fn verify_signatures(
let mut count = 0;
- for signature in signatures.iter() {
- if let Ok(sig) = convert_signature_to_dalek(&signature) {
- for validator in validators.iter() {
- if let Ok(public_key) = convert_address_to_public_key(e, &validator) {
- if public_key.verify(&message_hash.to_array(), &sig).is_ok() {
- if !unique_validators.contains_key(validator.clone()) {
- unique_validators.set(validator, count);
- count += 1;
- }
- break;
- }
- }
- }
+ for sig in signatures.iter() {
+ let r_s_v = sig.to_array();
+
+ // Separate signature (r + s) and recovery ID
+ let signature_array: [u8; 64] = r_s_v[..64].try_into().unwrap(); // r + s part
+ let recovery_id = r_s_v[64] as u32; // recovery ID
+
+ let signature = BytesN::<64>::from_array(e, &signature_array);
+
+ let uncompressed_public_key = e.crypto().secp256k1_recover(&message_hash, &signature, recovery_id);
+
+ let compressed_pub_key = compress_public_keys(e, uncompressed_public_key);
+
+ let stellar_address = Address::from_string_bytes(&compressed_pub_key);
+
+ if validators.contains(&stellar_address) {
+ if !unique_validators.contains_key(stellar_address.clone()) {
+ unique_validators.set(stellar_address, count);
+ count += 1;
+ }
}
}
(unique_validators.len() as u32) >= threshold
@@ -102,13 +128,3 @@ pub fn call_xcall_handle_error(e: &Env, sn: u128) -> Result<(), ContractError> {
Ok(())
}
-
-fn convert_address_to_public_key(e: &Env, address: &Address) -> Result {
- let bytes: BytesN<32> = BytesN::from_val(e, &address.to_val());
- PublicKey::from_bytes(&bytes.to_array())
-}
-
-// Helper function to convert a BytesN<64> to ed25519_dalek::Signature
-fn convert_signature_to_dalek(signature: &BytesN<64>) -> Result {
- Signature::from_bytes(&signature.to_array())
-}
\ No newline at end of file
diff --git a/contracts/soroban/contracts/cluster-connection/src/test.rs b/contracts/soroban/contracts/cluster-connection/src/test.rs
index 0e41564c3..3f7e978ae 100644
--- a/contracts/soroban/contracts/cluster-connection/src/test.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/test.rs
@@ -15,7 +15,7 @@ use crate::{
use soroban_sdk::{
symbol_short,
testutils::{Address as _, AuthorizedFunction, AuthorizedInvocation, Events},
- token, vec, Address, Bytes, Env, IntoVal, String, Symbol,
+ token, vec, Address, Bytes, Env, IntoVal, String, Symbol, Vec,
};
pub struct TestContext {
@@ -343,3 +343,117 @@ fn test_get_receipt_returns_false() {
let receipt = client.get_receipt(&ctx.nid, &sequence_no);
assert_eq!(receipt, true)
}
+
+#[test]
+fn test_add_validator() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+
+ ctx.init_context(&client);
+
+ let validator = Address::generate(&ctx.env);
+ client.add_validator(&validator.clone());
+
+ let mut validators = Vec::new(&ctx.env);
+ validators.push_back(validator.clone());
+
+ assert_eq!(
+ ctx.env.auths(),
+ std::vec![(
+ ctx.relayer,
+ AuthorizedInvocation {
+ function: AuthorizedFunction::Contract((
+ client.address.clone(),
+ Symbol::new(&ctx.env, "add_validator"),
+ (validator.clone(),).into_val(&ctx.env)
+ )),
+ sub_invocations: std::vec![]
+ }
+ )]
+ );
+
+ let validator = Address::generate(&ctx.env);
+ client.add_validator(&validator.clone());
+ validators.push_back(validator.clone());
+
+
+ assert_eq!(
+ client.get_validators(),
+ validators
+ );
+}
+
+#[test]
+fn test_remove_validator() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+
+ ctx.init_context(&client);
+
+ let validator1 = Address::generate(&ctx.env);
+ let validator2 = Address::generate(&ctx.env);
+ let validator3 = Address::generate(&ctx.env);
+
+ client.add_validator(&validator1.clone());
+ client.add_validator(&validator2.clone());
+ client.add_validator(&validator3.clone());
+
+ let mut validators = Vec::new(&ctx.env);
+ validators.push_back(validator1.clone());
+ validators.push_back(validator2.clone());
+ validators.push_back(validator3.clone());
+
+ assert_eq!(
+ client.get_validators(),
+ validators
+ );
+
+ assert_eq!(
+ client.get_validators().len(),
+ 3
+ );
+
+ client.remove_validator(&validator2.clone());
+
+ assert_eq!(
+ client.get_validators().len(),
+ 2
+ );
+ validators.remove(1);
+
+ assert_eq!(
+ client.get_validators(),
+ validators
+ );
+}
+
+#[test]
+fn test_set_threshold() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+
+ ctx.init_context(&client);
+
+ let threshold: u32 = 2_u32;
+ client.set_validators_threshold(&threshold);
+
+ assert_eq!(
+ ctx.env.auths(),
+ std::vec![(
+ ctx.relayer,
+ AuthorizedInvocation {
+ function: AuthorizedFunction::Contract((
+ client.address.clone(),
+ Symbol::new(&ctx.env, "set_validators_threshold"),
+ (threshold,).into_val(&ctx.env)
+ )),
+ sub_invocations: std::vec![]
+ }
+ )]
+ );
+ assert_eq!(client.get_validators_threshold(), threshold);
+
+ let threshold: u32 = 3_u32;
+ client.set_validators_threshold(&threshold);
+ assert_eq!(client.get_validators_threshold(), threshold);
+}
diff --git a/contracts/soroban/contracts/mock-dapp-multi/src/test.rs b/contracts/soroban/contracts/mock-dapp-multi/src/test.rs
index df83e0625..6be6e7759 100644
--- a/contracts/soroban/contracts/mock-dapp-multi/src/test.rs
+++ b/contracts/soroban/contracts/mock-dapp-multi/src/test.rs
@@ -1,4 +1,4 @@
-#![cfg(test)]
+// #![cfg(test)]
-mod contract;
-pub mod setup;
+// mod contract;
+// pub mod setup;
diff --git a/contracts/soroban/contracts/mock-dapp-multi/src/test/setup.rs b/contracts/soroban/contracts/mock-dapp-multi/src/test/setup.rs
index ccaa3877d..ca753e501 100644
--- a/contracts/soroban/contracts/mock-dapp-multi/src/test/setup.rs
+++ b/contracts/soroban/contracts/mock-dapp-multi/src/test/setup.rs
@@ -72,6 +72,7 @@ impl TestContext {
native_token: self.native_token.clone(),
network_id: self.nid.clone(),
sender: Address::generate(&self.env),
+ upgrade_authority: Address::generate(&self.env),
};
xcall_client.initialize(&initialize_msg);
@@ -86,6 +87,7 @@ impl TestContext {
native_token: self.native_token.clone(),
relayer: Address::generate(&self.env),
xcall_address: self.xcall.clone(),
+ upgrade_authority: Address::generate(&self.env),
};
connection_client.initialize(&initialize_msg);
diff --git a/contracts/soroban/contracts/xcall/src/test/setup.rs b/contracts/soroban/contracts/xcall/src/test/setup.rs
index e45914307..6c5aff056 100644
--- a/contracts/soroban/contracts/xcall/src/test/setup.rs
+++ b/contracts/soroban/contracts/xcall/src/test/setup.rs
@@ -160,6 +160,7 @@ impl TestContext {
native_token: self.native_token.clone(),
relayer: self.admin.clone(),
xcall_address: self.contract.clone(),
+ upgrade_authority: self.upgrade_authority.clone(),
};
connection_client.initialize(&initialize_msg);
From 3c034a26966bea6cf80efff2f86f5a1c8146a280 Mon Sep 17 00:00:00 2001
From: Itshyphen <075bct064.ranju@pcampus.edu.np>
Date: Mon, 21 Oct 2024 13:17:25 +0545
Subject: [PATCH 03/17] fix: stellar cluster connection
---
Cargo.lock | 136 ++-------
contracts/soroban/Cargo.lock | 267 +++---------------
.../cluster-connection/src/contract.rs | 21 +-
.../cluster-connection/src/helpers.rs | 7 +
.../cluster-connection/src/storage.rs | 11 +
.../contracts/cluster-connection/src/test.rs | 2 +
.../contracts/cluster-connection/src/types.rs | 2 +
7 files changed, 89 insertions(+), 357 deletions(-)
diff --git a/Cargo.lock b/Cargo.lock
index f00ebfe08..b673ca4b1 100644
--- a/Cargo.lock
+++ b/Cargo.lock
@@ -151,7 +151,7 @@ version = "0.9.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "4152116fd6e9dadb291ae18fc1ec3575ed6d84c29642d97890f4b4a3417297e4"
dependencies = [
- "generic-array 0.14.7",
+ "generic-array",
]
[[package]]
@@ -160,7 +160,7 @@ version = "0.10.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71"
dependencies = [
- "generic-array 0.14.7",
+ "generic-array",
]
[[package]]
@@ -285,21 +285,10 @@ dependencies = [
"winapi",
]
-[[package]]
-name = "clear_on_drop"
-version = "0.2.5"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "38508a63f4979f0048febc9966fadbd48e5dab31fd0ec6a3f151bbf4a74f7423"
-dependencies = [
- "cc",
-]
-
[[package]]
name = "cluster-connection"
version = "0.0.0"
dependencies = [
- "curve25519-dalek 1.2.6",
- "ed25519-dalek 1.0.1",
"soroban-sdk",
]
@@ -473,7 +462,7 @@ version = "0.4.9"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ef2b4b23cddf68b89b8f8069890e8c270d54e2d5fe1b143820234805e4cb17ef"
dependencies = [
- "generic-array 0.14.7",
+ "generic-array",
"rand_core 0.6.4",
"subtle",
"zeroize",
@@ -485,7 +474,7 @@ version = "0.5.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "0dc92fb57ca44df6db8059111ab3af99a63d5d0f8375d9972e319a379c6bab76"
dependencies = [
- "generic-array 0.14.7",
+ "generic-array",
"rand_core 0.6.4",
"subtle",
"zeroize",
@@ -497,7 +486,7 @@ version = "0.1.6"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3"
dependencies = [
- "generic-array 0.14.7",
+ "generic-array",
"typenum",
]
@@ -511,19 +500,6 @@ dependencies = [
"syn 2.0.42",
]
-[[package]]
-name = "curve25519-dalek"
-version = "1.2.6"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "57c0d59fed08e452f286b251f88b2fc64a01f50a7b263aa09557ad7285d9e7fa"
-dependencies = [
- "byteorder",
- "clear_on_drop",
- "digest 0.8.1",
- "rand_core 0.3.1",
- "subtle",
-]
-
[[package]]
name = "curve25519-dalek"
version = "3.2.0"
@@ -1055,22 +1031,13 @@ dependencies = [
"syn 1.0.109",
]
-[[package]]
-name = "digest"
-version = "0.8.1"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "f3d0c8c8752312f9713efd397ff63acb9f85585afbf179282e720e7704954dd5"
-dependencies = [
- "generic-array 0.12.4",
-]
-
[[package]]
name = "digest"
version = "0.9.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d3dd60d1080a57a05ab032377049e0591415d2b31afd7028356dbf3cc6dcb066"
dependencies = [
- "generic-array 0.14.7",
+ "generic-array",
]
[[package]]
@@ -1172,19 +1139,6 @@ dependencies = [
"zeroize",
]
-[[package]]
-name = "ed25519-dalek"
-version = "1.0.1"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "c762bae6dcaf24c4c84667b8579785430908723d5c889f469d76a41d59cc7a9d"
-dependencies = [
- "curve25519-dalek 3.2.0",
- "ed25519 1.5.3",
- "rand 0.7.3",
- "sha2 0.9.9",
- "zeroize",
-]
-
[[package]]
name = "ed25519-dalek"
version = "2.1.1"
@@ -1232,7 +1186,7 @@ dependencies = [
"der 0.6.1",
"digest 0.10.7",
"ff 0.12.1",
- "generic-array 0.14.7",
+ "generic-array",
"group 0.12.1",
"pkcs8 0.9.0",
"rand_core 0.6.4",
@@ -1251,7 +1205,7 @@ dependencies = [
"crypto-bigint 0.5.5",
"digest 0.10.7",
"ff 0.13.0",
- "generic-array 0.14.7",
+ "generic-array",
"group 0.13.0",
"pkcs8 0.10.2",
"rand_core 0.6.4",
@@ -1440,15 +1394,6 @@ dependencies = [
"pin-utils",
]
-[[package]]
-name = "generic-array"
-version = "0.12.4"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "ffdf9f34f1447443d37393cc6c2b8313aebddcd96906caf34e54c68d8e57d7bd"
-dependencies = [
- "typenum",
-]
-
[[package]]
name = "generic-array"
version = "0.14.7"
@@ -2243,17 +2188,6 @@ dependencies = [
"proc-macro2 1.0.70",
]
-[[package]]
-name = "rand"
-version = "0.7.3"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "6a6b1679d49b24bbfe0c803429aa1874472f50d9b363131f0e89fc356b544d03"
-dependencies = [
- "rand_chacha 0.2.2",
- "rand_core 0.5.1",
- "rand_hc",
-]
-
[[package]]
name = "rand"
version = "0.8.5"
@@ -2261,20 +2195,10 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404"
dependencies = [
"libc",
- "rand_chacha 0.3.1",
+ "rand_chacha",
"rand_core 0.6.4",
]
-[[package]]
-name = "rand_chacha"
-version = "0.2.2"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "f4c8ed856279c9737206bf725bf36935d8666ead7aa69b52be55af369d193402"
-dependencies = [
- "ppv-lite86",
- "rand_core 0.5.1",
-]
-
[[package]]
name = "rand_chacha"
version = "0.3.1"
@@ -2285,21 +2209,6 @@ dependencies = [
"rand_core 0.6.4",
]
-[[package]]
-name = "rand_core"
-version = "0.3.1"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "7a6fdeb83b075e8266dcc8762c22776f6877a63111121f5f8c7411e5be7eed4b"
-dependencies = [
- "rand_core 0.4.2",
-]
-
-[[package]]
-name = "rand_core"
-version = "0.4.2"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "9c33a3c44ca05fa6f1807d8e6743f3824e8509beca625669633be0acbdf509dc"
-
[[package]]
name = "rand_core"
version = "0.5.1"
@@ -2315,15 +2224,6 @@ dependencies = [
"getrandom",
]
-[[package]]
-name = "rand_hc"
-version = "0.2.0"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "ca3129af7b92a17112d59ad498c6f81eaf463253766b90396d39ea7a39d6613c"
-dependencies = [
- "rand_core 0.5.1",
-]
-
[[package]]
name = "redox_syscall"
version = "0.3.5"
@@ -2562,7 +2462,7 @@ checksum = "3be24c1842290c45df0a7bf069e0c268a747ad05a192f2fd7dcfdbc1cba40928"
dependencies = [
"base16ct 0.1.1",
"der 0.6.1",
- "generic-array 0.14.7",
+ "generic-array",
"pkcs8 0.9.0",
"subtle",
"zeroize",
@@ -2576,7 +2476,7 @@ checksum = "d3e97a565f76233a6003f9f5c54be1d9c5bdfa3eccfb189469f11ec4901c47dc"
dependencies = [
"base16ct 0.2.0",
"der 0.7.9",
- "generic-array 0.14.7",
+ "generic-array",
"pkcs8 0.10.2",
"subtle",
"zeroize",
@@ -2840,9 +2740,9 @@ dependencies = [
"backtrace",
"curve25519-dalek 4.1.3",
"ecdsa 0.16.9",
- "ed25519-dalek 2.1.1",
+ "ed25519-dalek",
"elliptic-curve 0.13.8",
- "generic-array 0.14.7",
+ "generic-array",
"getrandom",
"hex-literal 0.4.1",
"hmac",
@@ -2851,8 +2751,8 @@ dependencies = [
"num-integer",
"num-traits",
"p256",
- "rand 0.8.5",
- "rand_chacha 0.3.1",
+ "rand",
+ "rand_chacha",
"sec1 0.7.3",
"sha2 0.10.8",
"sha3",
@@ -2909,8 +2809,8 @@ dependencies = [
"arbitrary",
"bytes-lit",
"ctor",
- "ed25519-dalek 2.1.1",
- "rand 0.8.5",
+ "ed25519-dalek",
+ "rand",
"serde",
"serde_json",
"soroban-env-guest",
@@ -3208,7 +3108,7 @@ dependencies = [
"cosmwasm-std",
"cw-multi-test 0.16.4",
"hex",
- "hex-literal 0.4.1",
+ "hex-literal 0.3.4",
"ibc-proto",
"prost 0.11.9",
"serde",
diff --git a/contracts/soroban/Cargo.lock b/contracts/soroban/Cargo.lock
index b02fd6efe..b8c8a485d 100644
--- a/contracts/soroban/Cargo.lock
+++ b/contracts/soroban/Cargo.lock
@@ -92,22 +92,13 @@ version = "1.6.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8c3c1a368f70d6cf7302d78f8f7093da241fb8e8807c05cc9e51a125895a6d5b"
-[[package]]
-name = "block-buffer"
-version = "0.9.0"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "4152116fd6e9dadb291ae18fc1ec3575ed6d84c29642d97890f4b4a3417297e4"
-dependencies = [
- "generic-array 0.14.7",
-]
-
[[package]]
name = "block-buffer"
version = "0.10.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71"
dependencies = [
- "generic-array 0.14.7",
+ "generic-array",
]
[[package]]
@@ -116,12 +107,6 @@ version = "3.16.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "79296716171880943b8470b5f8d03aa55eb2e645a4874bdbb28adb49162e012c"
-[[package]]
-name = "byteorder"
-version = "1.5.0"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b"
-
[[package]]
name = "bytes-lit"
version = "0.0.5"
@@ -166,21 +151,10 @@ dependencies = [
"windows-targets",
]
-[[package]]
-name = "clear_on_drop"
-version = "0.2.5"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "38508a63f4979f0048febc9966fadbd48e5dab31fd0ec6a3f151bbf4a74f7423"
-dependencies = [
- "cc",
-]
-
[[package]]
name = "cluster-connection"
version = "0.0.0"
dependencies = [
- "curve25519-dalek 1.2.6",
- "ed25519-dalek 1.0.1",
"soroban-sdk",
]
@@ -222,8 +196,8 @@ version = "0.5.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "0dc92fb57ca44df6db8059111ab3af99a63d5d0f8375d9972e319a379c6bab76"
dependencies = [
- "generic-array 0.14.7",
- "rand_core 0.6.4",
+ "generic-array",
+ "rand_core",
"subtle",
"zeroize",
]
@@ -234,7 +208,7 @@ version = "0.1.6"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3"
dependencies = [
- "generic-array 0.14.7",
+ "generic-array",
"typenum",
]
@@ -248,32 +222,6 @@ dependencies = [
"syn",
]
-[[package]]
-name = "curve25519-dalek"
-version = "1.2.6"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "57c0d59fed08e452f286b251f88b2fc64a01f50a7b263aa09557ad7285d9e7fa"
-dependencies = [
- "byteorder",
- "clear_on_drop",
- "digest 0.8.1",
- "rand_core 0.3.1",
- "subtle",
-]
-
-[[package]]
-name = "curve25519-dalek"
-version = "3.2.0"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "0b9fdf9972b2bd6af2d913799d9ebc165ea4d2e65878e329d9c6b372c4491b61"
-dependencies = [
- "byteorder",
- "digest 0.9.0",
- "rand_core 0.5.1",
- "subtle",
- "zeroize",
-]
-
[[package]]
name = "curve25519-dalek"
version = "4.1.2"
@@ -283,7 +231,7 @@ dependencies = [
"cfg-if",
"cpufeatures",
"curve25519-dalek-derive",
- "digest 0.10.7",
+ "digest",
"fiat-crypto",
"platforms",
"rustc_version",
@@ -368,31 +316,13 @@ dependencies = [
"syn",
]
-[[package]]
-name = "digest"
-version = "0.8.1"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "f3d0c8c8752312f9713efd397ff63acb9f85585afbf179282e720e7704954dd5"
-dependencies = [
- "generic-array 0.12.4",
-]
-
-[[package]]
-name = "digest"
-version = "0.9.0"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "d3dd60d1080a57a05ab032377049e0591415d2b31afd7028356dbf3cc6dcb066"
-dependencies = [
- "generic-array 0.14.7",
-]
-
[[package]]
name = "digest"
version = "0.10.7"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292"
dependencies = [
- "block-buffer 0.10.4",
+ "block-buffer",
"const-oid",
"crypto-common",
"subtle",
@@ -411,22 +341,13 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ee27f32b5c5292967d2d4a9d7f1e0b0aed2c15daded5a60300e4abb9d8020bca"
dependencies = [
"der",
- "digest 0.10.7",
+ "digest",
"elliptic-curve",
"rfc6979",
- "signature 2.2.0",
+ "signature",
"spki",
]
-[[package]]
-name = "ed25519"
-version = "1.5.3"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "91cff35c70bba8a626e3185d8cd48cc11b5437e1a5bcd15b9b5fa3c64b6dfee7"
-dependencies = [
- "signature 1.6.4",
-]
-
[[package]]
name = "ed25519"
version = "2.2.3"
@@ -434,20 +355,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "115531babc129696a58c64a4fef0a8bf9e9698629fb97e9e40767d235cfbcd53"
dependencies = [
"pkcs8",
- "signature 2.2.0",
-]
-
-[[package]]
-name = "ed25519-dalek"
-version = "1.0.1"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "c762bae6dcaf24c4c84667b8579785430908723d5c889f469d76a41d59cc7a9d"
-dependencies = [
- "curve25519-dalek 3.2.0",
- "ed25519 1.5.3",
- "rand 0.7.3",
- "sha2 0.9.9",
- "zeroize",
+ "signature",
]
[[package]]
@@ -456,11 +364,11 @@ version = "2.0.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "7277392b266383ef8396db7fdeb1e77b6c52fed775f5df15bb24f35b72156980"
dependencies = [
- "curve25519-dalek 4.1.2",
- "ed25519 2.2.3",
- "rand_core 0.6.4",
+ "curve25519-dalek",
+ "ed25519",
+ "rand_core",
"serde",
- "sha2 0.10.8",
+ "sha2",
"zeroize",
]
@@ -478,12 +386,12 @@ checksum = "b5e6043086bf7973472e0c7dff2142ea0b680d30e18d9cc40f267efbf222bd47"
dependencies = [
"base16ct",
"crypto-bigint",
- "digest 0.10.7",
+ "digest",
"ff",
- "generic-array 0.14.7",
+ "generic-array",
"group",
"pkcs8",
- "rand_core 0.6.4",
+ "rand_core",
"sec1",
"subtle",
"zeroize",
@@ -513,7 +421,7 @@ version = "0.13.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ded41244b729663b1e574f1b4fb731469f69f79c17667b5d776b16cda0479449"
dependencies = [
- "rand_core 0.6.4",
+ "rand_core",
"subtle",
]
@@ -529,15 +437,6 @@ version = "1.0.7"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1"
-[[package]]
-name = "generic-array"
-version = "0.12.4"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "ffdf9f34f1447443d37393cc6c2b8313aebddcd96906caf34e54c68d8e57d7bd"
-dependencies = [
- "typenum",
-]
-
[[package]]
name = "generic-array"
version = "0.14.7"
@@ -575,7 +474,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f0f9ef7462f7c099f518d754361858f86d8a07af53ba9af0fe635bbccb151a63"
dependencies = [
"ff",
- "rand_core 0.6.4",
+ "rand_core",
"subtle",
]
@@ -612,7 +511,7 @@ version = "0.12.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "6c49c37c09c17a53d937dfbb742eb3a961d65a994e6bcdcf37e7399d0cc8ab5e"
dependencies = [
- "digest 0.10.7",
+ "digest",
]
[[package]]
@@ -706,8 +605,8 @@ dependencies = [
"ecdsa",
"elliptic-curve",
"once_cell",
- "sha2 0.10.8",
- "signature 2.2.0",
+ "sha2",
+ "signature",
]
[[package]]
@@ -824,12 +723,6 @@ version = "1.19.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3fdb12b2476b595f9358c5161aa467c2438859caa136dec86c26fdd2efe17b92"
-[[package]]
-name = "opaque-debug"
-version = "0.3.1"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "c08d65885ee38876c4f86fa503fb49d7b507c2b62552df7c70b2fce627e06381"
-
[[package]]
name = "paste"
version = "1.0.14"
@@ -892,17 +785,6 @@ dependencies = [
"proc-macro2",
]
-[[package]]
-name = "rand"
-version = "0.7.3"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "6a6b1679d49b24bbfe0c803429aa1874472f50d9b363131f0e89fc356b544d03"
-dependencies = [
- "rand_chacha 0.2.2",
- "rand_core 0.5.1",
- "rand_hc",
-]
-
[[package]]
name = "rand"
version = "0.8.5"
@@ -910,18 +792,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404"
dependencies = [
"libc",
- "rand_chacha 0.3.1",
- "rand_core 0.6.4",
-]
-
-[[package]]
-name = "rand_chacha"
-version = "0.2.2"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "f4c8ed856279c9737206bf725bf36935d8666ead7aa69b52be55af369d193402"
-dependencies = [
- "ppv-lite86",
- "rand_core 0.5.1",
+ "rand_chacha",
+ "rand_core",
]
[[package]]
@@ -931,30 +803,9 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88"
dependencies = [
"ppv-lite86",
- "rand_core 0.6.4",
+ "rand_core",
]
-[[package]]
-name = "rand_core"
-version = "0.3.1"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "7a6fdeb83b075e8266dcc8762c22776f6877a63111121f5f8c7411e5be7eed4b"
-dependencies = [
- "rand_core 0.4.2",
-]
-
-[[package]]
-name = "rand_core"
-version = "0.4.2"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "9c33a3c44ca05fa6f1807d8e6743f3824e8509beca625669633be0acbdf509dc"
-
-[[package]]
-name = "rand_core"
-version = "0.5.1"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "90bde5296fc891b0cef12a6d03ddccc162ce7b2aff54160af9338f8d40df6d19"
-
[[package]]
name = "rand_core"
version = "0.6.4"
@@ -964,15 +815,6 @@ dependencies = [
"getrandom",
]
-[[package]]
-name = "rand_hc"
-version = "0.2.0"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "ca3129af7b92a17112d59ad498c6f81eaf463253766b90396d39ea7a39d6613c"
-dependencies = [
- "rand_core 0.5.1",
-]
-
[[package]]
name = "rfc6979"
version = "0.4.0"
@@ -1012,7 +854,7 @@ checksum = "d3e97a565f76233a6003f9f5c54be1d9c5bdfa3eccfb189469f11ec4901c47dc"
dependencies = [
"base16ct",
"der",
- "generic-array 0.14.7",
+ "generic-array",
"pkcs8",
"subtle",
"zeroize",
@@ -1085,19 +927,6 @@ dependencies = [
"syn",
]
-[[package]]
-name = "sha2"
-version = "0.9.9"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "4d58a1e1bf39749807d89cf2d98ac2dfa0ff1cb3faa38fbb64dd88ac8013d800"
-dependencies = [
- "block-buffer 0.9.0",
- "cfg-if",
- "cpufeatures",
- "digest 0.9.0",
- "opaque-debug",
-]
-
[[package]]
name = "sha2"
version = "0.10.8"
@@ -1106,7 +935,7 @@ checksum = "793db75ad2bcafc3ffa7c68b215fee268f537982cd901d132f89c6343f3a3dc8"
dependencies = [
"cfg-if",
"cpufeatures",
- "digest 0.10.7",
+ "digest",
]
[[package]]
@@ -1115,24 +944,18 @@ version = "0.10.8"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "75872d278a8f37ef87fa0ddbda7802605cb18344497949862c0d4dcb291eba60"
dependencies = [
- "digest 0.10.7",
+ "digest",
"keccak",
]
-[[package]]
-name = "signature"
-version = "1.6.4"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "74233d3b3b2f6d4b006dc19dee745e73e2a6bfb6f93607cd3b02bd5b00797d7c"
-
[[package]]
name = "signature"
version = "2.2.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "77549399552de45a898a580c1b41d445bf730df867cc44e6c0233bbc4b8329de"
dependencies = [
- "digest 0.10.7",
- "rand_core 0.6.4",
+ "digest",
+ "rand_core",
]
[[package]]
@@ -1188,8 +1011,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "114a0fa0d0cc39d0be16b1ee35b6e5f4ee0592ddcf459bde69391c02b03cf520"
dependencies = [
"backtrace",
- "curve25519-dalek 4.1.2",
- "ed25519-dalek 2.0.0",
+ "curve25519-dalek",
+ "ed25519-dalek",
"getrandom",
"hex-literal",
"hmac",
@@ -1197,9 +1020,9 @@ dependencies = [
"num-derive",
"num-integer",
"num-traits",
- "rand 0.8.5",
- "rand_chacha 0.3.1",
- "sha2 0.10.8",
+ "rand",
+ "rand_chacha",
+ "sha2",
"sha3",
"soroban-builtin-sdk-macros",
"soroban-env-common",
@@ -1253,8 +1076,8 @@ dependencies = [
"arbitrary",
"bytes-lit",
"ctor",
- "ed25519-dalek 2.0.0",
- "rand 0.8.5",
+ "ed25519-dalek",
+ "rand",
"serde",
"serde_json",
"soroban-env-guest",
@@ -1276,7 +1099,7 @@ dependencies = [
"proc-macro2",
"quote",
"rustc_version",
- "sha2 0.10.8",
+ "sha2",
"soroban-env-common",
"soroban-spec",
"soroban-spec-rust",
@@ -1305,7 +1128,7 @@ dependencies = [
"prettyplease",
"proc-macro2",
"quote",
- "sha2 0.10.8",
+ "sha2",
"soroban-spec",
"stellar-xdr",
"syn",
@@ -1656,17 +1479,3 @@ name = "zeroize"
version = "1.7.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "525b4ec142c6b68a2d10f01f7bbf6755599ca3f81ea53b8431b7dd348f5fdb2d"
-dependencies = [
- "zeroize_derive",
-]
-
-[[package]]
-name = "zeroize_derive"
-version = "1.4.2"
-source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "ce36e65b0d2999d2aafac989fb249189a141aee1f53c612c1f37d72631959f69"
-dependencies = [
- "proc-macro2",
- "quote",
- "syn",
-]
diff --git a/contracts/soroban/contracts/cluster-connection/src/contract.rs b/contracts/soroban/contracts/cluster-connection/src/contract.rs
index c9502f03b..a8d329f61 100644
--- a/contracts/soroban/contracts/cluster-connection/src/contract.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/contract.rs
@@ -12,7 +12,8 @@ impl ClusterConnection {
storage::store_native_token(&env, msg.native_token);
storage::store_conn_sn(&env, 0);
- storage::store_admin(&env, msg.relayer);
+ storage::store_relayer(&env, msg.relayer);
+ storage::store_admin(&env, msg.admin);
storage::store_xcall(&env, msg.xcall_address);
storage::store_upgrade_authority(&env, msg.upgrade_authority);
storage::store_validator_threshold(&env, 0);
@@ -27,7 +28,7 @@ impl ClusterConnection {
}
pub fn set_admin(env: Env, address: Address) -> Result<(), ContractError> {
- helpers::ensure_admin(&env)?;
+ helpers::ensure_relayer(&env)?;
storage::store_admin(&env, address);
Ok(())
}
@@ -74,7 +75,7 @@ impl ClusterConnection {
conn_sn: u128,
msg: Bytes,
) -> Result<(), ContractError> {
- helpers::ensure_admin(&env)?;
+ helpers::ensure_relayer(&env)?;
if storage::get_sn_receipt(&env, src_network.clone(), conn_sn) {
return Err(ContractError::DuplicateMessage);
@@ -86,7 +87,7 @@ impl ClusterConnection {
}
pub fn revert_message(env: &Env, sn: u128) -> Result<(), ContractError> {
- helpers::ensure_admin(&env)?;
+ helpers::ensure_relayer(&env)?;
helpers::call_xcall_handle_error(&env, sn)?;
Ok(())
@@ -99,7 +100,7 @@ impl ClusterConnection {
msg: Bytes,
signatures: Vec>,
) -> Result<(), ContractError> {
- helpers::ensure_admin(&env)?;
+ helpers::ensure_relayer(&env)?;
if !helpers::verify_signatures(&env, signatures, msg.clone()){
return Err(ContractError::SignatureVerificationFailed);
@@ -120,14 +121,14 @@ impl ClusterConnection {
message_fee: u128,
response_fee: u128,
) -> Result<(), ContractError> {
- helpers::ensure_admin(&env)?;
+ helpers::ensure_relayer(&env)?;
storage::store_network_fee(&env, network_id, message_fee, response_fee);
Ok(())
}
pub fn claim_fees(env: Env) -> Result<(), ContractError> {
- let admin = helpers::ensure_admin(&env)?;
+ let admin = helpers::ensure_relayer(&env)?;
let token_addr = storage::native_token(&env)?;
let client = token::Client::new(&env, &token_addr);
@@ -138,7 +139,7 @@ impl ClusterConnection {
}
pub fn add_validator(env: Env, address: Address) -> Result<(), ContractError> {
- helpers::ensure_admin(&env)?;
+ helpers::ensure_relayer(&env)?;
let validators = storage::get_validators(&env).unwrap();
if validators.contains(&address) {
return Err(ContractError::ValidatorAlreadyAdded);
@@ -153,13 +154,13 @@ impl ClusterConnection {
}
pub fn set_validators_threshold(env: Env, threshold: u32) -> Result<(), ContractError> {
- helpers::ensure_admin(&env)?;
+ helpers::ensure_relayer(&env)?;
storage::store_validator_threshold(&env, threshold);
Ok(())
}
pub fn remove_validator(env: Env, address: Address) -> Result<(), ContractError> {
- helpers::ensure_admin(&env)?;
+ helpers::ensure_relayer(&env)?;
let validators: Vec = storage::get_validators(&env).unwrap();
let threshold = storage::get_validators_threshold(&env).unwrap();
let admin = storage::admin(&env).unwrap();
diff --git a/contracts/soroban/contracts/cluster-connection/src/helpers.rs b/contracts/soroban/contracts/cluster-connection/src/helpers.rs
index 6f88586e8..eb823868d 100644
--- a/contracts/soroban/contracts/cluster-connection/src/helpers.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/helpers.rs
@@ -3,6 +3,13 @@ use core::panic;
use soroban_sdk::{token, Address, Bytes, BytesN, Env, FromVal, Map, String, Vec};
use crate::{errors::ContractError, interfaces::interface_xcall::XcallClient, storage};
+pub fn ensure_relayer(e: &Env) -> Result {
+ let relayer = storage::relayer(&e)?;
+ relayer.require_auth();
+
+ Ok(relayer)
+}
+
pub fn ensure_admin(e: &Env) -> Result {
let admin = storage::admin(&e)?;
admin.require_auth();
diff --git a/contracts/soroban/contracts/cluster-connection/src/storage.rs b/contracts/soroban/contracts/cluster-connection/src/storage.rs
index 256a44bc8..3e98e7310 100644
--- a/contracts/soroban/contracts/cluster-connection/src/storage.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/storage.rs
@@ -29,6 +29,13 @@ pub fn admin(e: &Env) -> Result {
.ok_or(ContractError::Uninitialized)
}
+pub fn relayer(e: &Env) -> Result {
+ e.storage()
+ .instance()
+ .get(&StorageKey::Relayer)
+ .ok_or(ContractError::Uninitialized)
+}
+
pub fn get_upgrade_authority(e: &Env) -> Result {
e.storage()
.instance()
@@ -123,6 +130,10 @@ pub fn store_receipt(e: &Env, network_id: String, sn: u128) {
extend_persistent(e, &key);
}
+pub fn store_relayer(e: &Env, relayer: Address) {
+ e.storage().instance().set(&StorageKey::Relayer, &relayer);
+}
+
pub fn store_admin(e: &Env, admin: Address) {
e.storage().instance().set(&StorageKey::Admin, &admin);
}
diff --git a/contracts/soroban/contracts/cluster-connection/src/test.rs b/contracts/soroban/contracts/cluster-connection/src/test.rs
index 3f7e978ae..17470c70f 100644
--- a/contracts/soroban/contracts/cluster-connection/src/test.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/test.rs
@@ -49,6 +49,7 @@ impl TestContext {
self.env.mock_all_auths();
client.initialize(&InitializeMsg {
+ admin: self.relayer.clone(),
relayer: self.relayer.clone(),
native_token: self.native_token.clone(),
xcall_address: self.xcall.clone(),
@@ -66,6 +67,7 @@ impl TestContext {
fn get_dummy_initialize_msg(env: &Env) -> InitializeMsg {
InitializeMsg {
+ admin: Address::generate(&env),
relayer: Address::generate(&env),
native_token: env.register_stellar_asset_contract(Address::generate(&env)),
xcall_address: Address::generate(&env),
diff --git a/contracts/soroban/contracts/cluster-connection/src/types.rs b/contracts/soroban/contracts/cluster-connection/src/types.rs
index cffcfc46f..bf5f551d7 100644
--- a/contracts/soroban/contracts/cluster-connection/src/types.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/types.rs
@@ -4,6 +4,7 @@ use soroban_sdk::{contracttype, Address, String};
#[derive(Clone)]
pub enum StorageKey {
Xcall,
+ Relayer,
Admin,
UpgradeAuthority,
Xlm,
@@ -17,6 +18,7 @@ pub enum StorageKey {
#[contracttype]
pub struct InitializeMsg {
pub relayer: Address,
+ pub admin: Address,
pub native_token: Address,
pub xcall_address: Address,
pub upgrade_authority: Address,
From 8af3ecd7c145e962a49432aa18424b2d81bf9711 Mon Sep 17 00:00:00 2001
From: Itshyphen <075bct064.ranju@pcampus.edu.np>
Date: Fri, 25 Oct 2024 13:31:55 +0545
Subject: [PATCH 04/17] fix: stellar cluster conenction
---
.../cluster-connection/src/contract.rs | 65 +++++--------------
.../cluster-connection/src/helpers.rs | 14 +++-
.../cluster-connection/src/storage.rs | 14 ----
.../contracts/cluster-connection/src/test.rs | 59 ++++-------------
4 files changed, 40 insertions(+), 112 deletions(-)
diff --git a/contracts/soroban/contracts/cluster-connection/src/contract.rs b/contracts/soroban/contracts/cluster-connection/src/contract.rs
index a8d329f61..86377d616 100644
--- a/contracts/soroban/contracts/cluster-connection/src/contract.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/contract.rs
@@ -69,30 +69,6 @@ impl ClusterConnection {
Ok(())
}
- pub fn recv_message(
- env: Env,
- src_network: String,
- conn_sn: u128,
- msg: Bytes,
- ) -> Result<(), ContractError> {
- helpers::ensure_relayer(&env)?;
-
- if storage::get_sn_receipt(&env, src_network.clone(), conn_sn) {
- return Err(ContractError::DuplicateMessage);
- }
- storage::store_receipt(&env, src_network.clone(), conn_sn);
-
- helpers::call_xcall_handle_message(&env, &src_network, msg)?;
- Ok(())
- }
-
- pub fn revert_message(env: &Env, sn: u128) -> Result<(), ContractError> {
- helpers::ensure_relayer(&env)?;
- helpers::call_xcall_handle_error(&env, sn)?;
-
- Ok(())
- }
-
pub fn recv_message_with_signatures(
env: Env,
src_network: String,
@@ -102,7 +78,7 @@ impl ClusterConnection {
) -> Result<(), ContractError> {
helpers::ensure_relayer(&env)?;
- if !helpers::verify_signatures(&env, signatures, msg.clone()){
+ if !helpers::verify_signatures(&env, signatures, &src_network, &conn_sn, &msg){
return Err(ContractError::SignatureVerificationFailed);
};
@@ -138,13 +114,20 @@ impl ClusterConnection {
Ok(())
}
- pub fn add_validator(env: Env, address: Address) -> Result<(), ContractError> {
+ pub fn update_validators(env: Env, addresses: Vec, threshold: u32) -> Result<(), ContractError> {
helpers::ensure_relayer(&env)?;
- let validators = storage::get_validators(&env).unwrap();
- if validators.contains(&address) {
- return Err(ContractError::ValidatorAlreadyAdded);
+ let mut validators = Vec::new(&env);
+
+ for address in addresses.clone() {
+ if !validators.contains(&address) {
+ validators.push_back(address);
+ }
}
- storage::add_validator(&env, address);
+ if (validators.len() as u32) < threshold {
+ return Err(ContractError::ThresholdExceeded);
+
+ }
+ storage::store_validators(&env, addresses);
Ok(())
}
@@ -155,27 +138,11 @@ impl ClusterConnection {
pub fn set_validators_threshold(env: Env, threshold: u32) -> Result<(), ContractError> {
helpers::ensure_relayer(&env)?;
- storage::store_validator_threshold(&env, threshold);
- Ok(())
- }
-
- pub fn remove_validator(env: Env, address: Address) -> Result<(), ContractError> {
- helpers::ensure_relayer(&env)?;
- let validators: Vec = storage::get_validators(&env).unwrap();
- let threshold = storage::get_validators_threshold(&env).unwrap();
- let admin = storage::admin(&env).unwrap();
-
- let index = validators.iter().position(|v| v == address);
- if index.is_none() {
- return Err(ContractError::ValidatorNotFound);
- }
- if address == admin {
- return Err(ContractError::CannotRemoveAdmin);
- }
- if (validators.len() as u32) <= threshold {
+ let validators = storage::get_validators(&env).unwrap();
+ if (validators.len() as u32) < threshold {
return Err(ContractError::ThresholdExceeded);
}
- storage::remove_validator(&env, address);
+ storage::store_validator_threshold(&env, threshold);
Ok(())
}
diff --git a/contracts/soroban/contracts/cluster-connection/src/helpers.rs b/contracts/soroban/contracts/cluster-connection/src/helpers.rs
index eb823868d..40a930ccc 100644
--- a/contracts/soroban/contracts/cluster-connection/src/helpers.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/helpers.rs
@@ -80,7 +80,9 @@ pub fn compress_public_keys(e: &Env, uncompressed_public_key: BytesN<65>) -> Byt
pub fn verify_signatures(
e: &Env,
signatures: Vec>,
- message: Bytes,
+ src_network: &String,
+ conn_sn: &u128,
+ message: &Bytes,
) -> bool {
let validators = storage::get_validators(e).unwrap();
let threshold = storage::get_validators_threshold(e).unwrap();
@@ -88,14 +90,13 @@ pub fn verify_signatures(
if signatures.len() < threshold {
return false
}
- let message_hash = e.crypto().sha256(&message);
+ let message_hash = e.crypto().keccak256(&get_encoded_message(e, src_network, conn_sn, message));
let mut unique_validators = Map::new(e);
let mut count = 0;
for sig in signatures.iter() {
let r_s_v = sig.to_array();
-
// Separate signature (r + s) and recovery ID
let signature_array: [u8; 64] = r_s_v[..64].try_into().unwrap(); // r + s part
let recovery_id = r_s_v[64] as u32; // recovery ID
@@ -120,6 +121,13 @@ pub fn verify_signatures(
}
+pub fn get_encoded_message(e: &Env, src_network: &String, conn_sn: &u128, message: &Bytes) -> Bytes {
+ let mut result = Bytes::from_val(e, &src_network.to_val());
+ result.extend_from_slice(&conn_sn.to_be_bytes());
+ result.append(message);
+ result
+}
+
pub fn call_xcall_handle_message(e: &Env, nid: &String, msg: Bytes) -> Result<(), ContractError> {
let xcall_addr = storage::get_xcall(&e)?;
let client = XcallClient::new(&e, &xcall_addr);
diff --git a/contracts/soroban/contracts/cluster-connection/src/storage.rs b/contracts/soroban/contracts/cluster-connection/src/storage.rs
index 3e98e7310..538b21ddd 100644
--- a/contracts/soroban/contracts/cluster-connection/src/storage.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/storage.rs
@@ -164,20 +164,6 @@ pub fn store_validators(e: &Env, validators: Vec) {
e.storage().instance().set(&StorageKey::Validators, &validators);
}
-pub fn remove_validator(e: &Env, address: Address) {
- let mut validators: Vec = get_validators(e).unwrap();
- if let Some(index) = validators.iter().position(|v| v == address) {
- validators.remove(index.try_into().unwrap());
- e.storage().instance().set(&StorageKey::Validators, &validators);
- }
-}
-
-pub fn add_validator(e: &Env, address: Address) {
- let mut validators = get_validators(e).unwrap();
- validators.push_back(address);
- store_validators(e, validators);
-}
-
pub fn store_network_fee(e: &Env, network_id: String, message_fee: u128, response_fee: u128) {
let key = StorageKey::NetworkFee(network_id);
let network_fee = NetworkFee {
diff --git a/contracts/soroban/contracts/cluster-connection/src/test.rs b/contracts/soroban/contracts/cluster-connection/src/test.rs
index 17470c70f..bab627e00 100644
--- a/contracts/soroban/contracts/cluster-connection/src/test.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/test.rs
@@ -353,11 +353,16 @@ fn test_add_validator() {
ctx.init_context(&client);
- let validator = Address::generate(&ctx.env);
- client.add_validator(&validator.clone());
+ let validator1 = Address::generate(&ctx.env);
+ let validator2 = Address::generate(&ctx.env);
+ let validator3 = Address::generate(&ctx.env);
let mut validators = Vec::new(&ctx.env);
- validators.push_back(validator.clone());
+ validators.push_back(validator1.clone());
+ validators.push_back(validator2.clone());
+ validators.push_back(validator3.clone());
+
+ client.update_validators(&validators, &3_u32);
assert_eq!(
ctx.env.auths(),
@@ -366,27 +371,23 @@ fn test_add_validator() {
AuthorizedInvocation {
function: AuthorizedFunction::Contract((
client.address.clone(),
- Symbol::new(&ctx.env, "add_validator"),
- (validator.clone(),).into_val(&ctx.env)
+ Symbol::new(&ctx.env, "update_validators"),
+ (validators.clone(), 3,).into_val(&ctx.env)
)),
sub_invocations: std::vec![]
}
)]
);
- let validator = Address::generate(&ctx.env);
- client.add_validator(&validator.clone());
- validators.push_back(validator.clone());
-
-
assert_eq!(
client.get_validators(),
validators
);
}
+
#[test]
-fn test_remove_validator() {
+fn test_set_threshold() {
let ctx = TestContext::default();
let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
@@ -396,46 +397,12 @@ fn test_remove_validator() {
let validator2 = Address::generate(&ctx.env);
let validator3 = Address::generate(&ctx.env);
- client.add_validator(&validator1.clone());
- client.add_validator(&validator2.clone());
- client.add_validator(&validator3.clone());
-
let mut validators = Vec::new(&ctx.env);
validators.push_back(validator1.clone());
validators.push_back(validator2.clone());
validators.push_back(validator3.clone());
- assert_eq!(
- client.get_validators(),
- validators
- );
-
- assert_eq!(
- client.get_validators().len(),
- 3
- );
-
- client.remove_validator(&validator2.clone());
-
- assert_eq!(
- client.get_validators().len(),
- 2
- );
- validators.remove(1);
-
- assert_eq!(
- client.get_validators(),
- validators
- );
-}
-
-#[test]
-fn test_set_threshold() {
- let ctx = TestContext::default();
- let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
-
- ctx.init_context(&client);
-
+ client.update_validators(&validators, &3_u32);
let threshold: u32 = 2_u32;
client.set_validators_threshold(&threshold);
From cb100095549103b526aebaa5b164d0b3d56b95d8 Mon Sep 17 00:00:00 2001
From: Itshyphen <075bct064.ranju@pcampus.edu.np>
Date: Fri, 25 Oct 2024 14:39:01 +0545
Subject: [PATCH 05/17] fix: be to le bytes
---
contracts/soroban/contracts/cluster-connection/src/helpers.rs | 3 +--
1 file changed, 1 insertion(+), 2 deletions(-)
diff --git a/contracts/soroban/contracts/cluster-connection/src/helpers.rs b/contracts/soroban/contracts/cluster-connection/src/helpers.rs
index 40a930ccc..f34d968ed 100644
--- a/contracts/soroban/contracts/cluster-connection/src/helpers.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/helpers.rs
@@ -60,7 +60,6 @@ pub fn transfer_token(
pub fn compress_public_keys(e: &Env, uncompressed_public_key: BytesN<65>) -> Bytes {
let uncompressed_pub_key_array = uncompressed_public_key.to_array();
if uncompressed_pub_key_array[0] != 0x04 {
- //return empty bytessize(33);
return Bytes::from_array(e, &[0u8; 33]);
}
@@ -123,7 +122,7 @@ pub fn verify_signatures(
pub fn get_encoded_message(e: &Env, src_network: &String, conn_sn: &u128, message: &Bytes) -> Bytes {
let mut result = Bytes::from_val(e, &src_network.to_val());
- result.extend_from_slice(&conn_sn.to_be_bytes());
+ result.extend_from_slice(&conn_sn.to_le_bytes());
result.append(message);
result
}
From a47a8c639b341b5d78aabbc104935fe8f55942e2 Mon Sep 17 00:00:00 2001
From: gcranju
Date: Tue, 26 Nov 2024 11:45:27 +0545
Subject: [PATCH 06/17] fix: stellar cluster connection finalize
---
Cargo.lock | 1 +
.../lib/openzeppelin-contracts-upgradeable | 2 +-
contracts/soroban/Cargo.lock | 1 +
.../contracts/cluster-connection/Cargo.toml | 1 +
.../cluster-connection/src/contract.rs | 26 +++-
.../cluster-connection/src/helpers.rs | 60 +++-----
.../cluster-connection/src/storage.rs | 6 +-
.../contracts/cluster-connection/src/test.rs | 137 ++++++++++++++----
.../soroban/libs/soroban-rlp/src/utils.rs | 2 +-
scripts/optimize-stellar.sh | 2 +-
10 files changed, 164 insertions(+), 74 deletions(-)
diff --git a/Cargo.lock b/Cargo.lock
index b673ca4b1..7ceede42a 100644
--- a/Cargo.lock
+++ b/Cargo.lock
@@ -289,6 +289,7 @@ dependencies = [
name = "cluster-connection"
version = "0.0.0"
dependencies = [
+ "soroban-rlp",
"soroban-sdk",
]
diff --git a/contracts/evm/lib/openzeppelin-contracts-upgradeable b/contracts/evm/lib/openzeppelin-contracts-upgradeable
index 625fb3c2b..4ca003c96 160000
--- a/contracts/evm/lib/openzeppelin-contracts-upgradeable
+++ b/contracts/evm/lib/openzeppelin-contracts-upgradeable
@@ -1 +1 @@
-Subproject commit 625fb3c2b2696f1747ba2e72d1e1113066e6c177
+Subproject commit 4ca003c9635d2c16756cf8c9db6760e2d3653dee
diff --git a/contracts/soroban/Cargo.lock b/contracts/soroban/Cargo.lock
index b8c8a485d..8d9e9437a 100644
--- a/contracts/soroban/Cargo.lock
+++ b/contracts/soroban/Cargo.lock
@@ -155,6 +155,7 @@ dependencies = [
name = "cluster-connection"
version = "0.0.0"
dependencies = [
+ "soroban-rlp",
"soroban-sdk",
]
diff --git a/contracts/soroban/contracts/cluster-connection/Cargo.toml b/contracts/soroban/contracts/cluster-connection/Cargo.toml
index be7f7e112..e6f6b4820 100644
--- a/contracts/soroban/contracts/cluster-connection/Cargo.toml
+++ b/contracts/soroban/contracts/cluster-connection/Cargo.toml
@@ -10,6 +10,7 @@ doctest = false
[dependencies]
soroban-sdk = { workspace = true, features = ["alloc"] }
+soroban-rlp = { path = "../../libs/soroban-rlp" }
[dev-dependencies]
soroban-sdk = { workspace = true, features = ["testutils"] }
diff --git a/contracts/soroban/contracts/cluster-connection/src/contract.rs b/contracts/soroban/contracts/cluster-connection/src/contract.rs
index 86377d616..b02d54eec 100644
--- a/contracts/soroban/contracts/cluster-connection/src/contract.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/contract.rs
@@ -28,7 +28,7 @@ impl ClusterConnection {
}
pub fn set_admin(env: Env, address: Address) -> Result<(), ContractError> {
- helpers::ensure_relayer(&env)?;
+ helpers::ensure_admin(&env)?;
storage::store_admin(&env, address);
Ok(())
}
@@ -45,6 +45,12 @@ impl ClusterConnection {
Ok(())
}
+ pub fn set_relayer(env: Env, address: Address) -> Result<(), ContractError> {
+ helpers::ensure_admin(&env)?;
+ storage::store_relayer(&env, address);
+ Ok(())
+ }
+
pub fn send_message(
env: Env,
tx_origin: Address,
@@ -114,11 +120,11 @@ impl ClusterConnection {
Ok(())
}
- pub fn update_validators(env: Env, addresses: Vec, threshold: u32) -> Result<(), ContractError> {
- helpers::ensure_relayer(&env)?;
+ pub fn update_validators(env: Env, pub_keys: Vec>, threshold: u32) -> Result<(), ContractError> {
+ helpers::ensure_admin(&env)?;
let mut validators = Vec::new(&env);
- for address in addresses.clone() {
+ for address in pub_keys.clone() {
if !validators.contains(&address) {
validators.push_back(address);
}
@@ -127,7 +133,8 @@ impl ClusterConnection {
return Err(ContractError::ThresholdExceeded);
}
- storage::store_validators(&env, addresses);
+ storage::store_validators(&env, pub_keys);
+ storage::store_validator_threshold(&env, threshold);
Ok(())
}
@@ -137,7 +144,7 @@ impl ClusterConnection {
}
pub fn set_validators_threshold(env: Env, threshold: u32) -> Result<(), ContractError> {
- helpers::ensure_relayer(&env)?;
+ helpers::ensure_admin(&env)?;
let validators = storage::get_validators(&env).unwrap();
if (validators.len() as u32) < threshold {
return Err(ContractError::ThresholdExceeded);
@@ -146,11 +153,16 @@ impl ClusterConnection {
Ok(())
}
- pub fn get_validators(env: Env) -> Result, ContractError> {
+ pub fn get_validators(env: Env) -> Result>, ContractError> {
let validators = storage::get_validators(&env).unwrap();
Ok(validators)
}
+ pub fn get_relayer(env: Env) -> Result {
+ let address = storage::relayer(&env)?;
+ Ok(address)
+ }
+
pub fn get_fee(env: Env, network_id: String, response: bool) -> Result {
helpers::get_network_fee(&env, network_id, response)
}
diff --git a/contracts/soroban/contracts/cluster-connection/src/helpers.rs b/contracts/soroban/contracts/cluster-connection/src/helpers.rs
index f34d968ed..f37ed14fb 100644
--- a/contracts/soroban/contracts/cluster-connection/src/helpers.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/helpers.rs
@@ -1,7 +1,6 @@
-use core::panic;
-
-use soroban_sdk::{token, Address, Bytes, BytesN, Env, FromVal, Map, String, Vec};
+use soroban_sdk::{token, vec, Address, Bytes, BytesN, Env, Map, String, Vec};
use crate::{errors::ContractError, interfaces::interface_xcall::XcallClient, storage};
+use soroban_rlp::encoder;
pub fn ensure_relayer(e: &Env) -> Result {
let relayer = storage::relayer(&e)?;
@@ -57,25 +56,6 @@ pub fn transfer_token(
Ok(())
}
-pub fn compress_public_keys(e: &Env, uncompressed_public_key: BytesN<65>) -> Bytes {
- let uncompressed_pub_key_array = uncompressed_public_key.to_array();
- if uncompressed_pub_key_array[0] != 0x04 {
- return Bytes::from_array(e, &[0u8; 33]);
- }
-
- let x = &uncompressed_pub_key_array[1..33];
- let y = &uncompressed_pub_key_array[33..65];
-
- let prefix = if y[31] % 2 == 0 { 0x02 } else { 0x03 };
-
- let mut compressed_pub_key_array = [0u8; 33];
- compressed_pub_key_array[0] = prefix;
- compressed_pub_key_array[1..].copy_from_slice(x);
- let compressed_pub_key = Bytes::from_array(e, &compressed_pub_key_array);
-
- compressed_pub_key
-}
-
pub fn verify_signatures(
e: &Env,
signatures: Vec>,
@@ -98,19 +78,17 @@ pub fn verify_signatures(
let r_s_v = sig.to_array();
// Separate signature (r + s) and recovery ID
let signature_array: [u8; 64] = r_s_v[..64].try_into().unwrap(); // r + s part
- let recovery_id = r_s_v[64] as u32; // recovery ID
-
+ let recovery_code = match r_s_v[64] {
+ rc if rc >= 27 => rc - 27,
+ rc => rc,
+ };
let signature = BytesN::<64>::from_array(e, &signature_array);
- let uncompressed_public_key = e.crypto().secp256k1_recover(&message_hash, &signature, recovery_id);
+ let public_key = e.crypto().secp256k1_recover(&message_hash, &signature, recovery_code as u32);
- let compressed_pub_key = compress_public_keys(e, uncompressed_public_key);
-
- let stellar_address = Address::from_string_bytes(&compressed_pub_key);
-
- if validators.contains(&stellar_address) {
- if !unique_validators.contains_key(stellar_address.clone()) {
- unique_validators.set(stellar_address, count);
+ if validators.contains(&public_key) {
+ if !unique_validators.contains_key(public_key.clone()) {
+ unique_validators.set(public_key, count);
count += 1;
}
}
@@ -121,12 +99,15 @@ pub fn verify_signatures(
pub fn get_encoded_message(e: &Env, src_network: &String, conn_sn: &u128, message: &Bytes) -> Bytes {
- let mut result = Bytes::from_val(e, &src_network.to_val());
- result.extend_from_slice(&conn_sn.to_le_bytes());
- result.append(message);
- result
+ let mut list = vec![&e];
+ list.push_back(encoder::encode_string(&e, src_network.clone()));
+ list.push_back(encoder::encode_u128(&e, conn_sn.clone()));
+ list.push_back(encoder::encode(&e, message.clone()));
+
+ encoder::encode_list(&e, list, false)
}
+#[cfg(not(test))]
pub fn call_xcall_handle_message(e: &Env, nid: &String, msg: Bytes) -> Result<(), ContractError> {
let xcall_addr = storage::get_xcall(&e)?;
let client = XcallClient::new(&e, &xcall_addr);
@@ -135,6 +116,13 @@ pub fn call_xcall_handle_message(e: &Env, nid: &String, msg: Bytes) -> Result<()
Ok(())
}
+#[cfg(test)]
+pub fn call_xcall_handle_message(_e: &Env, _nid: &String, _msg: Bytes) -> Result<(), ContractError> {
+ Ok(())
+}
+
+
+
pub fn call_xcall_handle_error(e: &Env, sn: u128) -> Result<(), ContractError> {
let xcall_addr = storage::get_xcall(&e)?;
let client = XcallClient::new(&e, &xcall_addr);
diff --git a/contracts/soroban/contracts/cluster-connection/src/storage.rs b/contracts/soroban/contracts/cluster-connection/src/storage.rs
index 538b21ddd..bdc017a16 100644
--- a/contracts/soroban/contracts/cluster-connection/src/storage.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/storage.rs
@@ -1,4 +1,4 @@
-use soroban_sdk::{vec, Address, Env, String, Vec};
+use soroban_sdk::{Address, BytesN, Env, String, Vec};
use crate::{
errors::ContractError,
@@ -117,7 +117,7 @@ pub fn get_validators_threshold(e: &Env) -> Result {
.ok_or(ContractError::Uninitialized)
}
-pub fn get_validators(e: &Env) -> Result, ContractError> {
+pub fn get_validators(e: &Env) -> Result>, ContractError> {
e.storage()
.instance()
.get(&StorageKey::Validators)
@@ -160,7 +160,7 @@ pub fn store_validator_threshold(e: &Env, threshold: u32) {
e.storage().instance().set(&StorageKey::ValidatorThreshold, &threshold);
}
-pub fn store_validators(e: &Env, validators: Vec) {
+pub fn store_validators(e: &Env, validators: Vec>) {
e.storage().instance().set(&StorageKey::Validators, &validators);
}
diff --git a/contracts/soroban/contracts/cluster-connection/src/test.rs b/contracts/soroban/contracts/cluster-connection/src/test.rs
index bab627e00..fe375ba57 100644
--- a/contracts/soroban/contracts/cluster-connection/src/test.rs
+++ b/contracts/soroban/contracts/cluster-connection/src/test.rs
@@ -13,15 +13,14 @@ use crate::{
types::InitializeMsg,
};
use soroban_sdk::{
- symbol_short,
- testutils::{Address as _, AuthorizedFunction, AuthorizedInvocation, Events},
- token, vec, Address, Bytes, Env, IntoVal, String, Symbol, Vec,
+ symbol_short, testutils::{Address as _, AuthorizedFunction, AuthorizedInvocation, Events}, token, vec, Address, Bytes, BytesN, Env, IntoVal, String, Symbol, Vec
};
pub struct TestContext {
env: Env,
xcall: Address,
contract: Address,
+ admin:Address,
relayer: Address,
native_token: Address,
token_admin: Address,
@@ -33,12 +32,14 @@ impl TestContext {
pub fn default() -> Self {
let env = Env::default();
let token_admin = Address::generate(&env);
+ let xcall = env.register_contract_wasm(None, xcall::WASM);
Self {
- xcall: env.register_contract_wasm(None, xcall::WASM),
+ xcall: xcall.clone(),
contract: env.register_contract(None, ClusterConnection),
relayer: Address::generate(&env),
+ admin: Address::generate(&env),
native_token: env.register_stellar_asset_contract(token_admin.clone()),
- nid: String::from_str(&env, "icon"),
+ nid: String::from_str(&env, "0x2.icon"),
upgrade_authority: Address::generate(&env),
env,
token_admin,
@@ -49,12 +50,13 @@ impl TestContext {
self.env.mock_all_auths();
client.initialize(&InitializeMsg {
- admin: self.relayer.clone(),
+ admin: self.admin.clone(),
relayer: self.relayer.clone(),
native_token: self.native_token.clone(),
xcall_address: self.xcall.clone(),
upgrade_authority: self.upgrade_authority.clone(),
});
+
}
pub fn init_send_message(&self, client: &ClusterConnectionClient<'static>) {
@@ -75,6 +77,7 @@ fn get_dummy_initialize_msg(env: &Env) -> InitializeMsg {
}
}
+
#[test]
fn test_initialize() {
let ctx = TestContext::default();
@@ -83,7 +86,7 @@ fn test_initialize() {
ctx.init_context(&client);
let admin = client.get_admin();
- assert_eq!(admin, ctx.relayer)
+ assert_eq!(admin, ctx.admin)
}
#[test]
@@ -109,7 +112,7 @@ fn test_set_admin() {
assert_eq!(
ctx.env.auths(),
std::vec![(
- ctx.relayer.clone(),
+ ctx.admin.clone(),
AuthorizedInvocation {
function: AuthorizedFunction::Contract((
client.address.clone(),
@@ -353,26 +356,25 @@ fn test_add_validator() {
ctx.init_context(&client);
- let validator1 = Address::generate(&ctx.env);
- let validator2 = Address::generate(&ctx.env);
- let validator3 = Address::generate(&ctx.env);
+ let val1 = [4, 174, 54, 168, 191, 216, 207, 101, 134, 243, 76, 104, 133, 40, 137, 72, 53, 245, 231, 193, 157, 54, 104, 155, 172, 84, 96, 101, 107, 97, 60, 94, 171, 241, 250, 152, 34, 18, 170, 39, 202, 236, 226, 58, 39, 8, 235, 60, 137, 54, 225, 50, 185, 253, 130, 197, 174, 226, 170, 75, 6, 145, 123, 87, 19];
+ let val2 = [4, 91, 65, 155, 222, 192, 210, 187, 193, 108, 232, 174, 20, 79, 248, 232, 37, 18, 63, 208, 203, 62, 54, 208, 7, 91, 109, 141, 229, 170, 181, 51, 136, 172, 143, 180, 194, 138, 138, 56, 67, 243, 7, 60, 218, 164, 12, 148, 63, 116, 115, 127, 192, 206, 164, 169, 95, 135, 119, 138, 255, 172, 115, 129, 144];
+ let val3 = [4, 248, 192, 175, 198, 228, 250, 20, 158, 23, 251, 176, 244, 208, 150, 71, 151, 27, 208, 22, 41, 30, 154, 198, 109, 10, 112, 142, 200, 47, 200, 213, 210, 172, 135, 141, 129, 183, 211, 241, 211, 127, 16, 19, 67, 159, 195, 235, 88, 164, 223, 47, 128, 47, 147, 28, 121, 28, 93, 129, 176, 144, 52, 243, 55];
let mut validators = Vec::new(&ctx.env);
- validators.push_back(validator1.clone());
- validators.push_back(validator2.clone());
- validators.push_back(validator3.clone());
-
+ validators.push_back(BytesN::from_array(&ctx.env, &val1));
+ validators.push_back(BytesN::from_array(&ctx.env, &val2));
+ validators.push_back(BytesN::from_array(&ctx.env, &val3));
client.update_validators(&validators, &3_u32);
assert_eq!(
ctx.env.auths(),
std::vec![(
- ctx.relayer,
+ ctx.admin.clone(),
AuthorizedInvocation {
function: AuthorizedFunction::Contract((
client.address.clone(),
Symbol::new(&ctx.env, "update_validators"),
- (validators.clone(), 3,).into_val(&ctx.env)
+ (validators.clone(), 3_u32,).into_val(&ctx.env)
)),
sub_invocations: std::vec![]
}
@@ -393,23 +395,23 @@ fn test_set_threshold() {
ctx.init_context(&client);
- let validator1 = Address::generate(&ctx.env);
- let validator2 = Address::generate(&ctx.env);
- let validator3 = Address::generate(&ctx.env);
+ let val1 = [4, 174, 54, 168, 191, 216, 207, 101, 134, 243, 76, 104, 133, 40, 137, 72, 53, 245, 231, 193, 157, 54, 104, 155, 172, 84, 96, 101, 107, 97, 60, 94, 171, 241, 250, 152, 34, 18, 170, 39, 202, 236, 226, 58, 39, 8, 235, 60, 137, 54, 225, 50, 185, 253, 130, 197, 174, 226, 170, 75, 6, 145, 123, 87, 19];
+ let val2 = [4, 91, 65, 155, 222, 192, 210, 187, 193, 108, 232, 174, 20, 79, 248, 232, 37, 18, 63, 208, 203, 62, 54, 208, 7, 91, 109, 141, 229, 170, 181, 51, 136, 172, 143, 180, 194, 138, 138, 56, 67, 243, 7, 60, 218, 164, 12, 148, 63, 116, 115, 127, 192, 206, 164, 169, 95, 135, 119, 138, 255, 172, 115, 129, 144];
+ let val3 = [4, 248, 192, 175, 198, 228, 250, 20, 158, 23, 251, 176, 244, 208, 150, 71, 151, 27, 208, 22, 41, 30, 154, 198, 109, 10, 112, 142, 200, 47, 200, 213, 210, 172, 135, 141, 129, 183, 211, 241, 211, 127, 16, 19, 67, 159, 195, 235, 88, 164, 223, 47, 128, 47, 147, 28, 121, 28, 93, 129, 176, 144, 52, 243, 55];
let mut validators = Vec::new(&ctx.env);
- validators.push_back(validator1.clone());
- validators.push_back(validator2.clone());
- validators.push_back(validator3.clone());
-
+ validators.push_back(BytesN::from_array(&ctx.env, &val1));
+ validators.push_back(BytesN::from_array(&ctx.env, &val2));
+ validators.push_back(BytesN::from_array(&ctx.env, &val3));
client.update_validators(&validators, &3_u32);
+
let threshold: u32 = 2_u32;
client.set_validators_threshold(&threshold);
assert_eq!(
ctx.env.auths(),
std::vec![(
- ctx.relayer,
+ ctx.admin.clone(),
AuthorizedInvocation {
function: AuthorizedFunction::Contract((
client.address.clone(),
@@ -426,3 +428,88 @@ fn test_set_threshold() {
client.set_validators_threshold(&threshold);
assert_eq!(client.get_validators_threshold(), threshold);
}
+
+
+#[test]
+fn test_receive_message() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+
+ ctx.init_context(&client);
+
+ let val1 = [4, 174, 54, 168, 191, 216, 207, 101, 134, 243, 76, 104, 133, 40, 137, 72, 53, 245, 231, 193, 157, 54, 104, 155, 172, 84, 96, 101, 107, 97, 60, 94, 171, 241, 250, 152, 34, 18, 170, 39, 202, 236, 226, 58, 39, 8, 235, 60, 137, 54, 225, 50, 185, 253, 130, 197, 174, 226, 170, 75, 6, 145, 123, 87, 19];
+ let val2 = [4, 91, 65, 155, 222, 192, 210, 187, 193, 108, 232, 174, 20, 79, 248, 232, 37, 18, 63, 208, 203, 62, 54, 208, 7, 91, 109, 141, 229, 170, 181, 51, 136, 172, 143, 180, 194, 138, 138, 56, 67, 243, 7, 60, 218, 164, 12, 148, 63, 116, 115, 127, 192, 206, 164, 169, 95, 135, 119, 138, 255, 172, 115, 129, 144];
+ let val3 = [4, 248, 192, 175, 198, 228, 250, 20, 158, 23, 251, 176, 244, 208, 150, 71, 151, 27, 208, 22, 41, 30, 154, 198, 109, 10, 112, 142, 200, 47, 200, 213, 210, 172, 135, 141, 129, 183, 211, 241, 211, 127, 16, 19, 67, 159, 195, 235, 88, 164, 223, 47, 128, 47, 147, 28, 121, 28, 93, 129, 176, 144, 52, 243, 55];
+
+ let mut validators = Vec::new(&ctx.env);
+ validators.push_back(BytesN::from_array(&ctx.env, &val1));
+ validators.push_back(BytesN::from_array(&ctx.env, &val2));
+ validators.push_back(BytesN::from_array(&ctx.env, &val3));
+ client.update_validators(&validators, &1_u32);
+
+ let conn_sn = 456456_u128;
+ let msg = Bytes::from_array(&ctx.env,&[104, 101, 108, 108, 111]);
+ let src_network = String::from_str(&ctx.env, "0x2.icon");
+
+ let mut signatures = Vec::new(&ctx.env);
+ signatures.push_back(BytesN::from_array(&ctx.env, &[35, 247, 49, 199, 251, 53, 83, 51, 115, 148, 35, 48, 85, 203, 185, 236, 5, 171, 221, 29, 247, 203, 190, 195, 208, 218, 204, 237, 88, 191, 91, 75, 48, 87, 108, 161, 75, 234, 147, 234, 65, 134, 233, 32, 249, 159, 43, 159, 86, 211, 1, 117, 176, 167, 53, 99, 34, 243, 165, 215, 93, 232, 67, 184, 27]));
+
+ client.recv_message_with_signatures(&src_network, &conn_sn, &msg, &signatures);
+}
+
+#[test]
+#[should_panic(expected = "HostError: Error(Contract, #11)")]
+fn test_receive_message_less_signatures() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+
+ ctx.init_context(&client);
+
+ let val1 = [4, 174, 54, 168, 191, 216, 207, 101, 134, 243, 76, 104, 133, 40, 137, 72, 53, 245, 231, 193, 157, 54, 104, 155, 172, 84, 96, 101, 107, 97, 60, 94, 171, 241, 250, 152, 34, 18, 170, 39, 202, 236, 226, 58, 39, 8, 235, 60, 137, 54, 225, 50, 185, 253, 130, 197, 174, 226, 170, 75, 6, 145, 123, 87, 19];
+ let val2 = [4, 91, 65, 155, 222, 192, 210, 187, 193, 108, 232, 174, 20, 79, 248, 232, 37, 18, 63, 208, 203, 62, 54, 208, 7, 91, 109, 141, 229, 170, 181, 51, 136, 172, 143, 180, 194, 138, 138, 56, 67, 243, 7, 60, 218, 164, 12, 148, 63, 116, 115, 127, 192, 206, 164, 169, 95, 135, 119, 138, 255, 172, 115, 129, 144];
+ let val3 = [4, 248, 192, 175, 198, 228, 250, 20, 158, 23, 251, 176, 244, 208, 150, 71, 151, 27, 208, 22, 41, 30, 154, 198, 109, 10, 112, 142, 200, 47, 200, 213, 210, 172, 135, 141, 129, 183, 211, 241, 211, 127, 16, 19, 67, 159, 195, 235, 88, 164, 223, 47, 128, 47, 147, 28, 121, 28, 93, 129, 176, 144, 52, 243, 55];
+
+ let mut validators = Vec::new(&ctx.env);
+ validators.push_back(BytesN::from_array(&ctx.env, &val1));
+ validators.push_back(BytesN::from_array(&ctx.env, &val2));
+ validators.push_back(BytesN::from_array(&ctx.env, &val3));
+ client.update_validators(&validators, &2_u32);
+
+ let conn_sn = 456456_u128;
+ let msg = Bytes::from_array(&ctx.env,&[104, 101, 108, 108, 111]);
+ let src_network = String::from_str(&ctx.env, "0x2.icon");
+
+ let mut signatures = Vec::new(&ctx.env);
+ signatures.push_back(BytesN::from_array(&ctx.env, &[35, 247, 49, 199, 251, 53, 83, 51, 115, 148, 35, 48, 85, 203, 185, 236, 5, 171, 221, 29, 247, 203, 190, 195, 208, 218, 204, 237, 88, 191, 91, 75, 48, 87, 108, 161, 75, 234, 147, 234, 65, 134, 233, 32, 249, 159, 43, 159, 86, 211, 1, 117, 176, 167, 53, 99, 34, 243, 165, 215, 93, 232, 67, 184, 27]));
+
+ client.recv_message_with_signatures(&src_network, &conn_sn, &msg, &signatures);
+}
+
+#[test]
+#[should_panic(expected = "HostError: Error(Contract, #11)")]
+fn test_receive_message_with_invalid_signature() {
+ let ctx = TestContext::default();
+ let client = ClusterConnectionClient::new(&ctx.env, &ctx.contract);
+
+ ctx.init_context(&client);
+
+ let val1 = [4, 174, 54, 168, 191, 216, 207, 101, 134, 243, 76, 104, 133, 40, 137, 72, 53, 245, 231, 193, 157, 54, 104, 155, 172, 84, 96, 101, 107, 97, 60, 94, 171, 241, 250, 152, 34, 18, 170, 39, 202, 236, 226, 58, 39, 8, 235, 60, 137, 54, 225, 50, 185, 253, 130, 197, 174, 226, 170, 75, 6, 145, 123, 87, 19];
+ let val2 = [4, 91, 65, 155, 222, 192, 210, 187, 193, 108, 232, 174, 20, 79, 248, 232, 37, 18, 63, 208, 203, 62, 54, 208, 7, 91, 109, 141, 229, 170, 181, 51, 136, 172, 143, 180, 194, 138, 138, 56, 67, 243, 7, 60, 218, 164, 12, 148, 63, 116, 115, 127, 192, 206, 164, 169, 95, 135, 119, 138, 255, 172, 115, 129, 144];
+ let val3 = [4, 248, 192, 175, 198, 228, 250, 20, 158, 23, 251, 176, 244, 208, 150, 71, 151, 27, 208, 22, 41, 30, 154, 198, 109, 10, 112, 142, 200, 47, 200, 213, 210, 172, 135, 141, 129, 183, 211, 241, 211, 127, 16, 19, 67, 159, 195, 235, 88, 164, 223, 47, 128, 47, 147, 28, 121, 28, 93, 129, 176, 144, 52, 243, 55];
+
+ let mut validators = Vec::new(&ctx.env);
+ validators.push_back(BytesN::from_array(&ctx.env, &val1));
+ validators.push_back(BytesN::from_array(&ctx.env, &val2));
+ validators.push_back(BytesN::from_array(&ctx.env, &val3));
+ client.update_validators(&validators, &1_u32);
+
+ let conn_sn = 456456_u128;
+ let msg = Bytes::from_array(&ctx.env,&[104, 100, 108, 108, 111]);
+ let src_network = String::from_str(&ctx.env, "0x2.icon");
+
+ let mut signatures = Vec::new(&ctx.env);
+ signatures.push_back(BytesN::from_array(&ctx.env, &[35, 247, 49, 199, 251, 53, 83, 51, 115, 148, 35, 48, 85, 203, 185, 236, 5, 171, 221, 29, 247, 203, 190, 195, 208, 218, 204, 237, 88, 191, 91, 75, 48, 87, 108, 161, 75, 234, 147, 234, 65, 134, 233, 32, 249, 159, 43, 159, 86, 211, 1, 117, 176, 167, 53, 99, 34, 243, 165, 215, 93, 232, 67, 184, 27]));
+
+ client.recv_message_with_signatures(&src_network, &conn_sn, &msg, &signatures);
+}
+
diff --git a/contracts/soroban/libs/soroban-rlp/src/utils.rs b/contracts/soroban/libs/soroban-rlp/src/utils.rs
index 403cac527..b359078ca 100644
--- a/contracts/soroban/libs/soroban-rlp/src/utils.rs
+++ b/contracts/soroban/libs/soroban-rlp/src/utils.rs
@@ -53,7 +53,7 @@ pub fn bytes_to_u64(bytes: Bytes) -> u64 {
}
pub fn u128_to_bytes(env: &Env, number: u128) -> Bytes {
- let mut bytes = bytes!(&env, 0x00);
+ let mut bytes: Bytes = Bytes::new(&env);
let mut i = 15;
let mut leading_zero = true;
while i >= 0 {
diff --git a/scripts/optimize-stellar.sh b/scripts/optimize-stellar.sh
index b527d71f3..691c6e4bd 100755
--- a/scripts/optimize-stellar.sh
+++ b/scripts/optimize-stellar.sh
@@ -11,7 +11,7 @@ cargo build --target wasm32-unknown-unknown --release
for WASM in $build_directory/*.wasm; do
NAME=$(basename "$WASM" .wasm)${SUFFIX}.wasm
echo "Optimizing $NAME ... $WASM"
- stellar contract optimize --wasm "$WASM"
+ /usr/local/bin/stellar2 contract optimize --wasm "$WASM"
done
cd -
From c7305abfbd2b7142eab5805bad6fc0f77d097543 Mon Sep 17 00:00:00 2001
From: DeepakBomjan <44976635+DeepakBomjan@users.noreply.github.com>
Date: Fri, 29 Nov 2024 11:25:20 +0545
Subject: [PATCH 07/17] ci: update rustc version to 1.81.0
---
.github/workflows/stellar-build-and-test.yml | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/.github/workflows/stellar-build-and-test.yml b/.github/workflows/stellar-build-and-test.yml
index bd23e8620..9c2cffe0e 100644
--- a/.github/workflows/stellar-build-and-test.yml
+++ b/.github/workflows/stellar-build-and-test.yml
@@ -27,7 +27,7 @@ jobs:
- name: Install stable toolchain
uses: actions-rs/toolchain@v1
with:
- toolchain: 1.79.0
+ toolchain: 1.81.0
target: wasm32-unknown-unknown
override: true
profile: minimal
From 2a357b9d614aa937ae65176e6beec7035714b91a Mon Sep 17 00:00:00 2001
From: gcranju <134275268+gcranju@users.noreply.github.com>
Date: Fri, 29 Nov 2024 21:51:32 +0545
Subject: [PATCH 08/17] fix: sui rlp upgrade (#417)
* fix: sui-rlp upgrade compatible
* fix: sui-rlp upgrade compatible
---
.../sui/libs/sui_rlp/sources/encoder.move | 10 +++----
contracts/sui/libs/sui_rlp/sources/utils.move | 29 +++++++++++++++----
2 files changed, 28 insertions(+), 11 deletions(-)
diff --git a/contracts/sui/libs/sui_rlp/sources/encoder.move b/contracts/sui/libs/sui_rlp/sources/encoder.move
index 5e2b93b1f..1e63672bf 100644
--- a/contracts/sui/libs/sui_rlp/sources/encoder.move
+++ b/contracts/sui/libs/sui_rlp/sources/encoder.move
@@ -49,7 +49,7 @@ module sui_rlp::encoder {
vector::append(&mut encoded_list,result);
} else {
- let length_bytes = utils::to_bytes_u64(len,false);
+ let length_bytes = utils::to_bytes_u64_sign(len,false);
let prefix = (0xf7 + vector::length(&length_bytes)) as u8;
vector::push_back(&mut encoded_list, prefix);
vector::append(&mut encoded_list, length_bytes);
@@ -69,7 +69,7 @@ module sui_rlp::encoder {
let len_u8=(len as u8);
vector::push_back(&mut length_info,(offset+len_u8));
}else {
- let length_bytes=utils::to_bytes_u64(len,false);
+ let length_bytes=utils::to_bytes_u64_sign(len,false);
let length_byte_len=vector::length(&length_bytes);
let length_byte_len=offset+(length_byte_len as u8);
vector::push_back(&mut length_info,length_byte_len);
@@ -86,19 +86,19 @@ module sui_rlp::encoder {
}
public fun encode_u32(num:u32):vector{
- let vec= utils::to_bytes_u32(num,true);
+ let vec= utils::to_bytes_u32_sign(num,true);
encode(&vec)
}
public fun encode_u64(num:u64):vector{
- let vec= utils::to_bytes_u64(num,true);
+ let vec= utils::to_bytes_u64_sign(num,true);
encode(&vec)
}
public fun encode_u128(num:u128):vector{
- let vec= utils::to_bytes_u128(num,true);
+ let vec= utils::to_bytes_u128_sign(num,true);
encode(&vec)
}
diff --git a/contracts/sui/libs/sui_rlp/sources/utils.move b/contracts/sui/libs/sui_rlp/sources/utils.move
index af37a07d3..53529c3d2 100644
--- a/contracts/sui/libs/sui_rlp/sources/utils.move
+++ b/contracts/sui/libs/sui_rlp/sources/utils.move
@@ -41,18 +41,35 @@ module sui_rlp::utils {
}
- public fun to_bytes_u128(number:u128,signed:bool):vector{
+ //Deprecated
+ public fun to_bytes_u128(number:u128):vector{
+ let bytes=bcs::to_bytes(&number);
+ to_signed_bytes(bytes,true)
+ }
+
+ public fun to_bytes_u128_sign(number:u128,signed:bool):vector{
let bytes=bcs::to_bytes(&number);
to_signed_bytes(bytes,signed)
}
+ //Deprecated
+ public fun to_bytes_u64(number:u64):vector{
+ let bytes=bcs::to_bytes(&number);
+ to_signed_bytes(bytes,true)
+ }
- public fun to_bytes_u64(number:u64,signed:bool):vector{
+ public fun to_bytes_u64_sign(number:u64,signed:bool):vector{
let bytes=bcs::to_bytes(&number);
to_signed_bytes(bytes,signed)
}
+
+ //Deprecated
+ public fun to_bytes_u32(number: u32): vector {
+ let bytes=bcs::to_bytes(&number);
+ to_signed_bytes(bytes,true)
+ }
- public fun to_bytes_u32(number: u32,signed:bool): vector {
+ public fun to_bytes_u32_sign(number: u32,signed:bool): vector {
let bytes=bcs::to_bytes(&number);
to_signed_bytes(bytes,signed)
}
@@ -120,7 +137,7 @@ module sui_rlp::utils_test {
#[test]
fun test_u32_conversion() {
let num= (122 as u32);
- let bytes= utils::to_bytes_u32(num,true);
+ let bytes= utils::to_bytes_u32_sign(num,true);
let converted=utils::from_bytes_u32(&bytes);
assert!(num==converted,0x01);
@@ -129,7 +146,7 @@ module sui_rlp::utils_test {
#[test]
fun test_u64_conversion() {
let num= (55000 as u64);
- let bytes= utils::to_bytes_u64(num,true);
+ let bytes= utils::to_bytes_u64_sign(num,true);
let converted=utils::from_bytes_u64(&bytes);
std::debug::print(&bytes);
std::debug::print(&converted);
@@ -140,7 +157,7 @@ module sui_rlp::utils_test {
#[test]
fun test_u128_conversion() {
let num= (1222223333 as u128);
- let bytes= utils::to_bytes_u128(num,true);
+ let bytes= utils::to_bytes_u128_sign(num,true);
std::debug::print(&bytes);
let converted=utils::from_bytes_u128(&bytes);
std::debug::print(&converted);
From b547292090ac29c797fc6c4ceceefe0ff4980ef8 Mon Sep 17 00:00:00 2001
From: gcranju
Date: Mon, 2 Dec 2024 15:37:39 +0545
Subject: [PATCH 09/17] fix: storage optimization in xcall
---
.../soroban/contracts/xcall/src/contract.rs | 5 ++
.../soroban/contracts/xcall/src/storage.rs | 75 +++++++------------
2 files changed, 31 insertions(+), 49 deletions(-)
diff --git a/contracts/soroban/contracts/xcall/src/contract.rs b/contracts/soroban/contracts/xcall/src/contract.rs
index 8d81f6ac2..6c5affce0 100644
--- a/contracts/soroban/contracts/xcall/src/contract.rs
+++ b/contracts/soroban/contracts/xcall/src/contract.rs
@@ -162,4 +162,9 @@ impl Xcall {
pub fn version(env: Env) -> u32 {
storage::get_contract_version(&env)
}
+
+ pub fn extend_instance_storage(env: Env) -> Result<(), ContractError> {
+ storage::extend_instance(&env);
+ Ok(())
+ }
}
diff --git a/contracts/soroban/contracts/xcall/src/storage.rs b/contracts/soroban/contracts/xcall/src/storage.rs
index cfa43a988..8879c6dc2 100644
--- a/contracts/soroban/contracts/xcall/src/storage.rs
+++ b/contracts/soroban/contracts/xcall/src/storage.rs
@@ -18,8 +18,8 @@ const LEDGER_BUMP_INSTANCE: u32 = LEDGER_THRESHOLD_INSTANCE + DAY_IN_LEDGERS; //
const LEDGER_THRESHOLD_PERSISTENT: u32 = DAY_IN_LEDGERS * 30; // ~ 30 days
const LEDGER_BUMP_PERSISTENT: u32 = LEDGER_THRESHOLD_PERSISTENT + DAY_IN_LEDGERS; // ~ 31 days
-const LEDGER_THRESHOLD_REQUEST: u32 = DAY_IN_LEDGERS * 7; // ~ 7 days
-const LEDGER_BUMP_REQUEST: u32 = LEDGER_THRESHOLD_REQUEST + DAY_IN_LEDGERS; // ~ 8 days
+const LEDGER_THRESHOLD_REQUEST: u32 = DAY_IN_LEDGERS * 3; // ~ 3 days
+const LEDGER_BUMP_REQUEST: u32 = LEDGER_THRESHOLD_REQUEST + DAY_IN_LEDGERS; // ~ 4 days
pub const MAX_ROLLBACK_SIZE: u64 = 1024;
pub const MAX_DATA_SIZE: u64 = 2048;
@@ -72,10 +72,9 @@ pub fn default_connection(e: &Env, nid: String) -> Result Result bool {
let key = StorageKey::SuccessfulResponses(sn);
- let res = e.storage().persistent().get(&key).unwrap_or(false);
- if res {
- extend_persistent(e, &key)
- }
-
+ let res = e.storage().instance().get(&key).unwrap_or(false);
res
}
@@ -114,31 +107,21 @@ pub fn get_proxy_request(e: &Env, req_id: u128) -> Result) -> Vec {
let key = StorageKey::PendingRequests(hash);
- let pending_request = e.storage().persistent().get(&key).unwrap_or(Vec::new(&e));
- if pending_request.len() > 0 {
- extend_persistent_request(e, &key);
- }
-
+ let pending_request = e.storage().temporary().get(&key).unwrap_or(Vec::new(&e));
pending_request
}
pub fn get_pending_response(e: &Env, hash: BytesN<32>) -> Vec {
let key = StorageKey::PendingResponses(hash);
- let pending_response = e.storage().persistent().get(&key).unwrap_or(Vec::new(&e));
- if pending_response.len() > 0 {
- extend_persistent_request(e, &key);
- }
-
+ let pending_response = e.storage().temporary().get(&key).unwrap_or(Vec::new(&e));
pending_response
}
@@ -196,53 +179,52 @@ pub fn store_protocol_fee(e: &Env, fee: u128) {
pub fn store_default_connection(e: &Env, nid: String, address: &Address) {
let key = StorageKey::DefaultConnections(nid);
- e.storage().persistent().set(&key, &address);
- extend_persistent(e, &key);
+ e.storage().instance().set(&key, &address);
}
pub fn store_rollback(e: &Env, sn: u128, rollback: &Rollback) {
let key = StorageKey::Rollback(sn);
- e.storage().persistent().set(&key, rollback);
- extend_persistent_request(e, &key)
+ e.storage().temporary().set(&key, rollback);
+ extend_temporary_request(e, &key)
}
pub fn remove_rollback(e: &Env, sn: u128) {
- e.storage().persistent().remove(&StorageKey::Rollback(sn));
+ e.storage().temporary().remove(&StorageKey::Rollback(sn));
}
pub fn store_proxy_request(e: &Env, req_id: u128, request: &CSMessageRequest) {
let key = StorageKey::ProxyRequest(req_id);
- e.storage().persistent().set(&key, request);
- extend_persistent_request(e, &key)
+ e.storage().temporary().set(&key, request);
+ extend_temporary_request(e, &key)
}
pub fn remove_proxy_request(e: &Env, req_id: u128) {
e.storage()
- .persistent()
+ .temporary()
.remove(&StorageKey::ProxyRequest(req_id))
}
pub fn store_pending_request(e: &Env, hash: BytesN<32>, sources: &Vec) {
let key = StorageKey::PendingRequests(hash.clone());
- e.storage().persistent().set(&key, sources);
- extend_persistent_request(e, &key)
+ e.storage().temporary().set(&key, sources);
+ extend_temporary_request(e, &key)
}
pub fn remove_pending_request(e: &Env, hash: BytesN<32>) {
e.storage()
- .persistent()
+ .temporary()
.remove(&StorageKey::PendingRequests(hash))
}
pub fn store_pending_response(e: &Env, hash: BytesN<32>, sources: &Vec) {
let key = StorageKey::PendingResponses(hash);
- e.storage().persistent().set(&key, sources);
- extend_persistent_request(e, &key)
+ e.storage().temporary().set(&key, sources);
+ extend_temporary_request(e, &key)
}
pub fn remove_pending_response(e: &Env, hash: BytesN<32>) {
e.storage()
- .persistent()
+ .temporary()
.remove(&StorageKey::PendingResponses(hash))
}
@@ -259,8 +241,8 @@ pub fn increment_last_request_id(e: &Env) -> u128 {
pub fn save_success_response(e: &Env, sn: u128) {
let key = StorageKey::SuccessfulResponses(sn);
- e.storage().persistent().set(&key, &true);
- extend_persistent(e, &key);
+ e.storage().instance().set(&key, &true);
+ extend_instance(e);
}
pub fn extend_instance(e: &Env) {
@@ -269,14 +251,9 @@ pub fn extend_instance(e: &Env) {
.extend_ttl(LEDGER_THRESHOLD_INSTANCE, LEDGER_BUMP_INSTANCE);
}
-pub fn extend_persistent(e: &Env, key: &StorageKey) {
- e.storage()
- .persistent()
- .extend_ttl(key, LEDGER_THRESHOLD_PERSISTENT, LEDGER_BUMP_PERSISTENT);
-}
-
-pub fn extend_persistent_request(e: &Env, key: &StorageKey) {
+pub fn extend_temporary_request(e: &Env, key: &StorageKey) {
e.storage()
- .persistent()
+ .temporary()
.extend_ttl(key, LEDGER_THRESHOLD_REQUEST, LEDGER_BUMP_REQUEST);
}
+
From c9a169e65f6426282285f454bf86b9cc51df4c66 Mon Sep 17 00:00:00 2001
From: gcranju
Date: Mon, 2 Dec 2024 15:58:38 +0545
Subject: [PATCH 10/17] fix: successful response to persistent
---
contracts/soroban/contracts/xcall/src/storage.rs | 12 +++++++++---
1 file changed, 9 insertions(+), 3 deletions(-)
diff --git a/contracts/soroban/contracts/xcall/src/storage.rs b/contracts/soroban/contracts/xcall/src/storage.rs
index 8879c6dc2..2e5ab5479 100644
--- a/contracts/soroban/contracts/xcall/src/storage.rs
+++ b/contracts/soroban/contracts/xcall/src/storage.rs
@@ -91,7 +91,7 @@ pub fn get_rollback(e: &Env, sequence_no: u128) -> Result bool {
let key = StorageKey::SuccessfulResponses(sn);
- let res = e.storage().instance().get(&key).unwrap_or(false);
+ let res = e.storage().persistent().get(&key).unwrap_or(false);
res
}
@@ -241,8 +241,8 @@ pub fn increment_last_request_id(e: &Env) -> u128 {
pub fn save_success_response(e: &Env, sn: u128) {
let key = StorageKey::SuccessfulResponses(sn);
- e.storage().instance().set(&key, &true);
- extend_instance(e);
+ e.storage().persistent().set(&key, &true);
+ extend_persistent(e);
}
pub fn extend_instance(e: &Env) {
@@ -251,6 +251,12 @@ pub fn extend_instance(e: &Env) {
.extend_ttl(LEDGER_THRESHOLD_INSTANCE, LEDGER_BUMP_INSTANCE);
}
+pub fn extend_persistent(e: &Env) {
+ e.storage()
+ .persistent()
+ .extend_ttl(LEDGER_THRESHOLD_PERSISTENT, LEDGER_BUMP_PERSISTENT);
+}
+
pub fn extend_temporary_request(e: &Env, key: &StorageKey) {
e.storage()
.temporary()
From 5f7cb9c53ef30d94337a8e708a669c224e9108b7 Mon Sep 17 00:00:00 2001
From: gcranju
Date: Mon, 2 Dec 2024 16:03:43 +0545
Subject: [PATCH 11/17] fix: successful response to persistent
---
contracts/soroban/contracts/xcall/src/storage.rs | 6 +++---
1 file changed, 3 insertions(+), 3 deletions(-)
diff --git a/contracts/soroban/contracts/xcall/src/storage.rs b/contracts/soroban/contracts/xcall/src/storage.rs
index 2e5ab5479..1fb56adda 100644
--- a/contracts/soroban/contracts/xcall/src/storage.rs
+++ b/contracts/soroban/contracts/xcall/src/storage.rs
@@ -242,7 +242,7 @@ pub fn increment_last_request_id(e: &Env) -> u128 {
pub fn save_success_response(e: &Env, sn: u128) {
let key = StorageKey::SuccessfulResponses(sn);
e.storage().persistent().set(&key, &true);
- extend_persistent(e);
+ extend_persistent(e, &key);
}
pub fn extend_instance(e: &Env) {
@@ -251,10 +251,10 @@ pub fn extend_instance(e: &Env) {
.extend_ttl(LEDGER_THRESHOLD_INSTANCE, LEDGER_BUMP_INSTANCE);
}
-pub fn extend_persistent(e: &Env) {
+pub fn extend_persistent(e: &Env, key: &StorageKey) {
e.storage()
.persistent()
- .extend_ttl(LEDGER_THRESHOLD_PERSISTENT, LEDGER_BUMP_PERSISTENT);
+ .extend_ttl(key, LEDGER_THRESHOLD_PERSISTENT, LEDGER_BUMP_PERSISTENT);
}
pub fn extend_temporary_request(e: &Env, key: &StorageKey) {
From 6a866bc5f942a96f4c23b7775803044612f74b1f Mon Sep 17 00:00:00 2001
From: gcranju
Date: Tue, 3 Dec 2024 12:31:56 +0545
Subject: [PATCH 12/17] fix: javascore settngs.gradle
---
contracts/javascore/settings.gradle | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/contracts/javascore/settings.gradle b/contracts/javascore/settings.gradle
index bfa809806..4f1da5a87 100644
--- a/contracts/javascore/settings.gradle
+++ b/contracts/javascore/settings.gradle
@@ -4,7 +4,7 @@ include(
'xcall',
'xcall-lib',
'centralized-connection',
- 'cluster-connection'
+ 'cluster-connection',
'aggregator'
)
From dc50331b72fcd31f68cda2c0725ec4246a8f44e3 Mon Sep 17 00:00:00 2001
From: gcranju
Date: Tue, 3 Dec 2024 12:44:47 +0545
Subject: [PATCH 13/17] fix: stellar script
---
scripts/optimize-stellar.sh | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/scripts/optimize-stellar.sh b/scripts/optimize-stellar.sh
index 691c6e4bd..b527d71f3 100755
--- a/scripts/optimize-stellar.sh
+++ b/scripts/optimize-stellar.sh
@@ -11,7 +11,7 @@ cargo build --target wasm32-unknown-unknown --release
for WASM in $build_directory/*.wasm; do
NAME=$(basename "$WASM" .wasm)${SUFFIX}.wasm
echo "Optimizing $NAME ... $WASM"
- /usr/local/bin/stellar2 contract optimize --wasm "$WASM"
+ stellar contract optimize --wasm "$WASM"
done
cd -
From 7662f6c7bc76a4487e2ba858e20e5bc2009fafc2 Mon Sep 17 00:00:00 2001
From: sherpalden
Date: Wed, 4 Dec 2024 10:29:31 +0545
Subject: [PATCH 14/17] fix: remove unwanted code
---
.../aggregator/src/main/java/relay/aggregator/Packet.java | 2 --
1 file changed, 2 deletions(-)
diff --git a/contracts/javascore/aggregator/src/main/java/relay/aggregator/Packet.java b/contracts/javascore/aggregator/src/main/java/relay/aggregator/Packet.java
index 0568de377..138ab2bb5 100644
--- a/contracts/javascore/aggregator/src/main/java/relay/aggregator/Packet.java
+++ b/contracts/javascore/aggregator/src/main/java/relay/aggregator/Packet.java
@@ -66,8 +66,6 @@ public Packet(String srcNetwork, String srcContractAddress, BigInteger srcSn, Bi
|| srcHeight == null || dstNetwork == null || dstContractAddress == null || data == null;
Context.require(!isIllegalArg,
"srcNetwork, contractAddress, srcSn, srcHeight, dstNetwork, and data cannot be null");
- if (isIllegalArg) {
- }
this.srcNetwork = srcNetwork;
this.srcContractAddress = srcContractAddress;
this.srcSn = srcSn;
From 1cea9f33b177bf79c70764ea4741a63ed2dba7fb Mon Sep 17 00:00:00 2001
From: sherpalden
Date: Wed, 4 Dec 2024 11:45:29 +0545
Subject: [PATCH 15/17] fix: do not included admin as relayers
---
.../main/java/relay/aggregator/RelayAggregator.java | 10 +---------
1 file changed, 1 insertion(+), 9 deletions(-)
diff --git a/contracts/javascore/aggregator/src/main/java/relay/aggregator/RelayAggregator.java b/contracts/javascore/aggregator/src/main/java/relay/aggregator/RelayAggregator.java
index ec475a999..7525d24d7 100644
--- a/contracts/javascore/aggregator/src/main/java/relay/aggregator/RelayAggregator.java
+++ b/contracts/javascore/aggregator/src/main/java/relay/aggregator/RelayAggregator.java
@@ -50,7 +50,6 @@ public RelayAggregator(Address _admin) {
if (admin.get() == null) {
admin.set(_admin);
signatureThreshold.set(DEFAULT_SIGNATURE_THRESHOLD);
- addRelayer(_admin);
}
}
@@ -60,12 +59,6 @@ public void setAdmin(Address _admin) {
Context.require(admin.get() != _admin, "admin already set");
- // add new admin as relayer
- addRelayer(_admin);
-
- // remove old admin from relayer list
- removeRelayer(admin.get());
-
admin.set(_admin);
}
@@ -107,10 +100,9 @@ public void setRelayers(Address[] newRelayers, int threshold) {
addRelayer(newRelayer);
}
- Address adminAdrr = admin.get();
for (int i = 0; i < relayers.size(); i++) {
Address oldRelayer = relayers.get(i);
- if (!oldRelayer.equals(adminAdrr) && !newRelayersMap.containsKey(oldRelayer)) {
+ if (!newRelayersMap.containsKey(oldRelayer)) {
removeRelayer(oldRelayer);
}
}
From 9dfbb74a9e2a218a891b677752fc367995b55688 Mon Sep 17 00:00:00 2001
From: sherpalden
Date: Wed, 4 Dec 2024 12:23:22 +0545
Subject: [PATCH 16/17] fix: create packet id from the hash of entire packet &&
update unit tests
---
.../main/java/relay/aggregator/Packet.java | 15 +-
.../relay/aggregator/RelayAggregator.java | 28 +-
.../relay/aggregator/RelayAggregatorTest.java | 596 +++++++++---------
3 files changed, 320 insertions(+), 319 deletions(-)
diff --git a/contracts/javascore/aggregator/src/main/java/relay/aggregator/Packet.java b/contracts/javascore/aggregator/src/main/java/relay/aggregator/Packet.java
index 138ab2bb5..16d600cf7 100644
--- a/contracts/javascore/aggregator/src/main/java/relay/aggregator/Packet.java
+++ b/contracts/javascore/aggregator/src/main/java/relay/aggregator/Packet.java
@@ -2,6 +2,7 @@
import java.math.BigInteger;
+import score.ByteArrayObjectWriter;
import score.Context;
import score.ObjectReader;
import score.ObjectWriter;
@@ -75,12 +76,8 @@ public Packet(String srcNetwork, String srcContractAddress, BigInteger srcSn, Bi
this.data = data;
}
- public String getId() {
- return createId(this.srcNetwork, this.srcContractAddress, this.srcSn);
- }
-
- public static String createId(String srcNetwork, String contractAddress, BigInteger srcSn) {
- return srcNetwork + "/" + contractAddress + "/" + srcSn.toString();
+ public byte[] getId() {
+ return Context.hash("sha-256", this.toBytes());
}
/**
@@ -171,4 +168,10 @@ public static Packet readObject(ObjectReader r) {
r.end();
return p;
}
+
+ public byte[] toBytes() {
+ ByteArrayObjectWriter writer = Context.newByteArrayObjectWriter("RLPn");
+ Packet.writeObject(writer, this);
+ return writer.toByteArray();
+ }
}
diff --git a/contracts/javascore/aggregator/src/main/java/relay/aggregator/RelayAggregator.java b/contracts/javascore/aggregator/src/main/java/relay/aggregator/RelayAggregator.java
index 7525d24d7..440b8c199 100644
--- a/contracts/javascore/aggregator/src/main/java/relay/aggregator/RelayAggregator.java
+++ b/contracts/javascore/aggregator/src/main/java/relay/aggregator/RelayAggregator.java
@@ -41,9 +41,9 @@ public class RelayAggregator {
private final ArrayDB relayers = Context.newArrayDB("relayers", Address.class);
private final DictDB relayersLookup = Context.newDictDB("relayersLookup", Boolean.class);
- private final DictDB packets = Context.newDictDB("packets", Packet.class);
+ private final DictDB packets = Context.newDictDB("packets", Packet.class);
- private final BranchDB> signatures = Context.newBranchDB("signatures",
+ private final BranchDB> signatures = Context.newBranchDB("signatures",
byte[].class);
public RelayAggregator(Address _admin) {
@@ -119,8 +119,13 @@ public boolean packetSubmitted(
Address relayer,
String srcNetwork,
String srcContractAddress,
- BigInteger srcSn) {
- String pktID = Packet.createId(srcNetwork, srcContractAddress, srcSn);
+ BigInteger srcSn,
+ BigInteger srcHeight,
+ String dstNetwork,
+ String dstContractAddress,
+ byte[] data) {
+ Packet pkt = new Packet(srcNetwork, srcContractAddress, srcSn, srcHeight, dstNetwork, dstContractAddress, data);
+ byte[] pktID = pkt.getId();
byte[] existingSign = signatures.at(pktID).get(relayer);
return existingSign != null;
}
@@ -139,7 +144,7 @@ public void submitPacket(
relayersOnly();
Packet pkt = new Packet(srcNetwork, srcContractAddress, srcSn, srcHeight, dstNetwork, dstContractAddress, data);
- String pktID = pkt.getId();
+ byte[] pktID = pkt.getId();
if (packets.get(pktID) == null) {
packets.set(pktID, pkt);
@@ -162,7 +167,7 @@ public void submitPacket(
setSignature(pktID, Context.getCaller(), signature);
if (signatureThresholdReached(pktID)) {
- byte[][] sigs = getSignatures(srcNetwork, srcContractAddress, srcSn);
+ byte[][] sigs = getSignatures(pktID);
byte[] encodedSigs = serializeSignatures(sigs);
PacketAcknowledged(
pkt.getSrcNetwork(),
@@ -177,8 +182,7 @@ public void submitPacket(
}
}
- private byte[][] getSignatures(String srcNetwork, String srcContractAddress, BigInteger srcSn) {
- String pktID = Packet.createId(srcNetwork, srcContractAddress, srcSn);
+ private byte[][] getSignatures(byte[] pktID) {
DictDB signDict = signatures.at(pktID);
ArrayList signatureList = new ArrayList();
@@ -197,7 +201,7 @@ private byte[][] getSignatures(String srcNetwork, String srcContractAddress, Big
return sigs;
}
- protected void setSignature(String pktID, Address addr, byte[] sign) {
+ protected void setSignature(byte[] pktID, Address addr, byte[] sign) {
signatures.at(pktID).set(addr, sign);
}
@@ -233,7 +237,7 @@ protected static byte[][] deserializeSignatures(byte[] encodedSigs) {
}
private void adminOnly() {
- Context.require(Context.getCaller().equals(admin.get()), "Unauthorized: caller is not the leader relayer");
+ Context.require(Context.getCaller().equals(admin.get()), "Unauthorized: caller is not the admin");
}
private void relayersOnly() {
@@ -262,7 +266,7 @@ private void removeRelayer(Address oldRelayer) {
}
}
- private Boolean signatureThresholdReached(String pktID) {
+ private Boolean signatureThresholdReached(byte[] pktID) {
int noOfSignatures = 0;
for (int i = 0; i < relayers.size(); i++) {
Address relayer = relayers.get(i);
@@ -274,7 +278,7 @@ private Boolean signatureThresholdReached(String pktID) {
return noOfSignatures >= signatureThreshold.get();
}
- private void removePacket(String pktID) {
+ private void removePacket(byte[] pktID) {
packets.set(pktID, null);
DictDB signDict = signatures.at(pktID);
diff --git a/contracts/javascore/aggregator/src/test/java/relay/aggregator/RelayAggregatorTest.java b/contracts/javascore/aggregator/src/test/java/relay/aggregator/RelayAggregatorTest.java
index 0da3391f4..4165d4e66 100644
--- a/contracts/javascore/aggregator/src/test/java/relay/aggregator/RelayAggregatorTest.java
+++ b/contracts/javascore/aggregator/src/test/java/relay/aggregator/RelayAggregatorTest.java
@@ -1,7 +1,6 @@
package relay.aggregator;
import java.math.BigInteger;
-import java.util.Arrays;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
@@ -21,347 +20,342 @@
import static org.junit.jupiter.api.Assertions.assertArrayEquals;
import static org.junit.jupiter.api.Assertions.assertEquals;
-import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertThrows;
-import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.mockito.Mockito.spy;
import static org.mockito.Mockito.verify;
class RelayAggregatorTest extends TestBase {
- private final ServiceManager sm = getServiceManager();
+ private final ServiceManager sm = getServiceManager();
- private KeyWallet admin;
- private Account adminAc;
+ private KeyWallet admin;
+ private Account adminAc;
- private KeyWallet relayerOne;
- private Account relayerOneAc;
+ private KeyWallet relayerOne;
+ private Account relayerOneAc;
- private KeyWallet relayerTwo;
- private Account relayerTwoAc;
+ private KeyWallet relayerTwo;
+ private Account relayerTwoAc;
- private KeyWallet relayerThree;
- private Account relayerThreeAc;
+ private KeyWallet relayerThree;
+ private Account relayerThreeAc;
- private KeyWallet relayerFour;
- private Account relayerFourAc;
+ private KeyWallet relayerFour;
+ private Account relayerFourAc;
- private Score aggregator;
- private RelayAggregator aggregatorSpy;
+ private Score aggregator;
+ private RelayAggregator aggregatorSpy;
- @BeforeEach
- void setup() throws Exception {
- admin = KeyWallet.create();
- adminAc = sm.getAccount(Address.fromString(admin.getAddress().toString()));
+ @BeforeEach
+ void setup() throws Exception {
+ admin = KeyWallet.create();
+ adminAc = sm.getAccount(Address.fromString(admin.getAddress().toString()));
- relayerOne = KeyWallet.create();
- relayerOneAc = sm.getAccount(Address.fromString(relayerOne.getAddress().toString()));
+ relayerOne = KeyWallet.create();
+ relayerOneAc = sm.getAccount(Address.fromString(relayerOne.getAddress().toString()));
- relayerTwo = KeyWallet.create();
- relayerTwoAc = sm.getAccount(Address.fromString(relayerTwo.getAddress().toString()));
+ relayerTwo = KeyWallet.create();
+ relayerTwoAc = sm.getAccount(Address.fromString(relayerTwo.getAddress().toString()));
- relayerThree = KeyWallet.create();
- relayerThreeAc = sm.getAccount(Address.fromString(relayerThree.getAddress().toString()));
+ relayerThree = KeyWallet.create();
+ relayerThreeAc = sm.getAccount(Address.fromString(relayerThree.getAddress().toString()));
- relayerFour = KeyWallet.create();
- relayerFourAc = sm.getAccount(Address.fromString(relayerFour.getAddress().toString()));
+ relayerFour = KeyWallet.create();
+ relayerFourAc = sm.getAccount(Address.fromString(relayerFour.getAddress().toString()));
- aggregator = sm.deploy(adminAc, RelayAggregator.class, adminAc.getAddress());
+ aggregator = sm.deploy(adminAc, RelayAggregator.class, adminAc.getAddress());
- Address[] relayers = new Address[] { adminAc.getAddress(), relayerOneAc.getAddress(), relayerTwoAc.getAddress(),
- relayerThreeAc.getAddress() };
+ Address[] relayers = new Address[] { relayerOneAc.getAddress(), relayerTwoAc.getAddress(),
+ relayerThreeAc.getAddress() };
- aggregator.invoke(adminAc, "setRelayers", (Object) relayers, 2);
+ aggregator.invoke(adminAc, "setRelayers", (Object) relayers, 2);
- aggregatorSpy = (RelayAggregator) spy(aggregator.getInstance());
- aggregator.setInstance(aggregatorSpy);
- }
+ aggregatorSpy = (RelayAggregator) spy(aggregator.getInstance());
+ aggregator.setInstance(aggregatorSpy);
+ }
+
+ @Test
+ public void testSetAdmin() {
+ Account newAdminAc = sm.createAccount();
+ aggregator.invoke(adminAc, "setAdmin", newAdminAc.getAddress());
+
+ Address newAdmin = (Address) aggregator.call("getAdmin");
+ assertEquals(newAdminAc.getAddress(), newAdmin);
+ }
+
+ @Test
+ public void testSetAdmin_unauthorized() {
+ Account normalAc = sm.createAccount();
+ Account newAdminAc = sm.createAccount();
+
+ Executable action = () -> aggregator.invoke(normalAc, "setAdmin", newAdminAc.getAddress());
+ UserRevertedException e = assertThrows(UserRevertedException.class, action);
+
+ assertEquals("Reverted(0): Unauthorized: caller is not the admin", e.getMessage());
+ }
+
+ @Test
+ public void testSetSignatureThreshold() {
+ int threshold = 3;
+ aggregator.invoke(adminAc, "setSignatureThreshold", threshold);
+
+ Integer result = (Integer) aggregator.call("getSignatureThreshold");
+ assertEquals(threshold, result);
+ }
+
+ @Test
+ public void testSetSignatureThreshold_unauthorised() {
+ int threshold = 3;
+
+ Executable action = () -> aggregator.invoke(relayerOneAc,
+ "setSignatureThreshold", threshold);
+ UserRevertedException e = assertThrows(UserRevertedException.class, action);
+
+ assertEquals("Reverted(0): Unauthorized: caller is not the admin",
+ e.getMessage());
+ }
+
+ @Test
+ public void testSetRelayers() {
+ Account relayerFiveAc = sm.createAccount();
+ Address[] newRelayers = new Address[] { relayerThreeAc.getAddress(), relayerFourAc.getAddress(),
+ relayerFiveAc.getAddress() };
+
+ Integer threshold = 3;
+ aggregator.invoke(adminAc, "setRelayers", (Object) newRelayers, threshold);
+
+ Address[] updatedRelayers = (Address[]) aggregator.call("getRelayers");
- @Test
- public void testSetAdmin() {
- Address oldAdmin = (Address) aggregator.call("getAdmin");
+ Address[] expectedRelayers = new Address[] { relayerThreeAc.getAddress(),
+ relayerFourAc.getAddress(),
+ relayerFiveAc.getAddress() };
- Account newAdminAc = sm.createAccount();
- aggregator.invoke(adminAc, "setAdmin", newAdminAc.getAddress());
+ HashSet updatedRelayersSet = new HashSet();
+ for (Address rlr : updatedRelayers) {
+ updatedRelayersSet.add(rlr);
+ }
- Address newAdmin = (Address) aggregator.call("getAdmin");
- assertEquals(newAdminAc.getAddress(), newAdmin);
+ HashSet expectedRelayersSet = new HashSet();
+ for (Address rlr : expectedRelayers) {
+ expectedRelayersSet.add(rlr);
+ }
+
+ assertEquals(expectedRelayersSet, updatedRelayersSet);
+
+ Integer result = (Integer) aggregator.call("getSignatureThreshold");
+ assertEquals(threshold, result);
+ }
- Address[] relayers = (Address[]) aggregator.call("getRelayers");
+ @Test
+ public void testSetRelayers_unauthorized() {
+ Account relayerFiveAc = sm.createAccount();
+ Address[] newRelayers = new Address[] { relayerThreeAc.getAddress(), relayerFourAc.getAddress(),
+ relayerFiveAc.getAddress() };
- boolean containsNewAdmin = Arrays.asList(relayers).contains(newAdmin);
- boolean containsOldAdmin = Arrays.asList(relayers).contains(oldAdmin);
+ Integer threshold = 3;
+ Executable action = () -> aggregator.invoke(relayerOneAc, "setRelayers",
+ (Object) newRelayers, threshold);
+ UserRevertedException e = assertThrows(UserRevertedException.class, action);
- assertTrue(containsNewAdmin);
- assertFalse(containsOldAdmin);
- }
+ assertEquals("Reverted(0): Unauthorized: caller is not the admin",
+ e.getMessage());
- @Test
- public void testSetAdmin_unauthorized() {
- Account normalAc = sm.createAccount();
- Account newAdminAc = sm.createAccount();
+ }
- Executable action = () -> aggregator.invoke(normalAc, "setAdmin", newAdminAc.getAddress());
- UserRevertedException e = assertThrows(UserRevertedException.class, action);
+ @Test
+ public void testSetRelayers_invalidThreshold() {
+ Account relayerFiveAc = sm.createAccount();
+ Address[] newRelayers = new Address[] { relayerThreeAc.getAddress(), relayerFourAc.getAddress(),
+ relayerFiveAc.getAddress() };
- assertEquals("Reverted(0): Unauthorized: caller is not the leader relayer", e.getMessage());
- }
+ Integer threshold = 5;
+ Executable action = () -> aggregator.invoke(adminAc, "setRelayers",
+ (Object) newRelayers, threshold);
+ UserRevertedException e = assertThrows(UserRevertedException.class, action);
- @Test
- public void testSetSignatureThreshold() {
- int threshold = 3;
- aggregator.invoke(adminAc, "setSignatureThreshold", threshold);
+ assertEquals("Reverted(0): threshold value should be at least 1 and not greater than relayers size",
+ e.getMessage());
- Integer result = (Integer) aggregator.call("getSignatureThreshold");
- assertEquals(threshold, result);
- }
+ }
- @Test
- public void testSetSignatureThreshold_unauthorised() {
- int threshold = 3;
+ @Test
+ public void testSetRelayers_invalidThresholdZero() {
+ Account relayerFiveAc = sm.createAccount();
+ Address[] newRelayers = new Address[] { relayerThreeAc.getAddress(), relayerFourAc.getAddress(),
+ relayerFiveAc.getAddress() };
- Executable action = () -> aggregator.invoke(relayerOneAc,
- "setSignatureThreshold", threshold);
- UserRevertedException e = assertThrows(UserRevertedException.class, action);
+ Integer threshold = 0;
+ Executable action = () -> aggregator.invoke(adminAc, "setRelayers",
+ (Object) newRelayers, threshold);
+ UserRevertedException e = assertThrows(UserRevertedException.class, action);
- assertEquals("Reverted(0): Unauthorized: caller is not the leader relayer",
- e.getMessage());
- }
+ assertEquals("Reverted(0): threshold value should be at least 1 and not greater than relayers size",
+ e.getMessage());
- @Test
- public void testSetRelayers() {
- Account relayerFiveAc = sm.createAccount();
- Address[] newRelayers = new Address[] { relayerThreeAc.getAddress(), relayerFourAc.getAddress(),
- relayerFiveAc.getAddress() };
+ }
- Integer threshold = 3;
- aggregator.invoke(adminAc, "setRelayers", (Object) newRelayers, threshold);
+ @Test
+ public void testPacketSubmitted_true() throws Exception {
+ String srcNetwork = "0x2.icon";
+ String dstNetwork = "sui";
+ BigInteger srcSn = BigInteger.ONE;
+ BigInteger srcHeight = BigInteger.ONE;
+ String srcContractAddress = "hxjuiod";
+ String dstContractAddress = "hxjuiod";
+ byte[] data = new byte[] { 0x01, 0x02 };
+
+ aggregator.invoke(adminAc, "setSignatureThreshold", 2);
+
+ byte[] dataHash = Context.hash("sha-256", data);
+ byte[] sign = relayerOne.sign(dataHash);
+
+ aggregator.invoke(relayerOneAc, "submitPacket", srcNetwork,
+ srcContractAddress, srcSn, srcHeight, dstNetwork,
+ dstContractAddress, data,
+ sign);
+
+ boolean submitted = (boolean) aggregator.call("packetSubmitted",
+ relayerOneAc.getAddress(), srcNetwork,
+ srcContractAddress, srcSn, srcHeight, dstNetwork, dstContractAddress, data);
+ assertEquals(submitted, true);
+ }
- Address[] updatedRelayers = (Address[]) aggregator.call("getRelayers");
+ @Test
+ public void testPacketSubmitted_false() throws Exception {
+ String srcNetwork = "0x2.icon";
+ String dstNetwork = "sui";
+ BigInteger srcSn = BigInteger.ONE;
+ BigInteger srcHeight = BigInteger.ONE;
+ String srcContractAddress = "hxjuiod";
+ String dstContractAddress = "hxjuiod";
+ byte[] data = new byte[] { 0x01, 0x02 };
+
+ boolean submitted = (boolean) aggregator.call("packetSubmitted",
+ relayerOneAc.getAddress(), srcNetwork,
+ srcContractAddress, srcSn, srcHeight, dstNetwork, dstContractAddress, data);
+ assertEquals(submitted, false);
+ }
- Address[] expectedRelayers = new Address[] { adminAc.getAddress(), relayerThreeAc.getAddress(),
- relayerFourAc.getAddress(),
- relayerFiveAc.getAddress() };
+ @Test
+ public void testSubmitPacket() throws Exception {
+ String srcNetwork = "0x2.icon";
+ String dstNetwork = "sui";
+ BigInteger srcSn = BigInteger.ONE;
+ BigInteger srcHeight = BigInteger.ONE;
+ String srcContractAddress = "hxjuiod";
+ String dstContractAddress = "hxjuiod";
+ byte[] data = new byte[] { 0x01, 0x02 };
+
+ aggregator.invoke(adminAc, "setSignatureThreshold", 2);
+
+ byte[] dataHash = Context.hash("sha-256", data);
+ byte[] sign = relayerOne.sign(dataHash);
+
+ aggregator.invoke(relayerOneAc, "submitPacket", srcNetwork,
+ srcContractAddress, srcSn, srcHeight, dstNetwork,
+ dstContractAddress, data,
+ sign);
+
+ Packet pkt = new Packet(srcNetwork, srcContractAddress, srcSn, srcHeight, dstNetwork,
+ dstContractAddress, data);
+ byte[] pktID = pkt.getId();
+ verify(aggregatorSpy).PacketRegistered(srcNetwork, srcContractAddress, srcSn,
+ srcHeight, dstNetwork,
+ dstContractAddress, data);
+ verify(aggregatorSpy).setSignature(pktID, relayerOneAc.getAddress(), sign);
+ }
- HashSet updatedRelayersSet = new HashSet();
- for (Address rlr : updatedRelayers) {
- updatedRelayersSet.add(rlr);
+ @Test
+ public void testSubmitPacket_thresholdReached() throws Exception {
+ String srcNetwork = "0x2.icon";
+ String dstNetwork = "sui";
+ BigInteger srcSn = BigInteger.ONE;
+ BigInteger srcHeight = BigInteger.ONE;
+ String srcContractAddress = "hxjuiod";
+ String dstContractAddress = "hxjuiod";
+ byte[] data = new byte[] { 0x01, 0x02 };
+
+ aggregator.invoke(adminAc, "setSignatureThreshold", 2);
+
+ byte[] dataHash = Context.hash("sha-256", data);
+
+ byte[] signOne = relayerOne.sign(dataHash);
+ aggregator.invoke(relayerOneAc, "submitPacket", srcNetwork,
+ srcContractAddress, srcSn, srcHeight, dstNetwork,
+ dstContractAddress,
+ data,
+ signOne);
+
+ byte[] signTwo = relayerTwo.sign(dataHash);
+ aggregator.invoke(relayerTwoAc, "submitPacket", srcNetwork,
+ srcContractAddress, srcSn, srcHeight, dstNetwork,
+ dstContractAddress,
+ data,
+ signTwo);
+
+ byte[][] sigs = new byte[2][];
+ sigs[0] = signOne;
+ sigs[1] = signTwo;
+
+ byte[] encodedSigs = RelayAggregator.serializeSignatures(sigs);
+ byte[][] decodedSigs = RelayAggregator.deserializeSignatures(encodedSigs);
+
+ assertArrayEquals(signOne, decodedSigs[0]);
+ assertArrayEquals(signTwo, decodedSigs[1]);
+
+ verify(aggregatorSpy).PacketAcknowledged(srcNetwork, srcContractAddress,
+ srcSn, srcHeight, dstNetwork,
+ dstContractAddress, data,
+ encodedSigs);
}
- HashSet expectedRelayersSet = new HashSet();
- for (Address rlr : expectedRelayers) {
- expectedRelayersSet.add(rlr);
+ @Test
+ public void testSubmitPacket_unauthorized() throws Exception {
+ String srcNetwork = "0x2.icon";
+ String dstNetwork = "sui";
+ BigInteger srcSn = BigInteger.ONE;
+ BigInteger srcHeight = BigInteger.ONE;
+ String srcContractAddress = "hxjuiod";
+ String dstContractAddress = "hxjuiod";
+ byte[] data = new byte[] { 0x01, 0x02 };
+
+ byte[] dataHash = Context.hash("sha-256", data);
+ byte[] sign = relayerFour.sign(dataHash);
+
+ Executable action = () -> aggregator.invoke(relayerFourAc, "submitPacket",
+ srcNetwork, srcContractAddress,
+ srcSn,
+ srcHeight, dstNetwork, dstContractAddress, data, sign);
+
+ UserRevertedException e = assertThrows(UserRevertedException.class, action);
+ assertEquals("Reverted(0): Unauthorized: caller is not a registered relayer",
+ e.getMessage());
}
- assertEquals(expectedRelayersSet, updatedRelayersSet);
-
- Integer result = (Integer) aggregator.call("getSignatureThreshold");
- assertEquals(threshold, result);
- }
-
- @Test
- public void testSetRelayers_unauthorized() {
- Account relayerFiveAc = sm.createAccount();
- Address[] newRelayers = new Address[] { relayerThreeAc.getAddress(), relayerFourAc.getAddress(),
- relayerFiveAc.getAddress() };
-
- Integer threshold = 3;
- Executable action = () -> aggregator.invoke(relayerOneAc, "setRelayers",
- (Object) newRelayers, threshold);
- UserRevertedException e = assertThrows(UserRevertedException.class, action);
-
- assertEquals("Reverted(0): Unauthorized: caller is not the leader relayer",
- e.getMessage());
-
- }
-
- @Test
- public void testSetRelayers_invalidThreshold() {
- Account relayerFiveAc = sm.createAccount();
- Address[] newRelayers = new Address[] { relayerThreeAc.getAddress(), relayerFourAc.getAddress(),
- relayerFiveAc.getAddress() };
-
- Integer threshold = 5;
- Executable action = () -> aggregator.invoke(adminAc, "setRelayers",
- (Object) newRelayers, threshold);
- UserRevertedException e = assertThrows(UserRevertedException.class, action);
-
- assertEquals("Reverted(0): threshold value should be at least 1 and not greater than relayers size",
- e.getMessage());
-
- }
-
- @Test
- public void testSetRelayers_invalidThresholdZero() {
- Account relayerFiveAc = sm.createAccount();
- Address[] newRelayers = new Address[] { relayerThreeAc.getAddress(), relayerFourAc.getAddress(),
- relayerFiveAc.getAddress() };
-
- Integer threshold = 0;
- Executable action = () -> aggregator.invoke(adminAc, "setRelayers",
- (Object) newRelayers, threshold);
- UserRevertedException e = assertThrows(UserRevertedException.class, action);
-
- assertEquals("Reverted(0): threshold value should be at least 1 and not greater than relayers size",
- e.getMessage());
-
- }
-
- @Test
- public void testPacketSubmitted_true() throws Exception {
- String srcNetwork = "0x2.icon";
- String dstNetwork = "sui";
- BigInteger srcSn = BigInteger.ONE;
- BigInteger srcHeight = BigInteger.ONE;
- String srcContractAddress = "hxjuiod";
- String dstContractAddress = "hxjuiod";
- byte[] data = new byte[] { 0x01, 0x02 };
-
- aggregator.invoke(adminAc, "setSignatureThreshold", 2);
-
- byte[] dataHash = Context.hash("sha-256", data);
- byte[] sign = relayerOne.sign(dataHash);
-
- aggregator.invoke(relayerOneAc, "submitPacket", srcNetwork,
- srcContractAddress, srcSn, srcHeight, dstNetwork,
- dstContractAddress, data,
- sign);
-
- boolean submitted = (boolean) aggregator.call("packetSubmitted",
- relayerOneAc.getAddress(), srcNetwork,
- srcContractAddress, srcSn);
- assertEquals(submitted, true);
- }
-
- @Test
- public void testPacketSubmitted_false() throws Exception {
- String srcNetwork = "0x2.icon";
- BigInteger srcSn = BigInteger.ONE;
- String srcContractAddress = "hxjuiod";
-
- boolean submitted = (boolean) aggregator.call("packetSubmitted",
- relayerOneAc.getAddress(), srcNetwork,
- srcContractAddress, srcSn);
- assertEquals(submitted, false);
- }
-
- @Test
- public void testSubmitPacket() throws Exception {
- String srcNetwork = "0x2.icon";
- String dstNetwork = "sui";
- BigInteger srcSn = BigInteger.ONE;
- BigInteger srcHeight = BigInteger.ONE;
- String srcContractAddress = "hxjuiod";
- String dstContractAddress = "hxjuiod";
- byte[] data = new byte[] { 0x01, 0x02 };
-
- aggregator.invoke(adminAc, "setSignatureThreshold", 2);
-
- byte[] dataHash = Context.hash("sha-256", data);
- byte[] sign = relayerOne.sign(dataHash);
-
- aggregator.invoke(relayerOneAc, "submitPacket", srcNetwork,
- srcContractAddress, srcSn, srcHeight, dstNetwork,
- dstContractAddress, data,
- sign);
-
- String pktID = Packet.createId(srcNetwork, srcContractAddress, srcSn);
- verify(aggregatorSpy).PacketRegistered(srcNetwork, srcContractAddress, srcSn,
- srcHeight, dstNetwork,
- dstContractAddress, data);
- verify(aggregatorSpy).setSignature(pktID, relayerOneAc.getAddress(), sign);
- }
-
- @Test
- public void testSubmitPacket_thresholdReached() throws Exception {
- String srcNetwork = "0x2.icon";
- String dstNetwork = "sui";
- BigInteger srcSn = BigInteger.ONE;
- BigInteger srcHeight = BigInteger.ONE;
- String srcContractAddress = "hxjuiod";
- String dstContractAddress = "hxjuiod";
- byte[] data = new byte[] { 0x01, 0x02 };
-
- aggregator.invoke(adminAc, "setSignatureThreshold", 2);
-
- byte[] dataHash = Context.hash("sha-256", data);
-
- byte[] signAdmin = admin.sign(dataHash);
- aggregator.invoke(adminAc, "submitPacket", srcNetwork, srcContractAddress,
- srcSn, srcHeight, dstNetwork,
- dstContractAddress, data,
- signAdmin);
-
- byte[] signOne = relayerOne.sign(dataHash);
- aggregator.invoke(relayerOneAc, "submitPacket", srcNetwork,
- srcContractAddress, srcSn, srcHeight, dstNetwork,
- dstContractAddress,
- data,
- signOne);
-
- byte[][] sigs = new byte[2][];
- sigs[0] = signAdmin;
- sigs[1] = signOne;
-
- byte[] encodedSigs = RelayAggregator.serializeSignatures(sigs);
- byte[][] decodedSigs = RelayAggregator.deserializeSignatures(encodedSigs);
-
- assertArrayEquals(signAdmin, decodedSigs[0]);
- assertArrayEquals(signOne, decodedSigs[1]);
-
- verify(aggregatorSpy).PacketAcknowledged(srcNetwork, srcContractAddress,
- srcSn, srcHeight, dstNetwork,
- dstContractAddress, data,
- encodedSigs);
- }
-
- @Test
- public void testSubmitPacket_unauthorized() throws Exception {
- String srcNetwork = "0x2.icon";
- String dstNetwork = "sui";
- BigInteger srcSn = BigInteger.ONE;
- BigInteger srcHeight = BigInteger.ONE;
- String srcContractAddress = "hxjuiod";
- String dstContractAddress = "hxjuiod";
- byte[] data = new byte[] { 0x01, 0x02 };
-
- byte[] dataHash = Context.hash("sha-256", data);
- byte[] sign = relayerFour.sign(dataHash);
-
- Executable action = () -> aggregator.invoke(relayerFourAc, "submitPacket",
- srcNetwork, srcContractAddress,
- srcSn,
- srcHeight, dstNetwork, dstContractAddress, data, sign);
-
- UserRevertedException e = assertThrows(UserRevertedException.class, action);
- assertEquals("Reverted(0): Unauthorized: caller is not a registered relayer",
- e.getMessage());
- }
-
- @Test
- public void testSubmitPacket_duplicate() throws Exception {
- String srcNetwork = "0x2.icon";
- String dstNetwork = "sui";
- BigInteger srcSn = BigInteger.ONE;
- BigInteger srcHeight = BigInteger.ONE;
- String srcContractAddress = "hxjuiod";
- String dstContractAddress = "hxjuiod";
- byte[] data = new byte[] { 0x01, 0x02 };
-
- aggregator.invoke(adminAc, "setSignatureThreshold", 2);
-
- byte[] dataHash = Context.hash("sha-256", data);
- byte[] sign = relayerOne.sign(dataHash);
-
- aggregator.invoke(relayerOneAc, "submitPacket", srcNetwork,
- srcContractAddress, srcSn, srcHeight, dstNetwork,
- dstContractAddress, data, sign);
-
- Executable action = () -> aggregator.invoke(relayerOneAc, "submitPacket",
- srcNetwork, srcContractAddress, srcSn,
- srcHeight, dstNetwork, dstContractAddress,
- data, sign);
- ;
- UserRevertedException e = assertThrows(UserRevertedException.class, action);
- assertEquals("Reverted(0): Signature already exists", e.getMessage());
- }
+ @Test
+ public void testSubmitPacket_duplicate() throws Exception {
+ String srcNetwork = "0x2.icon";
+ String dstNetwork = "sui";
+ BigInteger srcSn = BigInteger.ONE;
+ BigInteger srcHeight = BigInteger.ONE;
+ String srcContractAddress = "hxjuiod";
+ String dstContractAddress = "hxjuiod";
+ byte[] data = new byte[] { 0x01, 0x02 };
+
+ aggregator.invoke(adminAc, "setSignatureThreshold", 2);
+
+ byte[] dataHash = Context.hash("sha-256", data);
+ byte[] sign = relayerOne.sign(dataHash);
+
+ aggregator.invoke(relayerOneAc, "submitPacket", srcNetwork,
+ srcContractAddress, srcSn, srcHeight, dstNetwork,
+ dstContractAddress, data, sign);
+
+ Executable action = () -> aggregator.invoke(relayerOneAc, "submitPacket",
+ srcNetwork, srcContractAddress, srcSn,
+ srcHeight, dstNetwork, dstContractAddress,
+ data, sign);
+ ;
+ UserRevertedException e = assertThrows(UserRevertedException.class, action);
+ assertEquals("Reverted(0): Signature already exists", e.getMessage());
+ }
}
From c72a55083bdddda033e582ab36de338cc62d2529 Mon Sep 17 00:00:00 2001
From: gcranju <134275268+gcranju@users.noreply.github.com>
Date: Wed, 4 Dec 2024 14:59:14 +0545
Subject: [PATCH 17/17] fix: rlp issue resolved (#420)
---
contracts/soroban/libs/soroban-rlp/src/decoder.rs | 3 +++
contracts/soroban/libs/soroban-rlp/src/utils.rs | 2 +-
2 files changed, 4 insertions(+), 1 deletion(-)
diff --git a/contracts/soroban/libs/soroban-rlp/src/decoder.rs b/contracts/soroban/libs/soroban-rlp/src/decoder.rs
index 37b83405b..9d05760db 100644
--- a/contracts/soroban/libs/soroban-rlp/src/decoder.rs
+++ b/contracts/soroban/libs/soroban-rlp/src/decoder.rs
@@ -131,6 +131,9 @@ pub fn decode_u64(env: &Env, bytes: Bytes) -> u64 {
}
pub fn decode_u128(env: &Env, bytes: Bytes) -> u128 {
+ if bytes.len() == 1 {
+ return bytes_to_u128(bytes);
+ }
let decoded = decode(&env, bytes);
bytes_to_u128(decoded)
}
diff --git a/contracts/soroban/libs/soroban-rlp/src/utils.rs b/contracts/soroban/libs/soroban-rlp/src/utils.rs
index b359078ca..403cac527 100644
--- a/contracts/soroban/libs/soroban-rlp/src/utils.rs
+++ b/contracts/soroban/libs/soroban-rlp/src/utils.rs
@@ -53,7 +53,7 @@ pub fn bytes_to_u64(bytes: Bytes) -> u64 {
}
pub fn u128_to_bytes(env: &Env, number: u128) -> Bytes {
- let mut bytes: Bytes = Bytes::new(&env);
+ let mut bytes = bytes!(&env, 0x00);
let mut i = 15;
let mut leading_zero = true;
while i >= 0 {