-
Notifications
You must be signed in to change notification settings - Fork 454
/
Copy pathlib.rs
157 lines (140 loc) · 5.22 KB
/
lib.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
#![cfg_attr(not(feature = "std"), no_std, no_main)]
#[ink::contract]
mod contract_xcm {
use ink::{
env::Error as EnvError,
xcm::prelude::*,
};
/// A smart contract example using the XCM API for cross-chain communication.
#[ink(storage)]
#[derive(Default)]
pub struct ContractXcm;
/// Enumeration of runtime errors for the contract.
#[derive(Debug, PartialEq, Eq)]
#[ink::scale_derive(Encode, Decode, TypeInfo)]
pub enum RuntimeError {
XcmExecuteFailed,
XcmSendFailed,
UnexpectedEnvError,
}
impl From<EnvError> for RuntimeError {
fn from(e: EnvError) -> Self {
match e {
EnvError::ReturnError(code) => match code {
ink::env::ReturnErrorCode::XcmExecutionFailed => RuntimeError::XcmExecuteFailed,
ink::env::ReturnErrorCode::XcmSendFailed => RuntimeError::XcmSendFailed,
_ => RuntimeError::UnexpectedEnvError,
},
_ => RuntimeError::UnexpectedEnvError,
}
}
}
impl ContractXcm {
/// The constructor is `payable`, allowing the contract to receive initial tokens.
#[ink(constructor, payable)]
pub fn new() -> Self {
Default::default()
}
/// Helper function to build an XCM message.
///
/// # Arguments
/// * `receiver` - The target account to receive the assets.
/// * `value` - The amount of tokens to transfer.
/// * `fee` - Optional fee for the XCM execution.
fn build_xcm_message(
&self,
receiver: AccountId32,
value: Balance,
fee: Option<Balance>,
) -> Xcm<()> {
let asset: Asset = (Here, value).into();
let mut builder = Xcm::builder()
.withdraw_asset(asset.clone())
.deposit_asset(asset.clone(), receiver);
if let Some(fee) = fee {
builder = builder.buy_execution((Here, fee), WeightLimit::Unlimited);
}
builder.build()
}
/// Transfers funds through XCM to the given receiver.
///
/// Fails if:
/// - XCM execution fails.
/// - Insufficient funds.
/// - Unsupported environment or runtime configuration.
#[ink(message)]
pub fn transfer_through_xcm(
&mut self,
receiver: AccountId,
value: Balance,
) -> Result<(), RuntimeError> {
let beneficiary = AccountId32 {
network: None,
id: *receiver.as_ref(),
};
let message = self.build_xcm_message(beneficiary, value, None);
self.env()
.xcm_execute(&VersionedXcm::V4(message))
.map_err(Into::into)
}
/// Sends funds through XCM, paying a fee for execution.
///
/// Fails if:
/// - XCM execution fails.
/// - Insufficient funds or fees.
/// - Unsupported environment or runtime configuration.
#[ink(message)]
pub fn send_funds(
&mut self,
value: Balance,
fee: Balance,
) -> Result<XcmHash, RuntimeError> {
let beneficiary = AccountId32 {
network: None,
id: *self.env().caller().as_ref(),
};
let destination: Location = Parent.into();
let message = self.build_xcm_message(beneficiary, value, Some(fee));
let hash = self
.env()
.xcm_send(&VersionedLocation::V4(destination), &VersionedXcm::V4(message))?;
Ok(hash)
}
}
#[cfg(all(test, feature = "e2e-tests"))]
mod e2e_tests {
use super::*;
use ink::{
env::{test::default_accounts, DefaultEnvironment},
primitives::AccountId,
};
use ink_e2e::{
preset::mock_network::{primitives::CENTS, MockNetworkSandbox},
ChainBackend,
};
/// Initial contract balance for testing.
pub const CONTRACT_BALANCE: u128 = 1_000_000;
type E2EResult<T> = Result<T, Box<dyn std::error::Error>>;
#[ink_e2e::test(backend(runtime_only(sandbox = MockNetworkSandbox)))]
async fn transfer_through_xcm_works<Client: E2EBackend>(
mut client: Client,
) -> E2EResult<()> {
// Arrange: Instantiate the contract with a predefined balance.
let mut constructor = ContractXcmRef::new();
let contract = client
.instantiate("contract_xcm", &ink_e2e::alice(), &mut constructor)
.value(CONTRACT_BALANCE)
.submit()
.await
.expect("instantiation failed");
let receiver: AccountId = default_accounts::<DefaultEnvironment>().bob;
// Act: Execute the transfer through XCM.
let transfer_message = contract
.call_builder()
.transfer_through_xcm(receiver, 1_000 * CENTS);
let result = client.call(&ink_e2e::alice(), &transfer_message).submit().await?;
assert!(result.return_value().is_ok());
Ok(())
}
}
}