Skip to content

Commit

Permalink
TMP
Browse files Browse the repository at this point in the history
  • Loading branch information
Tibo-lg committed Oct 24, 2023
1 parent 80cd50c commit 65e15bb
Show file tree
Hide file tree
Showing 37 changed files with 3,250 additions and 383 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/rust.yml
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@ jobs:
key: test-cache-${{ github.run_id }}-${{ github.run_number }}
- uses: actions/checkout@v2
- id: set-matrix
run: cargo test --no-run && echo "::set-output name=matrix::$(scripts/get_test_list.sh execution manager channel_execution)"
run: cargo test --no-run && echo "::set-output name=matrix::$(scripts/get_test_list.sh execution manager channel_execution ordinals)"
integration_tests:
name: integration-tests
needs: integration_tests_prepare
Expand Down
2 changes: 2 additions & 0 deletions bitcoin-rpc-provider/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -12,4 +12,6 @@ dlc-manager = {path = "../dlc-manager"}
lightning = {version = "0.0.116"}
log = "0.4.14"
rust-bitcoin-coin-selection = { version = "0.1.0", git = "https://github.com/p2pderivatives/rust-bitcoin-coin-selection", rev = "405451929568422f7df809e35d6ad8f36fccce90", features = ["rand"] }
serde = "1.0"
simple-wallet = {path = "../simple-wallet"}
secp256k1-zkp = {version = "0.7.0", default-features=false, features = ["global-context"]}
164 changes: 125 additions & 39 deletions bitcoin-rpc-provider/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -13,8 +13,8 @@ use bitcoin::{
Txid,
};
use bitcoin::{Address, OutPoint, TxOut};
use bitcoincore_rpc::jsonrpc::serde_json::{self};
use bitcoincore_rpc::{json, Auth, Client, RpcApi};
use bitcoincore_rpc_json::AddressType;
use dlc_manager::error::Error as ManagerError;
use dlc_manager::{Blockchain, Signer, Utxo, Wallet};
use json::EstimateMode;
Expand All @@ -30,6 +30,8 @@ pub struct BitcoinCoreProvider {
// Used to implement the FeeEstimator interface, heavily inspired by
// https://github.com/lightningdevkit/ldk-sample/blob/main/src/bitcoind_client.rs#L26
fees: Arc<HashMap<ConfirmationTarget, AtomicU32>>,
/// Indicates whether the wallet is descriptor based or not.
is_descriptor: bool,
}

#[derive(Debug)]
Expand Down Expand Up @@ -95,18 +97,33 @@ impl BitcoinCoreProvider {
rpc_base
};
let auth = Auth::UserPass(rpc_user, rpc_password);
Ok(Self::new_from_rpc_client(Client::new(&rpc_url, auth)?))
Self::new_from_rpc_client(Client::new(&rpc_url, auth)?)
}

pub fn new_from_rpc_client(rpc_client: Client) -> Self {
pub fn new_from_rpc_client(rpc_client: Client) -> Result<Self, Error> {
let client = Arc::new(Mutex::new(rpc_client));
let mut fees: HashMap<ConfirmationTarget, AtomicU32> = HashMap::new();
fees.insert(ConfirmationTarget::Background, AtomicU32::new(MIN_FEERATE));
fees.insert(ConfirmationTarget::Normal, AtomicU32::new(2000));
fees.insert(ConfirmationTarget::HighPriority, AtomicU32::new(5000));
let fees = Arc::new(fees);
poll_for_fee_estimates(client.clone(), fees.clone());
BitcoinCoreProvider { client, fees }

#[derive(serde::Deserialize)]
struct Descriptor {
descriptors: bool,
}

let is_descriptor = client
.lock()
.unwrap()
.call::<Descriptor>("getwalletinfo", &[])?
.descriptors;
Ok(BitcoinCoreProvider {
client,
fees,
is_descriptor,
})
}
}

Expand Down Expand Up @@ -144,22 +161,46 @@ fn enc_err_to_manager_err(_e: EncodeError) -> ManagerError {
Error::BitcoinError.into()
}

#[derive(serde::Deserialize, Debug)]
struct DescriptorInfo {
desc: String,
}

#[derive(serde::Deserialize, Debug)]
struct DescriptorListResponse {
descriptors: Vec<DescriptorInfo>,
}

