mirror of
https://github.com/comit-network/xmr-btc-swap.git
synced 2025-01-07 05:38:07 -05:00
[WIP] Almost done
This commit is contained in:
parent
be9c2c7d6b
commit
64a0bd7f8a
3
Cargo.lock
generated
3
Cargo.lock
generated
@ -2265,7 +2265,7 @@ dependencies = [
|
|||||||
[[package]]
|
[[package]]
|
||||||
name = "monero"
|
name = "monero"
|
||||||
version = "0.12.0"
|
version = "0.12.0"
|
||||||
source = "git+https://github.com/comit-network/monero-rs#680ba69708d5112d8650a76aa032fdb2be56ff55"
|
source = "git+https://github.com/comit-network/monero-rs#f6a500b72dc3a011b8ab3145f98f573dd0d20a2f"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"base58-monero",
|
"base58-monero",
|
||||||
"clear_on_drop",
|
"clear_on_drop",
|
||||||
@ -2291,6 +2291,7 @@ dependencies = [
|
|||||||
"anyhow",
|
"anyhow",
|
||||||
"curve25519-dalek",
|
"curve25519-dalek",
|
||||||
"hash_edwards_to_edwards",
|
"hash_edwards_to_edwards",
|
||||||
|
"hex 0.4.3",
|
||||||
"itertools 0.10.0",
|
"itertools 0.10.0",
|
||||||
"monero",
|
"monero",
|
||||||
"monero-harness",
|
"monero-harness",
|
||||||
|
@ -13,6 +13,7 @@ hash_edwards_to_edwards = { git = "https://github.com/comit-network/hash_edwards
|
|||||||
monero = "0.12"
|
monero = "0.12"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
|
hex = "0.4"
|
||||||
monero-harness = { path = "../monero-harness" }
|
monero-harness = { path = "../monero-harness" }
|
||||||
monero-rpc = { path = "../monero-rpc" }
|
monero-rpc = { path = "../monero-rpc" }
|
||||||
monero-wallet = { path = "../monero-wallet" }
|
monero-wallet = { path = "../monero-wallet" }
|
||||||
|
@ -1,16 +1,19 @@
|
|||||||
#![allow(non_snake_case)]
|
#![allow(non_snake_case)]
|
||||||
|
|
||||||
|
use monero::blockdata::transaction::KeyImage;
|
||||||
|
use monero::util::key::H;
|
||||||
|
use monero::ViewPair;
|
||||||
use curve25519_dalek::constants::ED25519_BASEPOINT_POINT;
|
use curve25519_dalek::constants::ED25519_BASEPOINT_POINT;
|
||||||
use curve25519_dalek::edwards::EdwardsPoint;
|
use curve25519_dalek::edwards::EdwardsPoint;
|
||||||
use curve25519_dalek::scalar::Scalar;
|
use curve25519_dalek::scalar::Scalar;
|
||||||
use hash_edwards_to_edwards::hash_point_to_point;
|
use hash_edwards_to_edwards::hash_point_to_point;
|
||||||
use itertools::Itertools;
|
|
||||||
use monero::blockdata::transaction::{ExtraField, SubField, TxOutTarget};
|
use monero::blockdata::transaction::{ExtraField, SubField, TxOutTarget};
|
||||||
use monero::cryptonote::hash::Hashable;
|
use monero::cryptonote::hash::Hashable;
|
||||||
use monero::cryptonote::onetime_key::KeyGenerator;
|
use monero::cryptonote::onetime_key::KeyGenerator;
|
||||||
use monero::util::ringct::{EcdhInfo, RctSig, RctSigBase, RctSigPrunable, RctType};
|
use monero::util::ringct::{EcdhInfo, RctSig, RctSigBase, RctSigPrunable, RctType};
|
||||||
use monero::{PrivateKey, PublicKey};
|
use monero::{PrivateKey, PublicKey};
|
||||||
use monero::{Transaction, TransactionPrefix, TxIn, TxOut, VarInt};
|
use monero::{Transaction, TransactionPrefix, TxIn, TxOut, VarInt};
|
||||||
|
use monero_rpc::wallet::MoneroWalletRpc as _;
|
||||||
use monero_rpc::monerod;
|
use monero_rpc::monerod;
|
||||||
use monero_rpc::monerod::{GetOutputsOut, MonerodRpc};
|
use monero_rpc::monerod::{GetOutputsOut, MonerodRpc};
|
||||||
use monero_wallet::{MonerodClientExt, Wallet};
|
use monero_wallet::{MonerodClientExt, Wallet};
|
||||||
@ -49,6 +52,7 @@ async fn monerod_integration_test() {
|
|||||||
|
|
||||||
let mut o_indexes_response = client.get_o_indexes(spend_tx).await.unwrap();
|
let mut o_indexes_response = client.get_o_indexes(spend_tx).await.unwrap();
|
||||||
|
|
||||||
|
// TODO: Cannot rely on this, because outputs are shuffled
|
||||||
let real_key_offset = o_indexes_response.o_indexes.pop().unwrap();
|
let real_key_offset = o_indexes_response.o_indexes.pop().unwrap();
|
||||||
|
|
||||||
let (lower, upper) = client.calculate_key_offset_boundaries().await.unwrap();
|
let (lower, upper) = client.calculate_key_offset_boundaries().await.unwrap();
|
||||||
@ -56,7 +60,7 @@ async fn monerod_integration_test() {
|
|||||||
let mut key_offsets = Vec::with_capacity(11);
|
let mut key_offsets = Vec::with_capacity(11);
|
||||||
key_offsets.push(VarInt(real_key_offset));
|
key_offsets.push(VarInt(real_key_offset));
|
||||||
|
|
||||||
for i in 0..10 {
|
for _ in 0..10 {
|
||||||
loop {
|
loop {
|
||||||
let decoy_offset = VarInt(rng.gen_range(lower.0, upper.0));
|
let decoy_offset = VarInt(rng.gen_range(lower.0, upper.0));
|
||||||
|
|
||||||
@ -93,12 +97,19 @@ async fn monerod_integration_test() {
|
|||||||
|
|
||||||
let relative_key_offsets = to_relative_offsets(&key_offsets);
|
let relative_key_offsets = to_relative_offsets(&key_offsets);
|
||||||
|
|
||||||
let amount = 10_000_000;
|
let lock_amount = 10_000_000;
|
||||||
let fee = 10_000;
|
let fee = 10_000;
|
||||||
// TODO: Pay lock amount (= amount + fee) to shared address (s_prime_a + s_b)
|
let spend_amount = lock_amount - fee;
|
||||||
|
// TODO: Pay lock amount to shared address (s_prime_a + s_b)
|
||||||
|
|
||||||
let (bulletproof, out_pk, out_blindings) =
|
let target_address = "498AVruCDWgP9Az9LjMm89VWjrBrSZ2W2K3HFBiyzzrRjUJWUcCVxvY1iitfuKoek2FdX6MKGAD9Qb1G1P8QgR5jPmmt3Vj".parse::<monero::Address>().unwrap();
|
||||||
monero::make_bulletproof(&mut rng, &[amount]).unwrap();
|
|
||||||
|
let ecdh_key = PrivateKey::random(&mut rng);
|
||||||
|
let (ecdh_info, out_blinding) = EcdhInfo::new_bulletproof(spend_amount, ecdh_key.scalar);
|
||||||
|
|
||||||
|
// TODO: Modify API to let us determine the blindings ahead of time
|
||||||
|
let (bulletproof, out_pk) =
|
||||||
|
monero::make_bulletproof(&mut rng, &[spend_amount], &[out_blinding]).unwrap();
|
||||||
let out_pk = out_pk
|
let out_pk = out_pk
|
||||||
.iter()
|
.iter()
|
||||||
.map(|c| monero::util::ringct::CtKey {
|
.map(|c| monero::util::ringct::CtKey {
|
||||||
@ -106,9 +117,13 @@ async fn monerod_integration_test() {
|
|||||||
})
|
})
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
let target_address = "498AVruCDWgP9Az9LjMm89VWjrBrSZ2W2K3HFBiyzzrRjUJWUcCVxvY1iitfuKoek2FdX6MKGAD9Qb1G1P8QgR5jPmmt3Vj".parse::<monero::Address>().unwrap();
|
let k_image = {
|
||||||
|
let k = lock_kp.spend.scalar;
|
||||||
|
let K = ViewPair::from(&lock_kp).spend.point;
|
||||||
|
|
||||||
let ecdh_key = PrivateKey::random(&mut rng);
|
let k_image = k * hash_point_to_point(K.decompress().unwrap());
|
||||||
|
KeyImage { image: monero::cryptonote::hash::Hash(k_image.compress().to_bytes()) }
|
||||||
|
};
|
||||||
|
|
||||||
let prefix = TransactionPrefix {
|
let prefix = TransactionPrefix {
|
||||||
version: VarInt(2),
|
version: VarInt(2),
|
||||||
@ -116,7 +131,7 @@ async fn monerod_integration_test() {
|
|||||||
inputs: vec![TxIn::ToKey {
|
inputs: vec![TxIn::ToKey {
|
||||||
amount: VarInt(0),
|
amount: VarInt(0),
|
||||||
key_offsets: relative_key_offsets,
|
key_offsets: relative_key_offsets,
|
||||||
k_image: todo!(),
|
k_image,
|
||||||
}],
|
}],
|
||||||
outputs: vec![TxOut {
|
outputs: vec![TxOut {
|
||||||
amount: VarInt(0),
|
amount: VarInt(0),
|
||||||
@ -126,7 +141,7 @@ async fn monerod_integration_test() {
|
|||||||
target_address.public_spend,
|
target_address.public_spend,
|
||||||
ecdh_key,
|
ecdh_key,
|
||||||
)
|
)
|
||||||
.one_time_key(0),
|
.one_time_key(0), // TODO: It works with 1 output, but we must choose it based on the output index
|
||||||
},
|
},
|
||||||
}],
|
}],
|
||||||
extra: ExtraField(vec![SubField::TxPublicKey(PublicKey::from_private_key(
|
extra: ExtraField(vec![SubField::TxPublicKey(PublicKey::from_private_key(
|
||||||
@ -139,6 +154,14 @@ async fn monerod_integration_test() {
|
|||||||
|
|
||||||
let sig = adaptor_sig.adapt(adaptor);
|
let sig = adaptor_sig.adapt(adaptor);
|
||||||
|
|
||||||
|
let pseudo_out = {
|
||||||
|
let amount = Scalar::from(lock_amount);
|
||||||
|
let blinding = -out_blinding;
|
||||||
|
|
||||||
|
let commitment = (blinding * ED25519_BASEPOINT_POINT) + (amount * H.point.decompress().unwrap());
|
||||||
|
|
||||||
|
monero::util::ringct::Key { key: commitment.compress().to_bytes() }
|
||||||
|
};
|
||||||
let transaction = Transaction {
|
let transaction = Transaction {
|
||||||
prefix,
|
prefix,
|
||||||
signatures: Vec::new(),
|
signatures: Vec::new(),
|
||||||
@ -147,7 +170,7 @@ async fn monerod_integration_test() {
|
|||||||
rct_type: RctType::Clsag,
|
rct_type: RctType::Clsag,
|
||||||
txn_fee: VarInt(fee),
|
txn_fee: VarInt(fee),
|
||||||
pseudo_outs: Vec::new(),
|
pseudo_outs: Vec::new(),
|
||||||
ecdh_info: todo!(),
|
ecdh_info: vec![ecdh_info],
|
||||||
out_pk,
|
out_pk,
|
||||||
}),
|
}),
|
||||||
p: Some(RctSigPrunable {
|
p: Some(RctSigPrunable {
|
||||||
@ -155,12 +178,17 @@ async fn monerod_integration_test() {
|
|||||||
bulletproofs: vec![bulletproof],
|
bulletproofs: vec![bulletproof],
|
||||||
MGs: Vec::new(),
|
MGs: Vec::new(),
|
||||||
Clsags: vec![sig.into()],
|
Clsags: vec![sig.into()],
|
||||||
pseudo_outs: todo!("out_blindings + pseudo_outs == 0, 1 pseudo out per input: calculated by input amount * G + H * 'random blinding factor'"),
|
pseudo_outs: vec![pseudo_out],
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
todo!("broadcast transaction")
|
let wallet_client = monero_rpc::wallet::Client::localhost(0).unwrap();
|
||||||
|
|
||||||
|
let tx_hex = hex::encode(monero::consensus::encode::serialize(&transaction));
|
||||||
|
let tx_hash = wallet_client.submit_transfer(tx_hex).await.unwrap();
|
||||||
|
|
||||||
|
dbg!(tx_hash);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn to_relative_offsets(offsets: &[VarInt]) -> Vec<VarInt> {
|
fn to_relative_offsets(offsets: &[VarInt]) -> Vec<VarInt> {
|
||||||
|
@ -182,7 +182,7 @@ pub struct GetOutsResponse {
|
|||||||
pub outs: Vec<OutKey>,
|
pub outs: Vec<OutKey>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug, Deserialize, PartialEq)]
|
#[derive(Clone, Copy, Debug, Deserialize, PartialEq)]
|
||||||
pub struct OutKey {
|
pub struct OutKey {
|
||||||
pub height: u64,
|
pub height: u64,
|
||||||
#[serde(with = "byte_array")]
|
#[serde(with = "byte_array")]
|
||||||
|
@ -16,6 +16,10 @@ pub trait MoneroWalletRpc {
|
|||||||
destinations: Vec<Destination>,
|
destinations: Vec<Destination>,
|
||||||
get_tx_key: bool,
|
get_tx_key: bool,
|
||||||
) -> Transfer;
|
) -> Transfer;
|
||||||
|
async fn submit_transfer(
|
||||||
|
&self,
|
||||||
|
tx_data_hex: String,
|
||||||
|
) -> Vec<String>;
|
||||||
async fn get_height(&self) -> BlockHeight;
|
async fn get_height(&self) -> BlockHeight;
|
||||||
async fn check_tx_key(&self, txid: String, tx_key: String, address: String) -> CheckTxKey;
|
async fn check_tx_key(&self, txid: String, tx_key: String, address: String) -> CheckTxKey;
|
||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
|
@ -3,7 +3,7 @@ use monero::consensus::encode::VarInt;
|
|||||||
use monero::cryptonote::hash::Hashable;
|
use monero::cryptonote::hash::Hashable;
|
||||||
use monero_rpc::monerod;
|
use monero_rpc::monerod;
|
||||||
use monero_rpc::monerod::{GetBlockResponse, MonerodRpc as _};
|
use monero_rpc::monerod::{GetBlockResponse, MonerodRpc as _};
|
||||||
use rand::{CryptoRng, Rng, RngCore};
|
use rand::{CryptoRng, RngCore};
|
||||||
|
|
||||||
pub struct Wallet {
|
pub struct Wallet {
|
||||||
key: monero::KeyPair,
|
key: monero::KeyPair,
|
||||||
|
Loading…
Reference in New Issue
Block a user