Fix protocol compatibility and increase VAA test coverage
This commit is contained in:
parent
a1ba223361
commit
dece1a6a90
|
@ -63,6 +63,8 @@ type (
|
||||||
}
|
}
|
||||||
|
|
||||||
BodyTransfer struct {
|
BodyTransfer struct {
|
||||||
|
// Nonce is a user given unique nonce for this transfer
|
||||||
|
Nonce uint32
|
||||||
// SourceChain is the id of the chain the transfer was initiated from
|
// SourceChain is the id of the chain the transfer was initiated from
|
||||||
SourceChain ChainID
|
SourceChain ChainID
|
||||||
// TargetChain is the id of the chain the transfer is directed to
|
// TargetChain is the id of the chain the transfer is directed to
|
||||||
|
@ -106,7 +108,7 @@ func ParseVAA(data []byte) (*VAA, error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
v.Version = data[0]
|
v.Version = data[0]
|
||||||
if v.Version < supportedVAAVersion {
|
if v.Version != supportedVAAVersion {
|
||||||
return nil, fmt.Errorf("unsupported VAA version: %d", v.Version)
|
return nil, fmt.Errorf("unsupported VAA version: %d", v.Version)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -259,6 +261,10 @@ func (v *VAA) serializeBody() ([]byte, error) {
|
||||||
func parseBodyTransfer(r io.Reader) (*BodyTransfer, error) {
|
func parseBodyTransfer(r io.Reader) (*BodyTransfer, error) {
|
||||||
b := &BodyTransfer{}
|
b := &BodyTransfer{}
|
||||||
|
|
||||||
|
if err := binary.Read(r, binary.BigEndian, &b.Nonce); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to read nonce: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
if err := binary.Read(r, binary.BigEndian, &b.SourceChain); err != nil {
|
if err := binary.Read(r, binary.BigEndian, &b.SourceChain); err != nil {
|
||||||
return nil, fmt.Errorf("failed to read source chain: %w", err)
|
return nil, fmt.Errorf("failed to read source chain: %w", err)
|
||||||
}
|
}
|
||||||
|
@ -294,6 +300,7 @@ func (v *BodyTransfer) getActionID() Action {
|
||||||
|
|
||||||
func (v *BodyTransfer) serialize() ([]byte, error) {
|
func (v *BodyTransfer) serialize() ([]byte, error) {
|
||||||
buf := new(bytes.Buffer)
|
buf := new(bytes.Buffer)
|
||||||
|
MustWrite(buf, binary.BigEndian, v.Nonce)
|
||||||
MustWrite(buf, binary.BigEndian, v.SourceChain)
|
MustWrite(buf, binary.BigEndian, v.SourceChain)
|
||||||
MustWrite(buf, binary.BigEndian, v.TargetChain)
|
MustWrite(buf, binary.BigEndian, v.TargetChain)
|
||||||
buf.Write(v.TargetAddress[:])
|
buf.Write(v.TargetAddress[:])
|
||||||
|
|
|
@ -20,7 +20,7 @@ func TestSerializeDeserialize(t *testing.T) {
|
||||||
{
|
{
|
||||||
name: "BodyTransfer",
|
name: "BodyTransfer",
|
||||||
vaa: &VAA{
|
vaa: &VAA{
|
||||||
Version: 8,
|
Version: 1,
|
||||||
GuardianSetIndex: 9,
|
GuardianSetIndex: 9,
|
||||||
Signature: &Signature{
|
Signature: &Signature{
|
||||||
Sig: [32]byte{2, 8},
|
Sig: [32]byte{2, 8},
|
||||||
|
@ -28,6 +28,7 @@ func TestSerializeDeserialize(t *testing.T) {
|
||||||
},
|
},
|
||||||
Timestamp: time.Unix(2837, 0),
|
Timestamp: time.Unix(2837, 0),
|
||||||
Payload: &BodyTransfer{
|
Payload: &BodyTransfer{
|
||||||
|
Nonce: 38,
|
||||||
SourceChain: 2,
|
SourceChain: 2,
|
||||||
TargetChain: 1,
|
TargetChain: 1,
|
||||||
TargetAddress: Address{2, 1, 3},
|
TargetAddress: Address{2, 1, 3},
|
||||||
|
@ -42,7 +43,7 @@ func TestSerializeDeserialize(t *testing.T) {
|
||||||
{
|
{
|
||||||
name: "GuardianSetUpdate",
|
name: "GuardianSetUpdate",
|
||||||
vaa: &VAA{
|
vaa: &VAA{
|
||||||
Version: 8,
|
Version: 1,
|
||||||
GuardianSetIndex: 9,
|
GuardianSetIndex: 9,
|
||||||
Signature: &Signature{
|
Signature: &Signature{
|
||||||
Sig: [32]byte{2, 8},
|
Sig: [32]byte{2, 8},
|
||||||
|
|
|
@ -177,6 +177,7 @@ Size: `75 byte`
|
||||||
Payload:
|
Payload:
|
||||||
|
|
||||||
```
|
```
|
||||||
|
uint32 nonce
|
||||||
uint8 source_chain
|
uint8 source_chain
|
||||||
uint8 target_chain
|
uint8 target_chain
|
||||||
[32]uint8 target_address
|
[32]uint8 target_address
|
||||||
|
|
|
@ -34,10 +34,10 @@ dependencies = [
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "arrayref"
|
name = "arrayvec"
|
||||||
version = "0.3.6"
|
version = "0.5.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "a4c527152e37cf757a3f78aae5a06fbeefdb07ccc535c980a3208ee3060dd544"
|
checksum = "cff77d8686867eceff3105329d4698d96c2391c176d5d03adc90c7389162b5b8"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "assert_matches"
|
name = "assert_matches"
|
||||||
|
@ -99,6 +99,16 @@ version = "1.2.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "cf1de2fe8c75bc145a2f577add951f8134889b4795d47466a54a5c846d691693"
|
checksum = "cf1de2fe8c75bc145a2f577add951f8134889b4795d47466a54a5c846d691693"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "bitvec"
|
||||||
|
version = "0.17.4"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "41262f11d771fd4a61aa3ce019fca363b4b6c282fca9da2a31186d3965a47a5c"
|
||||||
|
dependencies = [
|
||||||
|
"either",
|
||||||
|
"radium",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "block-buffer"
|
name = "block-buffer"
|
||||||
version = "0.7.3"
|
version = "0.7.3"
|
||||||
|
@ -158,6 +168,12 @@ dependencies = [
|
||||||
"serde",
|
"serde",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "byte-slice-cast"
|
||||||
|
version = "0.3.5"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "b0a5e3906bcbf133e33c1d4d95afc664ad37fbdb9f6568d8043e7ea8c27d93d3"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "byte-tools"
|
name = "byte-tools"
|
||||||
version = "0.3.1"
|
version = "0.3.1"
|
||||||
|
@ -304,6 +320,12 @@ dependencies = [
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "crunchy"
|
||||||
|
version = "0.2.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "7a81dae078cea95a014a339291cec439d2f232ebe854a9d672b796c6afafa9b7"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "crypto-mac"
|
name = "crypto-mac"
|
||||||
version = "0.7.0"
|
version = "0.7.0"
|
||||||
|
@ -437,6 +459,18 @@ version = "0.1.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "835a3dc7d1ec9e75e2b5fb4ba75396837112d2060b03f7d43bc1897c7f7211da"
|
checksum = "835a3dc7d1ec9e75e2b5fb4ba75396837112d2060b03f7d43bc1897c7f7211da"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "fixed-hash"
|
||||||
|
version = "0.6.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "11498d382790b7a8f2fd211780bec78619bba81cdad3a283997c0c41f836759c"
|
||||||
|
dependencies = [
|
||||||
|
"byteorder",
|
||||||
|
"rand",
|
||||||
|
"rustc-hex",
|
||||||
|
"static_assertions",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "fnv"
|
name = "fnv"
|
||||||
version = "1.0.7"
|
version = "1.0.7"
|
||||||
|
@ -717,6 +751,15 @@ dependencies = [
|
||||||
"unicode-normalization",
|
"unicode-normalization",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "impl-codec"
|
||||||
|
version = "0.4.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "1be51a921b067b0eaca2fad532d9400041561aa922221cc65f95a85641c6bf53"
|
||||||
|
dependencies = [
|
||||||
|
"parity-scale-codec",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "indexmap"
|
name = "indexmap"
|
||||||
version = "1.5.0"
|
version = "1.5.0"
|
||||||
|
@ -1014,6 +1057,18 @@ version = "0.3.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "624a8340c38c1b80fd549087862da4ba43e08858af025b236e509b6649fc13d5"
|
checksum = "624a8340c38c1b80fd549087862da4ba43e08858af025b236e509b6649fc13d5"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "parity-scale-codec"
|
||||||
|
version = "1.3.4"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "34d38aeaffc032ec69faa476b3caaca8d4dd7f3f798137ff30359e5c7869ceb6"
|
||||||
|
dependencies = [
|
||||||
|
"arrayvec",
|
||||||
|
"bitvec",
|
||||||
|
"byte-slice-cast",
|
||||||
|
"serde",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "parking_lot"
|
name = "parking_lot"
|
||||||
version = "0.9.0"
|
version = "0.9.0"
|
||||||
|
@ -1094,6 +1149,17 @@ version = "0.2.8"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "237a5ed80e274dbc66f86bd59c1e25edc039660be53194b5fe0a482e0f2612ea"
|
checksum = "237a5ed80e274dbc66f86bd59c1e25edc039660be53194b5fe0a482e0f2612ea"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "primitive-types"
|
||||||
|
version = "0.7.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "c55c21c64d0eaa4d7ed885d959ef2d62d9e488c27c0e02d9aa5ce6c877b7d5f8"
|
||||||
|
dependencies = [
|
||||||
|
"fixed-hash",
|
||||||
|
"impl-codec",
|
||||||
|
"uint",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "proc-macro-hack"
|
name = "proc-macro-hack"
|
||||||
version = "0.5.18"
|
version = "0.5.18"
|
||||||
|
@ -1148,6 +1214,12 @@ dependencies = [
|
||||||
"proc-macro2 1.0.19",
|
"proc-macro2 1.0.19",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "radium"
|
||||||
|
version = "0.3.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "def50a86306165861203e7f84ecffbbdfdea79f0e51039b33de1e952358c47ac"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "rand"
|
name = "rand"
|
||||||
version = "0.7.3"
|
version = "0.7.3"
|
||||||
|
@ -1280,6 +1352,12 @@ version = "0.1.16"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "4c691c0e608126e00913e33f0ccf3727d5fc84573623b8d65b2df340b5201783"
|
checksum = "4c691c0e608126e00913e33f0ccf3727d5fc84573623b8d65b2df340b5201783"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "rustc-hex"
|
||||||
|
version = "2.1.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "3e75f6a532d0fd9f7f13144f392b6ad56a32696bfcd9c78f797f16bbb6f072d6"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "rustc_version"
|
name = "rustc_version"
|
||||||
version = "0.2.3"
|
version = "0.2.3"
|
||||||
|
@ -1565,6 +1643,12 @@ dependencies = [
|
||||||
"thiserror",
|
"thiserror",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "static_assertions"
|
||||||
|
version = "1.1.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "strsim"
|
name = "strsim"
|
||||||
version = "0.8.0"
|
version = "0.8.0"
|
||||||
|
@ -1967,6 +2051,18 @@ version = "1.12.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "373c8a200f9e67a0c95e62a4f52fbf80c23b4381c05a17845531982fa99e6b33"
|
checksum = "373c8a200f9e67a0c95e62a4f52fbf80c23b4381c05a17845531982fa99e6b33"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "uint"
|
||||||
|
version = "0.8.4"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "429ffcad8c8c15f874578c7337d156a3727eb4a1c2374c0ae937ad9a9b748c80"
|
||||||
|
dependencies = [
|
||||||
|
"byteorder",
|
||||||
|
"crunchy",
|
||||||
|
"rustc-hex",
|
||||||
|
"static_assertions",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "unicase"
|
name = "unicase"
|
||||||
version = "2.6.0"
|
version = "2.6.0"
|
||||||
|
@ -2216,11 +2312,11 @@ dependencies = [
|
||||||
name = "wormhole-bridge"
|
name = "wormhole-bridge"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"arrayref",
|
|
||||||
"byteorder",
|
"byteorder",
|
||||||
"hex",
|
"hex",
|
||||||
"num-derive 0.2.5",
|
"num-derive 0.2.5",
|
||||||
"num-traits",
|
"num-traits",
|
||||||
|
"primitive-types",
|
||||||
"rand",
|
"rand",
|
||||||
"remove_dir_all",
|
"remove_dir_all",
|
||||||
"sha3",
|
"sha3",
|
||||||
|
|
|
@ -22,10 +22,10 @@ remove_dir_all = "=0.5.0"
|
||||||
solana-sdk = { version = "=1.2.17", default-features = false, optional = true }
|
solana-sdk = { version = "=1.2.17", default-features = false, optional = true }
|
||||||
spl-token = { package = "spl-token", version = "1.0.6", default-features = false, optional = true }
|
spl-token = { package = "spl-token", version = "1.0.6", default-features = false, optional = true }
|
||||||
thiserror = "1.0"
|
thiserror = "1.0"
|
||||||
arrayref = "0.3.6"
|
|
||||||
byteorder = "1.3.4"
|
byteorder = "1.3.4"
|
||||||
zerocopy = "0.3.0"
|
zerocopy = "0.3.0"
|
||||||
sha3 = "0.9.1"
|
sha3 = "0.9.1"
|
||||||
|
primitive-types = "0.7.2"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
rand = { version = "0.7.0" }
|
rand = { version = "0.7.0" }
|
||||||
|
|
|
@ -6,7 +6,6 @@ use num_derive::FromPrimitive;
|
||||||
use num_traits::FromPrimitive;
|
use num_traits::FromPrimitive;
|
||||||
use solana_sdk::{
|
use solana_sdk::{
|
||||||
decode_error::DecodeError,
|
decode_error::DecodeError,
|
||||||
info,
|
|
||||||
program_error::{PrintProgramError, ProgramError},
|
program_error::{PrintProgramError, ProgramError},
|
||||||
};
|
};
|
||||||
use thiserror::Error;
|
use thiserror::Error;
|
||||||
|
@ -117,37 +116,3 @@ impl<T> DecodeError<T> for Error {
|
||||||
"Swap Error"
|
"Swap Error"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl PrintProgramError for Error {
|
|
||||||
fn print<E>(&self)
|
|
||||||
where
|
|
||||||
E: 'static + std::error::Error + DecodeError<E> + PrintProgramError + FromPrimitive,
|
|
||||||
{
|
|
||||||
match self {
|
|
||||||
Error::ExpectedToken => info!("Error: ExpectedToken"),
|
|
||||||
Error::ExpectedAccount => info!("Error: ExpectedAccount"),
|
|
||||||
Error::ExpectedBridge => info!("Error: ExpectedBridge"),
|
|
||||||
Error::ExpectedGuardianSet => info!("Error: ExpectedGuardianSet"),
|
|
||||||
Error::UninitializedState => info!("Error: State is unititialized"),
|
|
||||||
Error::InvalidProgramAddress => info!("Error: InvalidProgramAddress"),
|
|
||||||
Error::InvalidVAAFormat => info!("Error: InvalidVAAFormat"),
|
|
||||||
Error::InvalidVAAAction => info!("Error: InvalidVAAAction"),
|
|
||||||
Error::InvalidVAASignature => info!("Error: InvalidVAASignature"),
|
|
||||||
Error::AlreadyExists => info!("Error: AlreadyExists"),
|
|
||||||
Error::InvalidDerivedAccount => info!("Error: InvalidDerivedAccount"),
|
|
||||||
Error::TokenMintMismatch => info!("Error: TokenMintMismatch"),
|
|
||||||
Error::WrongMintOwner => info!("Error: WrongMintOwner"),
|
|
||||||
Error::WrongTokenAccountOwner => info!("Error: WrongTokenAccountOwner"),
|
|
||||||
Error::ParseFailed => info!("Error: ParseFailed"),
|
|
||||||
Error::GuardianSetExpired => info!("Error: GuardianSetExpired"),
|
|
||||||
Error::VAAExpired => info!("Error: VAAExpired"),
|
|
||||||
Error::VAAClaimed => info!("Error: VAAClaimed"),
|
|
||||||
Error::WrongBridgeOwner => info!("Error: WrongBridgeOwner"),
|
|
||||||
Error::OldGuardianSet => info!("Error: OldGuardianSet"),
|
|
||||||
Error::GuardianIndexNotIncreasing => info!("Error: GuardianIndexNotIncreasing"),
|
|
||||||
Error::ExpectedTransferOutProposal => info!("Error: ExpectedTransferOutProposal"),
|
|
||||||
Error::VAAProposalMismatch => info!("Error: VAAProposalMismatch"),
|
|
||||||
Error::SameChainTransfer => info!("Error: SameChainTransfer"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -0,0 +1,49 @@
|
||||||
|
#![cfg(feature = "program")]
|
||||||
|
|
||||||
|
use std::io;
|
||||||
|
|
||||||
|
use num_derive::FromPrimitive;
|
||||||
|
use num_traits::FromPrimitive;
|
||||||
|
use solana_sdk::{
|
||||||
|
decode_error::DecodeError,
|
||||||
|
info,
|
||||||
|
program_error::{PrintProgramError, ProgramError},
|
||||||
|
};
|
||||||
|
use thiserror::Error;
|
||||||
|
|
||||||
|
use crate::error;
|
||||||
|
use crate::error::*;
|
||||||
|
|
||||||
|
impl PrintProgramError for Error {
|
||||||
|
fn print<E>(&self)
|
||||||
|
where
|
||||||
|
E: 'static + std::error::Error + DecodeError<E> + PrintProgramError + FromPrimitive,
|
||||||
|
{
|
||||||
|
match self {
|
||||||
|
Error::ExpectedToken => info!("Error: ExpectedToken"),
|
||||||
|
Error::ExpectedAccount => info!("Error: ExpectedAccount"),
|
||||||
|
Error::ExpectedBridge => info!("Error: ExpectedBridge"),
|
||||||
|
Error::ExpectedGuardianSet => info!("Error: ExpectedGuardianSet"),
|
||||||
|
Error::UninitializedState => info!("Error: State is unititialized"),
|
||||||
|
Error::InvalidProgramAddress => info!("Error: InvalidProgramAddress"),
|
||||||
|
Error::InvalidVAAFormat => info!("Error: InvalidVAAFormat"),
|
||||||
|
Error::InvalidVAAAction => info!("Error: InvalidVAAAction"),
|
||||||
|
Error::InvalidVAASignature => info!("Error: InvalidVAASignature"),
|
||||||
|
Error::AlreadyExists => info!("Error: AlreadyExists"),
|
||||||
|
Error::InvalidDerivedAccount => info!("Error: InvalidDerivedAccount"),
|
||||||
|
Error::TokenMintMismatch => info!("Error: TokenMintMismatch"),
|
||||||
|
Error::WrongMintOwner => info!("Error: WrongMintOwner"),
|
||||||
|
Error::WrongTokenAccountOwner => info!("Error: WrongTokenAccountOwner"),
|
||||||
|
Error::ParseFailed => info!("Error: ParseFailed"),
|
||||||
|
Error::GuardianSetExpired => info!("Error: GuardianSetExpired"),
|
||||||
|
Error::VAAExpired => info!("Error: VAAExpired"),
|
||||||
|
Error::VAAClaimed => info!("Error: VAAClaimed"),
|
||||||
|
Error::WrongBridgeOwner => info!("Error: WrongBridgeOwner"),
|
||||||
|
Error::OldGuardianSet => info!("Error: OldGuardianSet"),
|
||||||
|
Error::GuardianIndexNotIncreasing => info!("Error: GuardianIndexNotIncreasing"),
|
||||||
|
Error::ExpectedTransferOutProposal => info!("Error: ExpectedTransferOutProposal"),
|
||||||
|
Error::VAAProposalMismatch => info!("Error: VAAProposalMismatch"),
|
||||||
|
Error::SameChainTransfer => info!("Error: SameChainTransfer"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,5 +1,4 @@
|
||||||
#![allow(clippy::too_many_arguments)]
|
#![allow(clippy::too_many_arguments)]
|
||||||
|
|
||||||
//! Instruction types
|
//! Instruction types
|
||||||
|
|
||||||
use std::io::Write;
|
use std::io::Write;
|
||||||
|
@ -132,7 +131,6 @@ impl BridgeInstruction {
|
||||||
Self::EvictExecutedVAA() => {
|
Self::EvictExecutedVAA() => {
|
||||||
output[0] = 4;
|
output[0] = 4;
|
||||||
}
|
}
|
||||||
_ => panic!(""),
|
|
||||||
}
|
}
|
||||||
Ok(output)
|
Ok(output)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,10 +1,9 @@
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
extern crate arrayref;
|
|
||||||
#[macro_use]
|
|
||||||
extern crate zerocopy;
|
extern crate zerocopy;
|
||||||
|
|
||||||
pub mod entrypoint;
|
pub mod entrypoint;
|
||||||
pub mod error;
|
pub mod error;
|
||||||
|
pub mod error_program;
|
||||||
pub mod instruction;
|
pub mod instruction;
|
||||||
pub mod processor;
|
pub mod processor;
|
||||||
pub mod state;
|
pub mod state;
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
//! Program instruction processing logic
|
//! Program instruction processing logic
|
||||||
|
#![cfg(feature = "program")]
|
||||||
|
|
||||||
use std::io::Write;
|
use std::io::Write;
|
||||||
use std::mem::size_of;
|
use std::mem::size_of;
|
||||||
|
@ -553,7 +554,7 @@ impl Bridge {
|
||||||
b.target_chain,
|
b.target_chain,
|
||||||
b.target_address,
|
b.target_address,
|
||||||
b.source_address,
|
b.source_address,
|
||||||
b.ref_block,
|
b.nonce,
|
||||||
)?;
|
)?;
|
||||||
if expected_proposal != *proposal_info.key {
|
if expected_proposal != *proposal_info.key {
|
||||||
return Err(Error::InvalidDerivedAccount.into());
|
return Err(Error::InvalidDerivedAccount.into());
|
||||||
|
@ -611,6 +612,158 @@ pub fn invoke_signed<'a>(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Implementation of actions
|
||||||
|
impl Bridge {
|
||||||
|
/// Burn a wrapped asset from account
|
||||||
|
pub fn wrapped_burn(
|
||||||
|
accounts: &[AccountInfo],
|
||||||
|
token_program_id: &Pubkey,
|
||||||
|
authority: &Pubkey,
|
||||||
|
token_account: &Pubkey,
|
||||||
|
amount: u64,
|
||||||
|
) -> Result<(), ProgramError> {
|
||||||
|
let all_signers: Vec<&Pubkey> = accounts
|
||||||
|
.iter()
|
||||||
|
.filter_map(|item| if item.is_signer { Some(item.key) } else { None })
|
||||||
|
.collect();
|
||||||
|
let ix = spl_token::instruction::burn(
|
||||||
|
token_program_id,
|
||||||
|
token_account,
|
||||||
|
authority,
|
||||||
|
all_signers.as_slice(),
|
||||||
|
amount,
|
||||||
|
)?;
|
||||||
|
invoke_signed(&ix, accounts, &[])
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Mint a wrapped asset to account
|
||||||
|
pub fn wrapped_mint_to(
|
||||||
|
accounts: &[AccountInfo],
|
||||||
|
token_program_id: &Pubkey,
|
||||||
|
mint: &Pubkey,
|
||||||
|
destination: &Pubkey,
|
||||||
|
bridge: &Pubkey,
|
||||||
|
amount: u64,
|
||||||
|
) -> Result<(), ProgramError> {
|
||||||
|
let ix = spl_token::instruction::mint_to(
|
||||||
|
token_program_id,
|
||||||
|
mint,
|
||||||
|
destination,
|
||||||
|
bridge,
|
||||||
|
&[],
|
||||||
|
amount,
|
||||||
|
)?;
|
||||||
|
invoke_signed(&ix, accounts, &[&[&bridge.to_bytes()[..32]][..]])
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Transfer tokens from a caller
|
||||||
|
pub fn token_transfer_caller(
|
||||||
|
accounts: &[AccountInfo],
|
||||||
|
token_program_id: &Pubkey,
|
||||||
|
source: &Pubkey,
|
||||||
|
destination: &Pubkey,
|
||||||
|
authority: &Pubkey,
|
||||||
|
amount: u64,
|
||||||
|
) -> Result<(), ProgramError> {
|
||||||
|
let all_signers: Vec<&Pubkey> = accounts
|
||||||
|
.iter()
|
||||||
|
.filter_map(|item| if item.is_signer { Some(item.key) } else { None })
|
||||||
|
.collect();
|
||||||
|
let ix = spl_token::instruction::transfer(
|
||||||
|
token_program_id,
|
||||||
|
source,
|
||||||
|
destination,
|
||||||
|
authority,
|
||||||
|
all_signers.as_slice(),
|
||||||
|
amount,
|
||||||
|
)?;
|
||||||
|
invoke_signed(&ix, accounts, &[])
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Transfer tokens from a custody account
|
||||||
|
pub fn token_transfer_custody(
|
||||||
|
accounts: &[AccountInfo],
|
||||||
|
token_program_id: &Pubkey,
|
||||||
|
bridge: &Pubkey,
|
||||||
|
source: &Pubkey,
|
||||||
|
destination: &Pubkey,
|
||||||
|
amount: u64,
|
||||||
|
) -> Result<(), ProgramError> {
|
||||||
|
let ix = spl_token::instruction::transfer(
|
||||||
|
token_program_id,
|
||||||
|
source,
|
||||||
|
destination,
|
||||||
|
bridge,
|
||||||
|
&[],
|
||||||
|
amount,
|
||||||
|
)?;
|
||||||
|
invoke_signed(&ix, accounts, &[&[&bridge.to_bytes()[..32]][..]])
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create a new account
|
||||||
|
pub fn create_custody_account(
|
||||||
|
program_id: &Pubkey,
|
||||||
|
accounts: &[AccountInfo],
|
||||||
|
token_program: &Pubkey,
|
||||||
|
bridge: &Pubkey,
|
||||||
|
account: &Pubkey,
|
||||||
|
mint: &Pubkey,
|
||||||
|
payer: &Pubkey,
|
||||||
|
) -> Result<(), ProgramError> {
|
||||||
|
Self::create_account::<Mint>(
|
||||||
|
program_id,
|
||||||
|
accounts,
|
||||||
|
mint,
|
||||||
|
payer,
|
||||||
|
Self::derive_custody_seeds(bridge, mint),
|
||||||
|
)?;
|
||||||
|
let ix = spl_token::instruction::initialize_account(token_program, account, mint, bridge)?;
|
||||||
|
invoke_signed(&ix, accounts, &[&[&bridge.to_bytes()[..32]][..]])
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create a mint for a wrapped asset
|
||||||
|
pub fn create_wrapped_mint(
|
||||||
|
program_id: &Pubkey,
|
||||||
|
accounts: &[AccountInfo],
|
||||||
|
token_program: &Pubkey,
|
||||||
|
mint: &Pubkey,
|
||||||
|
bridge: &Pubkey,
|
||||||
|
payer: &Pubkey,
|
||||||
|
asset: &AssetMeta,
|
||||||
|
) -> Result<(), ProgramError> {
|
||||||
|
Self::create_account::<Mint>(
|
||||||
|
program_id,
|
||||||
|
accounts,
|
||||||
|
mint,
|
||||||
|
payer,
|
||||||
|
Self::derive_wrapped_asset_seeds(bridge, asset.chain, asset.address),
|
||||||
|
)?;
|
||||||
|
let ix =
|
||||||
|
spl_token::instruction::initialize_mint(token_program, mint, None, Some(bridge), 0, 8)?;
|
||||||
|
invoke_signed(&ix, accounts, &[&[&bridge.to_bytes()[..32]][..]])
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create a new account
|
||||||
|
pub fn create_account<T: Sized>(
|
||||||
|
program_id: &Pubkey,
|
||||||
|
accounts: &[AccountInfo],
|
||||||
|
new_account: &Pubkey,
|
||||||
|
payer: &Pubkey,
|
||||||
|
seeds: Vec<Vec<u8>>,
|
||||||
|
) -> Result<(), ProgramError> {
|
||||||
|
let size = size_of::<T>();
|
||||||
|
let ix = create_account(
|
||||||
|
payer,
|
||||||
|
new_account,
|
||||||
|
Rent::default().minimum_balance(size as usize),
|
||||||
|
size as u64,
|
||||||
|
program_id,
|
||||||
|
);
|
||||||
|
let s: Vec<_> = seeds.iter().map(|item| item.as_slice()).collect();
|
||||||
|
invoke_signed(&ix, accounts, &[s.as_slice()])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use solana_sdk::{
|
use solana_sdk::{
|
||||||
|
|
|
@ -11,14 +11,13 @@ use solana_sdk::clock::Clock;
|
||||||
use solana_sdk::hash::hash;
|
use solana_sdk::hash::hash;
|
||||||
#[cfg(not(target_arch = "bpf"))]
|
#[cfg(not(target_arch = "bpf"))]
|
||||||
use solana_sdk::instruction::Instruction;
|
use solana_sdk::instruction::Instruction;
|
||||||
use solana_sdk::log::sol_log;
|
|
||||||
#[cfg(target_arch = "bpf")]
|
#[cfg(target_arch = "bpf")]
|
||||||
use solana_sdk::program::invoke_signed;
|
use solana_sdk::program::invoke_signed;
|
||||||
use solana_sdk::rent::Rent;
|
use solana_sdk::rent::Rent;
|
||||||
use solana_sdk::system_instruction::{create_account, SystemInstruction};
|
use solana_sdk::system_instruction::{create_account, SystemInstruction};
|
||||||
use solana_sdk::sysvar::Sysvar;
|
use solana_sdk::sysvar::Sysvar;
|
||||||
use solana_sdk::{
|
use solana_sdk::{
|
||||||
account_info::next_account_info, account_info::AccountInfo, entrypoint::ProgramResult, info,
|
account_info::next_account_info, account_info::AccountInfo, entrypoint::ProgramResult,
|
||||||
program_error::ProgramError, pubkey::bs58, pubkey::Pubkey,
|
program_error::ProgramError, pubkey::bs58, pubkey::Pubkey,
|
||||||
};
|
};
|
||||||
use spl_token::state::Mint;
|
use spl_token::state::Mint;
|
||||||
|
@ -27,12 +26,11 @@ use crate::instruction::BridgeInstruction::*;
|
||||||
use crate::instruction::{
|
use crate::instruction::{
|
||||||
BridgeInstruction, ForeignAddress, GuardianKey, TransferOutPayload, CHAIN_ID_SOLANA, VAA_BODY,
|
BridgeInstruction, ForeignAddress, GuardianKey, TransferOutPayload, CHAIN_ID_SOLANA, VAA_BODY,
|
||||||
};
|
};
|
||||||
#[cfg(not(target_arch = "bpf"))]
|
|
||||||
use crate::processor::invoke_signed;
|
|
||||||
|
|
||||||
use crate::syscalls::{sol_verify_schnorr, RawKey, SchnorrifyInput};
|
use crate::syscalls::{sol_verify_schnorr, RawKey, SchnorrifyInput};
|
||||||
use crate::vaa::{BodyTransfer, BodyUpdateGuardianSet, VAABody, VAA};
|
use crate::vaa::{BodyTransfer, BodyUpdateGuardianSet, VAABody, VAA};
|
||||||
use crate::{error::Error, instruction::unpack};
|
use crate::{error::Error, instruction::unpack};
|
||||||
|
use primitive_types::U256;
|
||||||
use zerocopy::AsBytes;
|
use zerocopy::AsBytes;
|
||||||
|
|
||||||
/// fee rate as a ratio
|
/// fee rate as a ratio
|
||||||
|
@ -73,7 +71,7 @@ impl IsInitialized for GuardianSet {
|
||||||
#[derive(Clone, Copy)]
|
#[derive(Clone, Copy)]
|
||||||
pub struct TransferOutProposal {
|
pub struct TransferOutProposal {
|
||||||
/// amount to transfer
|
/// amount to transfer
|
||||||
pub amount: u64,
|
pub amount: U256,
|
||||||
/// chain id to transfer to
|
/// chain id to transfer to
|
||||||
pub to_chain_id: u8,
|
pub to_chain_id: u8,
|
||||||
/// address on the foreign chain to transfer to
|
/// address on the foreign chain to transfer to
|
||||||
|
@ -219,158 +217,6 @@ impl Bridge {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Implementation of actions
|
|
||||||
impl Bridge {
|
|
||||||
/// Burn a wrapped asset from account
|
|
||||||
pub fn wrapped_burn(
|
|
||||||
accounts: &[AccountInfo],
|
|
||||||
token_program_id: &Pubkey,
|
|
||||||
authority: &Pubkey,
|
|
||||||
token_account: &Pubkey,
|
|
||||||
amount: u64,
|
|
||||||
) -> Result<(), ProgramError> {
|
|
||||||
let all_signers: Vec<&Pubkey> = accounts
|
|
||||||
.iter()
|
|
||||||
.filter_map(|item| if item.is_signer { Some(item.key) } else { None })
|
|
||||||
.collect();
|
|
||||||
let ix = spl_token::instruction::burn(
|
|
||||||
token_program_id,
|
|
||||||
token_account,
|
|
||||||
authority,
|
|
||||||
all_signers.as_slice(),
|
|
||||||
amount,
|
|
||||||
)?;
|
|
||||||
invoke_signed(&ix, accounts, &[])
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Mint a wrapped asset to account
|
|
||||||
pub fn wrapped_mint_to(
|
|
||||||
accounts: &[AccountInfo],
|
|
||||||
token_program_id: &Pubkey,
|
|
||||||
mint: &Pubkey,
|
|
||||||
destination: &Pubkey,
|
|
||||||
bridge: &Pubkey,
|
|
||||||
amount: u64,
|
|
||||||
) -> Result<(), ProgramError> {
|
|
||||||
let ix = spl_token::instruction::mint_to(
|
|
||||||
token_program_id,
|
|
||||||
mint,
|
|
||||||
destination,
|
|
||||||
bridge,
|
|
||||||
&[],
|
|
||||||
amount,
|
|
||||||
)?;
|
|
||||||
invoke_signed(&ix, accounts, &[&[&bridge.to_bytes()[..32]][..]])
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Transfer tokens from a caller
|
|
||||||
pub fn token_transfer_caller(
|
|
||||||
accounts: &[AccountInfo],
|
|
||||||
token_program_id: &Pubkey,
|
|
||||||
source: &Pubkey,
|
|
||||||
destination: &Pubkey,
|
|
||||||
authority: &Pubkey,
|
|
||||||
amount: u64,
|
|
||||||
) -> Result<(), ProgramError> {
|
|
||||||
let all_signers: Vec<&Pubkey> = accounts
|
|
||||||
.iter()
|
|
||||||
.filter_map(|item| if item.is_signer { Some(item.key) } else { None })
|
|
||||||
.collect();
|
|
||||||
let ix = spl_token::instruction::transfer(
|
|
||||||
token_program_id,
|
|
||||||
source,
|
|
||||||
destination,
|
|
||||||
authority,
|
|
||||||
all_signers.as_slice(),
|
|
||||||
amount,
|
|
||||||
)?;
|
|
||||||
invoke_signed(&ix, accounts, &[])
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Transfer tokens from a custody account
|
|
||||||
pub fn token_transfer_custody(
|
|
||||||
accounts: &[AccountInfo],
|
|
||||||
token_program_id: &Pubkey,
|
|
||||||
bridge: &Pubkey,
|
|
||||||
source: &Pubkey,
|
|
||||||
destination: &Pubkey,
|
|
||||||
amount: u64,
|
|
||||||
) -> Result<(), ProgramError> {
|
|
||||||
let ix = spl_token::instruction::transfer(
|
|
||||||
token_program_id,
|
|
||||||
source,
|
|
||||||
destination,
|
|
||||||
bridge,
|
|
||||||
&[],
|
|
||||||
amount,
|
|
||||||
)?;
|
|
||||||
invoke_signed(&ix, accounts, &[&[&bridge.to_bytes()[..32]][..]])
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Create a new account
|
|
||||||
pub fn create_custody_account(
|
|
||||||
program_id: &Pubkey,
|
|
||||||
accounts: &[AccountInfo],
|
|
||||||
token_program: &Pubkey,
|
|
||||||
bridge: &Pubkey,
|
|
||||||
account: &Pubkey,
|
|
||||||
mint: &Pubkey,
|
|
||||||
payer: &Pubkey,
|
|
||||||
) -> Result<(), ProgramError> {
|
|
||||||
Self::create_account::<Mint>(
|
|
||||||
program_id,
|
|
||||||
accounts,
|
|
||||||
mint,
|
|
||||||
payer,
|
|
||||||
Self::derive_custody_seeds(bridge, mint),
|
|
||||||
)?;
|
|
||||||
let ix = spl_token::instruction::initialize_account(token_program, account, mint, bridge)?;
|
|
||||||
invoke_signed(&ix, accounts, &[&[&bridge.to_bytes()[..32]][..]])
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Create a mint for a wrapped asset
|
|
||||||
pub fn create_wrapped_mint(
|
|
||||||
program_id: &Pubkey,
|
|
||||||
accounts: &[AccountInfo],
|
|
||||||
token_program: &Pubkey,
|
|
||||||
mint: &Pubkey,
|
|
||||||
bridge: &Pubkey,
|
|
||||||
payer: &Pubkey,
|
|
||||||
asset: &AssetMeta,
|
|
||||||
) -> Result<(), ProgramError> {
|
|
||||||
Self::create_account::<Mint>(
|
|
||||||
program_id,
|
|
||||||
accounts,
|
|
||||||
mint,
|
|
||||||
payer,
|
|
||||||
Self::derive_wrapped_asset_seeds(bridge, asset.chain, asset.address),
|
|
||||||
)?;
|
|
||||||
let ix =
|
|
||||||
spl_token::instruction::initialize_mint(token_program, mint, None, Some(bridge), 0, 8)?;
|
|
||||||
invoke_signed(&ix, accounts, &[&[&bridge.to_bytes()[..32]][..]])
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Create a new account
|
|
||||||
pub fn create_account<T: Sized>(
|
|
||||||
program_id: &Pubkey,
|
|
||||||
accounts: &[AccountInfo],
|
|
||||||
new_account: &Pubkey,
|
|
||||||
payer: &Pubkey,
|
|
||||||
seeds: Vec<Vec<u8>>,
|
|
||||||
) -> Result<(), ProgramError> {
|
|
||||||
let size = size_of::<T>();
|
|
||||||
let ix = create_account(
|
|
||||||
payer,
|
|
||||||
new_account,
|
|
||||||
Rent::default().minimum_balance(size as usize),
|
|
||||||
size as u64,
|
|
||||||
program_id,
|
|
||||||
);
|
|
||||||
let s: Vec<_> = seeds.iter().map(|item| item.as_slice()).collect();
|
|
||||||
invoke_signed(&ix, accounts, &[s.as_slice()])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Implementation of derivations
|
/// Implementation of derivations
|
||||||
impl Bridge {
|
impl Bridge {
|
||||||
/// Calculates derived seeds for a guardian set
|
/// Calculates derived seeds for a guardian set
|
||||||
|
@ -404,7 +250,7 @@ impl Bridge {
|
||||||
target_chain: u8,
|
target_chain: u8,
|
||||||
target_address: ForeignAddress,
|
target_address: ForeignAddress,
|
||||||
user: ForeignAddress,
|
user: ForeignAddress,
|
||||||
slot: u64,
|
slot: u32,
|
||||||
) -> Vec<Vec<u8>> {
|
) -> Vec<Vec<u8>> {
|
||||||
vec![
|
vec![
|
||||||
"transfer".as_bytes().to_vec(),
|
"transfer".as_bytes().to_vec(),
|
||||||
|
@ -498,7 +344,7 @@ impl Bridge {
|
||||||
target_chain: u8,
|
target_chain: u8,
|
||||||
target_address: ForeignAddress,
|
target_address: ForeignAddress,
|
||||||
user: ForeignAddress,
|
user: ForeignAddress,
|
||||||
slot: u64,
|
slot: u32,
|
||||||
) -> Result<Pubkey, Error> {
|
) -> Result<Pubkey, Error> {
|
||||||
Self::derive_key(
|
Self::derive_key(
|
||||||
program_id,
|
program_id,
|
||||||
|
|
|
@ -10,11 +10,16 @@ pub struct SchnorrifyInput {
|
||||||
#[derive(Copy, Clone, Debug, PartialOrd, PartialEq)]
|
#[derive(Copy, Clone, Debug, PartialOrd, PartialEq)]
|
||||||
pub struct RawKey {
|
pub struct RawKey {
|
||||||
pub x: [u8; 32],
|
pub x: [u8; 32],
|
||||||
pub y: [u8; 32],
|
pub y_parity: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SchnorrifyInput {
|
impl SchnorrifyInput {
|
||||||
pub fn new(pub_key: RawKey, message: [u8; 32], signature: [u8; 32], addr: [u8; 20]) -> SchnorrifyInput {
|
pub fn new(
|
||||||
|
pub_key: RawKey,
|
||||||
|
message: [u8; 32],
|
||||||
|
signature: [u8; 32],
|
||||||
|
addr: [u8; 20],
|
||||||
|
) -> SchnorrifyInput {
|
||||||
SchnorrifyInput {
|
SchnorrifyInput {
|
||||||
message,
|
message,
|
||||||
addr,
|
addr,
|
||||||
|
@ -29,9 +34,7 @@ impl SchnorrifyInput {
|
||||||
/// @param input - Input for signature verification
|
/// @param input - Input for signature verification
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn sol_verify_schnorr(input: &SchnorrifyInput) -> bool {
|
pub fn sol_verify_schnorr(input: &SchnorrifyInput) -> bool {
|
||||||
let res = unsafe {
|
let res = unsafe { sol_verify_ethschnorr(input as *const _ as *const u8) };
|
||||||
sol_verify_ethschnorr(input as *const _ as *const u8)
|
|
||||||
};
|
|
||||||
|
|
||||||
res == 1
|
res == 1
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,14 +3,14 @@ use std::io::{Cursor, Read, Write};
|
||||||
use std::mem::size_of;
|
use std::mem::size_of;
|
||||||
|
|
||||||
use byteorder::{BigEndian, ReadBytesExt, WriteBytesExt};
|
use byteorder::{BigEndian, ReadBytesExt, WriteBytesExt};
|
||||||
|
use primitive_types::U256;
|
||||||
use sha3::Digest;
|
use sha3::Digest;
|
||||||
use solana_sdk::program_error::ProgramError;
|
use solana_sdk::program_error::ProgramError;
|
||||||
|
|
||||||
use crate::error::Error;
|
use crate::error::Error;
|
||||||
use crate::error::Error::InvalidVAAFormat;
|
use crate::error::Error::InvalidVAAFormat;
|
||||||
use crate::instruction::unpack;
|
|
||||||
use crate::state::AssetMeta;
|
use crate::state::AssetMeta;
|
||||||
use crate::syscalls::{RawKey, SchnorrifyInput, sol_verify_schnorr};
|
use crate::syscalls::{sol_verify_schnorr, RawKey, SchnorrifyInput};
|
||||||
use crate::vaa::VAABody::UpdateGuardianSet;
|
use crate::vaa::VAABody::UpdateGuardianSet;
|
||||||
|
|
||||||
pub type ForeignAddress = [u8; 32];
|
pub type ForeignAddress = [u8; 32];
|
||||||
|
@ -42,16 +42,20 @@ impl VAA {
|
||||||
|
|
||||||
pub fn verify(&self, guardian_key: &RawKey) -> bool {
|
pub fn verify(&self, guardian_key: &RawKey) -> bool {
|
||||||
let body = match self.signature_body() {
|
let body = match self.signature_body() {
|
||||||
Ok(v) => { v }
|
Ok(v) => v,
|
||||||
Err(_) => { return false; }
|
Err(_) => {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let mut h = sha3::Keccak256::default();
|
let mut h = sha3::Keccak256::default();
|
||||||
if let Err(_) = h.write(body.as_slice()) { return false; };
|
if let Err(_) = h.write(body.as_slice()) {
|
||||||
|
return false;
|
||||||
|
};
|
||||||
let hash = h.finalize().into();
|
let hash = h.finalize().into();
|
||||||
|
|
||||||
let schnorr_input = SchnorrifyInput::new(*guardian_key, hash,
|
let schnorr_input =
|
||||||
self.signature_sig, self.signature_addr);
|
SchnorrifyInput::new(*guardian_key, hash, self.signature_sig, self.signature_addr);
|
||||||
sol_verify_schnorr(&schnorr_input)
|
sol_verify_schnorr(&schnorr_input)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -68,6 +72,7 @@ impl VAA {
|
||||||
v.write_u8(payload.action_id())?;
|
v.write_u8(payload.action_id())?;
|
||||||
|
|
||||||
let payload_data = payload.serialize()?;
|
let payload_data = payload.serialize()?;
|
||||||
|
v.write_u8(payload_data.len() as u8)?;
|
||||||
v.write(payload_data.as_slice())?;
|
v.write(payload_data.as_slice())?;
|
||||||
|
|
||||||
Ok(v.into_inner())
|
Ok(v.into_inner())
|
||||||
|
@ -124,14 +129,13 @@ impl VAABody {
|
||||||
fn deserialize(data: &Vec<u8>) -> Result<VAABody, Error> {
|
fn deserialize(data: &Vec<u8>) -> Result<VAABody, Error> {
|
||||||
let mut payload_data = Cursor::new(data);
|
let mut payload_data = Cursor::new(data);
|
||||||
let action = payload_data.read_u8()?;
|
let action = payload_data.read_u8()?;
|
||||||
|
let length = payload_data.read_u8()?;
|
||||||
|
|
||||||
let payload = match action {
|
let payload = match action {
|
||||||
0x01 => {
|
0x01 => {
|
||||||
VAABody::UpdateGuardianSet(BodyUpdateGuardianSet::deserialize(&mut payload_data)?)
|
VAABody::UpdateGuardianSet(BodyUpdateGuardianSet::deserialize(&mut payload_data)?)
|
||||||
}
|
}
|
||||||
0x10 => {
|
0x10 => VAABody::Transfer(BodyTransfer::deserialize(&mut payload_data)?),
|
||||||
VAABody::Transfer(BodyTransfer::deserialize(&mut payload_data)?)
|
|
||||||
}
|
|
||||||
_ => {
|
_ => {
|
||||||
return Err(Error::InvalidVAAAction);
|
return Err(Error::InvalidVAAAction);
|
||||||
}
|
}
|
||||||
|
@ -142,12 +146,8 @@ impl VAABody {
|
||||||
|
|
||||||
fn serialize(&self) -> Result<Vec<u8>, Error> {
|
fn serialize(&self) -> Result<Vec<u8>, Error> {
|
||||||
match self {
|
match self {
|
||||||
VAABody::Transfer(b) => {
|
VAABody::Transfer(b) => b.serialize(),
|
||||||
b.serialize()
|
VAABody::UpdateGuardianSet(b) => b.serialize(),
|
||||||
}
|
|
||||||
VAABody::UpdateGuardianSet(b) => {
|
|
||||||
b.serialize()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -160,37 +160,45 @@ pub struct BodyUpdateGuardianSet {
|
||||||
|
|
||||||
#[derive(Copy, Clone, Debug, PartialEq)]
|
#[derive(Copy, Clone, Debug, PartialEq)]
|
||||||
pub struct BodyTransfer {
|
pub struct BodyTransfer {
|
||||||
pub ref_block: u64,
|
pub nonce: u32,
|
||||||
pub source_chain: u8,
|
pub source_chain: u8,
|
||||||
pub source_address: ForeignAddress,
|
|
||||||
pub target_chain: u8,
|
pub target_chain: u8,
|
||||||
pub target_address: ForeignAddress,
|
pub target_address: ForeignAddress,
|
||||||
pub asset: AssetMeta,
|
pub asset: AssetMeta,
|
||||||
pub amount: u64,
|
pub amount: U256,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl BodyUpdateGuardianSet {
|
impl BodyUpdateGuardianSet {
|
||||||
fn deserialize(data: &mut Cursor<&Vec<u8>>) -> Result<BodyUpdateGuardianSet, Error> {
|
fn deserialize(data: &mut Cursor<&Vec<u8>>) -> Result<BodyUpdateGuardianSet, Error> {
|
||||||
let new_index = data.read_u32::<BigEndian>()?;
|
|
||||||
let mut new_key_x: [u8; 32] = [0; 32];
|
let mut new_key_x: [u8; 32] = [0; 32];
|
||||||
data.read(&mut new_key_x)?;
|
data.read(&mut new_key_x)?;
|
||||||
let mut new_key_y: [u8; 32] = [0; 32];
|
let new_key_y_parity = match data.read_u8()? {
|
||||||
data.read(&mut new_key_y)?;
|
0 => false,
|
||||||
|
1 => true,
|
||||||
|
_ => return Err(InvalidVAAFormat),
|
||||||
|
};
|
||||||
|
|
||||||
|
let new_index = data.read_u32::<BigEndian>()?;
|
||||||
|
|
||||||
Ok(BodyUpdateGuardianSet {
|
Ok(BodyUpdateGuardianSet {
|
||||||
new_index,
|
new_index,
|
||||||
new_key: RawKey {
|
new_key: RawKey {
|
||||||
x: new_key_x,
|
x: new_key_x,
|
||||||
y: new_key_y,
|
y_parity: new_key_y_parity,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn serialize(&self) -> Result<Vec<u8>, Error> {
|
fn serialize(&self) -> Result<Vec<u8>, Error> {
|
||||||
let mut v: Cursor<Vec<u8>> = Cursor::new(Vec::new());
|
let mut v: Cursor<Vec<u8>> = Cursor::new(Vec::new());
|
||||||
v.write_u32::<BigEndian>(self.new_index)?;
|
|
||||||
v.write(&self.new_key.x)?;
|
v.write(&self.new_key.x)?;
|
||||||
v.write(&self.new_key.y)?;
|
v.write_u8({
|
||||||
|
match self.new_key.y_parity {
|
||||||
|
false => 0,
|
||||||
|
true => 1,
|
||||||
|
}
|
||||||
|
})?;
|
||||||
|
v.write_u32::<BigEndian>(self.new_index)?;
|
||||||
|
|
||||||
Ok(v.into_inner())
|
Ok(v.into_inner())
|
||||||
}
|
}
|
||||||
|
@ -198,26 +206,26 @@ impl BodyUpdateGuardianSet {
|
||||||
|
|
||||||
impl BodyTransfer {
|
impl BodyTransfer {
|
||||||
fn deserialize(data: &mut Cursor<&Vec<u8>>) -> Result<BodyTransfer, Error> {
|
fn deserialize(data: &mut Cursor<&Vec<u8>>) -> Result<BodyTransfer, Error> {
|
||||||
let ref_block = data.read_u64::<BigEndian>()?;
|
let nonce = data.read_u32::<BigEndian>()?;
|
||||||
let source_chain = data.read_u8()?;
|
let source_chain = data.read_u8()?;
|
||||||
let mut source_address: ForeignAddress = ForeignAddress::default();
|
|
||||||
data.read(&mut source_address)?;
|
|
||||||
let target_chain = data.read_u8()?;
|
let target_chain = data.read_u8()?;
|
||||||
let mut target_address: ForeignAddress = ForeignAddress::default();
|
let mut target_address: ForeignAddress = ForeignAddress::default();
|
||||||
data.read(&mut target_address)?;
|
data.read_exact(&mut target_address)?;
|
||||||
let token_chain = data.read_u8()?;
|
let token_chain = data.read_u8()?;
|
||||||
let mut token_address: ForeignAddress = ForeignAddress::default();
|
let mut token_address: ForeignAddress = ForeignAddress::default();
|
||||||
data.read(&mut token_address)?;
|
data.read_exact(&mut token_address)?;
|
||||||
let amount = data.read_u64::<BigEndian>()?;
|
|
||||||
|
let mut am_data: [u8; 32] = [0; 32];
|
||||||
|
data.read_exact(&mut am_data)?;
|
||||||
|
let amount = U256::from_big_endian(&am_data);
|
||||||
|
|
||||||
Ok(BodyTransfer {
|
Ok(BodyTransfer {
|
||||||
ref_block,
|
nonce,
|
||||||
source_chain,
|
source_chain,
|
||||||
source_address,
|
|
||||||
target_chain,
|
target_chain,
|
||||||
target_address,
|
target_address,
|
||||||
asset: AssetMeta {
|
asset: AssetMeta {
|
||||||
address: target_address,
|
address: token_address,
|
||||||
chain: token_chain,
|
chain: token_chain,
|
||||||
},
|
},
|
||||||
amount,
|
amount,
|
||||||
|
@ -226,12 +234,16 @@ impl BodyTransfer {
|
||||||
|
|
||||||
fn serialize(&self) -> Result<Vec<u8>, Error> {
|
fn serialize(&self) -> Result<Vec<u8>, Error> {
|
||||||
let mut v: Cursor<Vec<u8>> = Cursor::new(Vec::new());
|
let mut v: Cursor<Vec<u8>> = Cursor::new(Vec::new());
|
||||||
|
v.write_u32::<BigEndian>(self.nonce)?;
|
||||||
v.write_u8(self.source_chain)?;
|
v.write_u8(self.source_chain)?;
|
||||||
v.write_u8(self.target_chain)?;
|
v.write_u8(self.target_chain)?;
|
||||||
v.write(&self.target_address)?;
|
v.write(&self.target_address)?;
|
||||||
v.write_u8(self.asset.chain)?;
|
v.write_u8(self.asset.chain)?;
|
||||||
v.write(&self.asset.address)?;
|
v.write(&self.asset.address)?;
|
||||||
v.write_u64::<BigEndian>(self.amount)?;
|
|
||||||
|
let mut am_data: [u8; 32] = [0; 32];
|
||||||
|
self.amount.to_big_endian(&mut am_data);
|
||||||
|
v.write(&am_data[..])?;
|
||||||
|
|
||||||
Ok(v.into_inner())
|
Ok(v.into_inner())
|
||||||
}
|
}
|
||||||
|
@ -242,11 +254,12 @@ mod tests {
|
||||||
use std::io::Write;
|
use std::io::Write;
|
||||||
|
|
||||||
use hex;
|
use hex;
|
||||||
|
use primitive_types::U256;
|
||||||
|
|
||||||
use crate::error::Error;
|
use crate::error::Error;
|
||||||
use crate::state::AssetMeta;
|
use crate::state::AssetMeta;
|
||||||
use crate::syscalls::RawKey;
|
use crate::syscalls::RawKey;
|
||||||
use crate::vaa::{BodyTransfer, BodyUpdateGuardianSet, VAA, VAABody};
|
use crate::vaa::{BodyTransfer, BodyUpdateGuardianSet, VAABody, VAA};
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn serialize_deserialize_vaa_transfer() {
|
fn serialize_deserialize_vaa_transfer() {
|
||||||
|
@ -257,16 +270,15 @@ mod tests {
|
||||||
signature_addr: [9; 20],
|
signature_addr: [9; 20],
|
||||||
timestamp: 83,
|
timestamp: 83,
|
||||||
payload: Some(VAABody::Transfer(BodyTransfer {
|
payload: Some(VAABody::Transfer(BodyTransfer {
|
||||||
ref_block: 28,
|
nonce: 28,
|
||||||
source_chain: 1,
|
source_chain: 1,
|
||||||
source_address: [9; 32],
|
|
||||||
target_chain: 2,
|
target_chain: 2,
|
||||||
target_address: [1; 32],
|
target_address: [1; 32],
|
||||||
asset: AssetMeta {
|
asset: AssetMeta {
|
||||||
address: [2; 32],
|
address: [2; 32],
|
||||||
chain: 8,
|
chain: 8,
|
||||||
},
|
},
|
||||||
amount: 4,
|
amount: U256::from(3),
|
||||||
})),
|
})),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -287,7 +299,7 @@ mod tests {
|
||||||
new_index: 29,
|
new_index: 29,
|
||||||
new_key: RawKey {
|
new_key: RawKey {
|
||||||
x: [2; 32],
|
x: [2; 32],
|
||||||
y: [3; 32],
|
y_parity: true,
|
||||||
},
|
},
|
||||||
})),
|
})),
|
||||||
};
|
};
|
||||||
|
@ -296,4 +308,71 @@ mod tests {
|
||||||
let parsed_vaa = VAA::deserialize(data.as_slice()).unwrap();
|
let parsed_vaa = VAA::deserialize(data.as_slice()).unwrap();
|
||||||
assert_eq!(vaa, parsed_vaa)
|
assert_eq!(vaa, parsed_vaa)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn parse_given_guardian_set_update() {
|
||||||
|
let vaa = VAA {
|
||||||
|
version: 1,
|
||||||
|
guardian_set_index: 9,
|
||||||
|
signature_sig: [
|
||||||
|
2, 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0,
|
||||||
|
],
|
||||||
|
signature_addr: [1, 2, 3, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
|
||||||
|
timestamp: 2837,
|
||||||
|
payload: Some(VAABody::UpdateGuardianSet(BodyUpdateGuardianSet {
|
||||||
|
new_index: 2,
|
||||||
|
new_key: RawKey {
|
||||||
|
x: [
|
||||||
|
34, 23, 130, 103, 189, 101, 144, 104, 196, 19, 115, 119, 37, 80, 123, 46,
|
||||||
|
218, 191, 167, 75, 3, 40, 130, 168, 218, 203, 128, 99, 120, 238, 102, 1,
|
||||||
|
],
|
||||||
|
y_parity: true,
|
||||||
|
},
|
||||||
|
})),
|
||||||
|
};
|
||||||
|
let data = hex::decode("01000000090208000000000000000000000000000000000000000000000000000000000000010203040000000000000000000000000000000000000b15012522178267bd659068c413737725507b2edabfa74b032882a8dacb806378ee66010100000002").unwrap();
|
||||||
|
let parsed_vaa = VAA::deserialize(data.as_slice()).unwrap();
|
||||||
|
assert_eq!(vaa, parsed_vaa);
|
||||||
|
|
||||||
|
let rec_data = parsed_vaa.serialize().unwrap();
|
||||||
|
assert_eq!(data, rec_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn parse_given_transfer() {
|
||||||
|
let vaa = VAA {
|
||||||
|
version: 1,
|
||||||
|
guardian_set_index: 9,
|
||||||
|
signature_sig: [
|
||||||
|
2, 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0,
|
||||||
|
],
|
||||||
|
signature_addr: [1, 2, 3, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
|
||||||
|
timestamp: 2837,
|
||||||
|
payload: Some(VAABody::Transfer(BodyTransfer {
|
||||||
|
nonce: 38,
|
||||||
|
source_chain: 2,
|
||||||
|
target_chain: 1,
|
||||||
|
target_address: [
|
||||||
|
2, 1, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0,
|
||||||
|
],
|
||||||
|
asset: AssetMeta {
|
||||||
|
address: [
|
||||||
|
9, 2, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0,
|
||||||
|
],
|
||||||
|
chain: 9,
|
||||||
|
},
|
||||||
|
amount: U256::from(29),
|
||||||
|
})),
|
||||||
|
};
|
||||||
|
let data = hex::decode("01000000090208000000000000000000000000000000000000000000000000000000000000010203040000000000000000000000000000000000000b1510670000002602010201030000000000000000000000000000000000000000000000000000000000090902040000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001d").unwrap();
|
||||||
|
let parsed_vaa = VAA::deserialize(data.as_slice()).unwrap();
|
||||||
|
assert_eq!(vaa, parsed_vaa);
|
||||||
|
|
||||||
|
let rec_data = parsed_vaa.serialize().unwrap();
|
||||||
|
assert_eq!(data, rec_data);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -127,6 +127,16 @@ version = "1.2.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "cf1de2fe8c75bc145a2f577add951f8134889b4795d47466a54a5c846d691693"
|
checksum = "cf1de2fe8c75bc145a2f577add951f8134889b4795d47466a54a5c846d691693"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "bitvec"
|
||||||
|
version = "0.17.4"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "41262f11d771fd4a61aa3ce019fca363b4b6c282fca9da2a31186d3965a47a5c"
|
||||||
|
dependencies = [
|
||||||
|
"either",
|
||||||
|
"radium",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "blake2b_simd"
|
name = "blake2b_simd"
|
||||||
version = "0.5.10"
|
version = "0.5.10"
|
||||||
|
@ -197,6 +207,12 @@ dependencies = [
|
||||||
"serde",
|
"serde",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "byte-slice-cast"
|
||||||
|
version = "0.3.5"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "b0a5e3906bcbf133e33c1d4d95afc664ad37fbdb9f6568d8043e7ea8c27d93d3"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "byte-tools"
|
name = "byte-tools"
|
||||||
version = "0.3.1"
|
version = "0.3.1"
|
||||||
|
@ -426,6 +442,12 @@ dependencies = [
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "crunchy"
|
||||||
|
version = "0.2.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "7a81dae078cea95a014a339291cec439d2f232ebe854a9d672b796c6afafa9b7"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "crypto-mac"
|
name = "crypto-mac"
|
||||||
version = "0.7.0"
|
version = "0.7.0"
|
||||||
|
@ -597,6 +619,18 @@ version = "0.1.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "835a3dc7d1ec9e75e2b5fb4ba75396837112d2060b03f7d43bc1897c7f7211da"
|
checksum = "835a3dc7d1ec9e75e2b5fb4ba75396837112d2060b03f7d43bc1897c7f7211da"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "fixed-hash"
|
||||||
|
version = "0.6.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "11498d382790b7a8f2fd211780bec78619bba81cdad3a283997c0c41f836759c"
|
||||||
|
dependencies = [
|
||||||
|
"byteorder",
|
||||||
|
"rand",
|
||||||
|
"rustc-hex",
|
||||||
|
"static_assertions",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "fnv"
|
name = "fnv"
|
||||||
version = "1.0.7"
|
version = "1.0.7"
|
||||||
|
@ -913,6 +947,15 @@ dependencies = [
|
||||||
"unicode-normalization",
|
"unicode-normalization",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "impl-codec"
|
||||||
|
version = "0.4.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "1be51a921b067b0eaca2fad532d9400041561aa922221cc65f95a85641c6bf53"
|
||||||
|
dependencies = [
|
||||||
|
"parity-scale-codec",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "indexmap"
|
name = "indexmap"
|
||||||
version = "1.5.0"
|
version = "1.5.0"
|
||||||
|
@ -1323,6 +1366,18 @@ dependencies = [
|
||||||
"vcpkg",
|
"vcpkg",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "parity-scale-codec"
|
||||||
|
version = "1.3.4"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "34d38aeaffc032ec69faa476b3caaca8d4dd7f3f798137ff30359e5c7869ceb6"
|
||||||
|
dependencies = [
|
||||||
|
"arrayvec",
|
||||||
|
"bitvec",
|
||||||
|
"byte-slice-cast",
|
||||||
|
"serde",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "parking_lot"
|
name = "parking_lot"
|
||||||
version = "0.9.0"
|
version = "0.9.0"
|
||||||
|
@ -1433,6 +1488,17 @@ version = "0.2.8"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "237a5ed80e274dbc66f86bd59c1e25edc039660be53194b5fe0a482e0f2612ea"
|
checksum = "237a5ed80e274dbc66f86bd59c1e25edc039660be53194b5fe0a482e0f2612ea"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "primitive-types"
|
||||||
|
version = "0.7.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "c55c21c64d0eaa4d7ed885d959ef2d62d9e488c27c0e02d9aa5ce6c877b7d5f8"
|
||||||
|
dependencies = [
|
||||||
|
"fixed-hash",
|
||||||
|
"impl-codec",
|
||||||
|
"uint",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "proc-macro-hack"
|
name = "proc-macro-hack"
|
||||||
version = "0.5.18"
|
version = "0.5.18"
|
||||||
|
@ -1487,6 +1553,12 @@ dependencies = [
|
||||||
"proc-macro2 1.0.19",
|
"proc-macro2 1.0.19",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "radium"
|
||||||
|
version = "0.3.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "def50a86306165861203e7f84ecffbbdfdea79f0e51039b33de1e952358c47ac"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "rand"
|
name = "rand"
|
||||||
version = "0.7.3"
|
version = "0.7.3"
|
||||||
|
@ -1683,6 +1755,12 @@ version = "1.1.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "08d43f7aa6b08d49f382cde6a7982047c3426db949b1424bc4b7ec9ae12c6ce2"
|
checksum = "08d43f7aa6b08d49f382cde6a7982047c3426db949b1424bc4b7ec9ae12c6ce2"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "rustc-hex"
|
||||||
|
version = "2.1.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "3e75f6a532d0fd9f7f13144f392b6ad56a32696bfcd9c78f797f16bbb6f072d6"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "rustc_version"
|
name = "rustc_version"
|
||||||
version = "0.2.3"
|
version = "0.2.3"
|
||||||
|
@ -2241,6 +2319,12 @@ dependencies = [
|
||||||
"thiserror",
|
"thiserror",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "static_assertions"
|
||||||
|
version = "1.1.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "strsim"
|
name = "strsim"
|
||||||
version = "0.8.0"
|
version = "0.8.0"
|
||||||
|
@ -2698,6 +2782,18 @@ version = "1.12.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "373c8a200f9e67a0c95e62a4f52fbf80c23b4381c05a17845531982fa99e6b33"
|
checksum = "373c8a200f9e67a0c95e62a4f52fbf80c23b4381c05a17845531982fa99e6b33"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "uint"
|
||||||
|
version = "0.8.4"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "429ffcad8c8c15f874578c7337d156a3727eb4a1c2374c0ae937ad9a9b748c80"
|
||||||
|
dependencies = [
|
||||||
|
"byteorder",
|
||||||
|
"crunchy",
|
||||||
|
"rustc-hex",
|
||||||
|
"static_assertions",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "unicase"
|
name = "unicase"
|
||||||
version = "2.6.0"
|
version = "2.6.0"
|
||||||
|
@ -2965,10 +3061,10 @@ dependencies = [
|
||||||
name = "wormhole-bridge"
|
name = "wormhole-bridge"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"arrayref",
|
|
||||||
"byteorder",
|
"byteorder",
|
||||||
"num-derive 0.2.5",
|
"num-derive 0.2.5",
|
||||||
"num-traits",
|
"num-traits",
|
||||||
|
"primitive-types",
|
||||||
"remove_dir_all",
|
"remove_dir_all",
|
||||||
"sha3",
|
"sha3",
|
||||||
"solana-sdk",
|
"solana-sdk",
|
||||||
|
|
Loading…
Reference in New Issue