impl Signer for BitcoinCoreProvider {
fn get_secret_key_for_pubkey(&self, pubkey: &PublicKey) -> Result<SecretKey, ManagerError> {
let b_pubkey = bitcoin::PublicKey {
compressed: true,
inner: *pubkey,
};
let address =
Address::p2wpkh(&b_pubkey, self.get_network()?).or(Err(Error::BitcoinError))?;

let pk = self
.client
.lock()
.unwrap()
.dump_private_key(&address)
.map_err(rpc_err_to_manager_err)?;
Ok(pk.inner)
if self.is_descriptor {
let client = self.client.lock().unwrap();
let DescriptorListResponse { descriptors } = client
.call::<DescriptorListResponse>("listdescriptors", &[serde_json::Value::Bool(true)])
.map_err(rpc_err_to_manager_err)?;
descriptors
.iter()
.filter_map(|x| descriptor_to_secret_key(&x.desc))
.find(|x| x.public_key(secp256k1_zkp::SECP256K1) == *pubkey)
.ok_or(ManagerError::InvalidState(
"Expected a descriptor at this position".to_string(),
))
} else {
let b_pubkey = bitcoin::PublicKey {
compressed: true,
inner: *pubkey,
};
let address =
Address::p2wpkh(&b_pubkey, self.get_network()?).or(Err(Error::BitcoinError))?;

let pk = self
.client
.lock()
.unwrap()
.dump_private_key(&address)
.map_err(rpc_err_to_manager_err)?;
Ok(pk.inner)
}
}

fn sign_tx_input(
Expand Down Expand Up @@ -200,27 +241,34 @@ impl Wallet for BitcoinCoreProvider {
self.client
.lock()
.unwrap()
.get_new_address(None, Some(AddressType::Bech32))
.call(
"getnewaddress",
&[
serde_json::Value::Null,
serde_json::Value::String("bech32m".to_string()),
],
)
.map_err(rpc_err_to_manager_err)
}

fn get_new_secret_key(&self) -> Result<SecretKey, ManagerError> {
let sk = SecretKey::new(&mut thread_rng());
let network = self.get_network()?;
self.client
.lock()
.unwrap()
.import_private_key(
&PrivateKey {
compressed: true,
network,
inner: sk,
},
None,
Some(false),
)
.map_err(rpc_err_to_manager_err)?;

let client = self.client.lock().unwrap();
let pk = PrivateKey {
compressed: true,
network,
inner: sk,
};
if self.is_descriptor {
let wif = pk.to_wif();
let desc = format!("rawtr({wif})");
import_descriptor(&client, &desc)?;
} else {
client
.import_private_key(&pk, None, Some(false))
.map_err(rpc_err_to_manager_err)?;
}
Ok(sk)
}

Expand All @@ -234,9 +282,18 @@ impl Wallet for BitcoinCoreProvider {
let utxo_res = client
.list_unspent(None, None, None, Some(false), None)
.map_err(rpc_err_to_manager_err)?;
let locked = client
.call::<Vec<serde_json::Value>>("listlockunspent", &[])
.map_err(rpc_err_to_manager_err)?
.iter()
.map(|x| OutPoint {
txid: x["txid"].as_str().unwrap().parse().unwrap(),
vout: x["vout"].as_u64().unwrap() as u32,
})
.collect::<Vec<_>>();
let mut utxo_pool: Vec<UtxoWrap> = utxo_res
.iter()
.filter(|x| x.spendable)
.filter(|x| x.spendable && locked.iter().all(|y| y.txid != x.txid || y.vout != x.vout))
.map(|x| {
Ok(UtxoWrap(Utxo {
tx_out: TxOut {
Expand Down Expand Up @@ -267,11 +324,16 @@ impl Wallet for BitcoinCoreProvider {
}

fn import_address(&self, address: &Address) -> Result<(), ManagerError> {
self.client
.lock()
.unwrap()
.import_address(address, None, Some(false))
.map_err(rpc_err_to_manager_err)
if self.is_descriptor {
let desc = format!("addr({address})");
import_descriptor(&self.client.lock().unwrap(), &desc)
} else {
self.client
.lock()
.unwrap()
.import_address(address, None, Some(false))
.map_err(rpc_err_to_manager_err)
}
}
}

Expand Down Expand Up @@ -412,3 +474,27 @@ fn poll_for_fee_estimates(
std::thread::sleep(Duration::from_secs(60));
});
}

fn import_descriptor(client: &Client, desc: &str) -> Result<(), ManagerError> {
let info = client
.get_descriptor_info(desc)
.map_err(rpc_err_to_manager_err)?;
let checksum = info.checksum;
let args: serde_json::Value = serde_json::from_str(&format!(
"[{{ \"desc\": \"{desc}#{checksum}\", \"timestamp\": \"now\" }}]"
))
.unwrap();
client
.call::<Vec<serde_json::Value>>("importdescriptors", &[args])
.map_err(rpc_err_to_manager_err)?;
Ok(())
}

fn descriptor_to_secret_key(desc: &str) -> Option<SecretKey> {
if !desc.starts_with("rawtr") {
return None;
}
let wif = desc.split_once('(')?.1.split_once(')')?.0;
let priv_key = PrivateKey::from_wif(wif).ok()?;
Some(priv_key.inner)
}
7 changes: 3 additions & 4 deletions bitcoin-test-utils/src/rpc_helpers.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,4 @@
use bitcoincore_rpc::{Auth, Client, RpcApi};
use bitcoincore_rpc_json::AddressType;
use std::env;

pub const OFFER_PARTY: &str = "alice";
Expand Down Expand Up @@ -64,13 +63,13 @@ pub fn init_clients() -> (Client, Client, Client) {
let sink_rpc = get_new_wallet_rpc(&rpc, SINK, auth).unwrap();

let offer_address = offer_rpc
.get_new_address(None, Some(AddressType::Bech32))
.call("getnewaddress", &["".into(), "bech32m".into()])
.unwrap();
let accept_address = accept_rpc
.get_new_address(None, Some(AddressType::Bech32))
.call("getnewaddress", &["".into(), "bech32m".into()])
.unwrap();
let sink_address = sink_rpc
.get_new_address(None, Some(AddressType::Bech32))
.call("getnewaddress", &["".into(), "bech32m".into()])
.unwrap();

sink_rpc.generate_to_address(1, &offer_address).unwrap();
Expand Down
Loading

0 comments on commit 65e15bb

Please sign in to comment.