init
This commit is contained in:
commit
a85202b8b8
1062
Cargo.lock
generated
Normal file
1062
Cargo.lock
generated
Normal file
File diff suppressed because it is too large
Load Diff
18
Cargo.toml
Normal file
18
Cargo.toml
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
[package]
|
||||||
|
name = "bal-pusher"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
[dependencies]
|
||||||
|
log = "0.4.21"
|
||||||
|
env_logger = "0.11.5"
|
||||||
|
bitcoincore-rpc = "0.19.0"
|
||||||
|
bitcoincore-rpc-json = "0.19.0"
|
||||||
|
sqlite = "0.34.0"
|
||||||
|
confy = "0.6.1"
|
||||||
|
serde = { version = "1.0.152", features = ["derive"] } # <- Only one serde version needed (serde or serde_derive)
|
||||||
|
zmq = "0.10.0"
|
||||||
|
hex = "0.4.3"
|
||||||
|
byteorder = "1.5.0"
|
||||||
|
|
68
README.md
Normal file
68
README.md
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
# bal-pusher
|
||||||
|
|
||||||
|
`bal-pusher` is a tool that retrieves Bitcoin transactions from a database and pushes them to the Bitcoin network when their **locktime** exceeds the **median time past** (MTP). It listens for Bitcoin block updates via ZMQ.
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
To use `bal-pusher`, you need to compile and install Bitcoin with ZMQ (ZeroMQ) support enabled. Then, configure your Bitcoin node and `bal-pusher` to push the transactions.
|
||||||
|
|
||||||
|
### Prerequisites
|
||||||
|
|
||||||
|
1. **Bitcoin with ZMQ Support**:
|
||||||
|
Ensure that Bitcoin is compiled with ZMQ support. Add the following line to your `bitcoin.conf` file:
|
||||||
|
|
||||||
|
```
|
||||||
|
zmqpubhashblock=tcp://127.0.0.1:28332
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Install Rust and Cargo**:
|
||||||
|
If you haven't already installed Rust and Cargo, you can follow the official instructions to do so: [Rust Installation](https://www.rust-lang.org/tools/install).
|
||||||
|
|
||||||
|
### Installation Steps
|
||||||
|
|
||||||
|
1. Clone the repository:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git clone <repo-url>
|
||||||
|
cd bal-pusher
|
||||||
|
```
|
||||||
|
|
||||||
|
2. Build the project:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cargo build --release
|
||||||
|
```
|
||||||
|
|
||||||
|
3. Install the binary:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
sudo cp target/release/bal-pusher /usr/local/bin
|
||||||
|
```
|
||||||
|
|
||||||
|
## Configuration
|
||||||
|
|
||||||
|
`bal-pusher` can be configured using environment variables. If no configuration file is provided, a default configuration file will be created.
|
||||||
|
|
||||||
|
### Available Configuration Variables
|
||||||
|
|
||||||
|
| Variable | Description | Default |
|
||||||
|
|---------------------------------------|------------------------------------------|----------------------------------------------|
|
||||||
|
| `BAL_PUSHER_CONFIG_FILE` | Path to the configuration file. If the file does not exist, it will be created. | `$HOME/.config/bal-pusher/default-config.toml` |
|
||||||
|
| `BAL_PUSHER_DB_FILE` | Path to the SQLite3 database file. If the file does not exist, it will be created. | `bal.db` |
|
||||||
|
| `BAL_PUSHER_ZMQ_LISTENER` | ZMQ listener for Bitcoin updates. | `tcp://127.0.0.1:28332` |
|
||||||
|
| `BAL_PUSHER_BITCOIN_HOST` | Bitcoin server host for RPC connections. | `http://127.0.0.1` |
|
||||||
|
| `BAL_PUSHER_BITCOIN_PORT` | Bitcoin RPC server port. | `8332` |
|
||||||
|
| `BAL_PUSHER_BITCOIN_COOKIE_FILE` | Path to Bitcoin RPC cookie file. | `$HOME/.bitcoin/.cookie` |
|
||||||
|
| `BAL_PUSHER_BITCOIN_RPC_USER` | Bitcoin RPC username. | - |
|
||||||
|
| `BAL_PUSHER_BITCOIN_RPC_PASSWORD` | Bitcoin RPC password. | - |
|
||||||
|
|
||||||
|
|
||||||
|
## Running `bal-pusher`
|
||||||
|
|
||||||
|
Once the application is installed and configured, you can start `bal-pusher` by running the following command:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ bal-pusher
|
||||||
|
```
|
||||||
|
|
||||||
|
This will start the service, which will listen for Bitcoin blocks via ZMQ and push transactions from the database when their locktime exceeds the median time past.
|
10
bal-pusher.env
Normal file
10
bal-pusher.env
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
RUST_LOG=info
|
||||||
|
|
||||||
|
|
||||||
|
BAL_PUSHER_DB_FILE=/home/bal/bal.db
|
||||||
|
BAL_PUSHER_BITCOIN_COOKIE_FILE=/home/bitcoin/.bitcoin/.cookie
|
||||||
|
BAL_PUSHER_REGTEST_COOKIE_FILE=/home/bitcoin/.bitcoin/regtest/.cookie
|
||||||
|
BAL_PUSHER_TESTNET_COOKIE_FILE=/home/bitcoin/.bitcoin/testnet3/.cookie
|
||||||
|
BAL_PUSHER_SIGNET_COOKIE_FILE=/home/bitcoin/.bitcoin/signet/.cookie
|
||||||
|
|
||||||
|
|
42
bal-pusher.service
Normal file
42
bal-pusher.service
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
[Unit]
|
||||||
|
Description=bal-pusher daemon
|
||||||
|
After=bitcoind.service
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
|
||||||
|
EnvironmentFile=/etc/bal/bal-pusher.env
|
||||||
|
|
||||||
|
ExecStart=/usr/local/bin/bal-pusher bitcoin
|
||||||
|
|
||||||
|
StandardOutput=syslog
|
||||||
|
StandardError=syslog
|
||||||
|
SyslogIdentifier=bal-pusher
|
||||||
|
|
||||||
|
|
||||||
|
Type=simple
|
||||||
|
PIDFile=/run/bal-pusher/bal-pusher.pid
|
||||||
|
Restart=always
|
||||||
|
TimeoutSec=120
|
||||||
|
RestartSec=300
|
||||||
|
KillMode=process
|
||||||
|
|
||||||
|
User=bal
|
||||||
|
Group=bitcoin
|
||||||
|
UMask=0027
|
||||||
|
|
||||||
|
RuntimeDirectory=bal-pusher
|
||||||
|
RuntimeDirectoryMode=0710
|
||||||
|
|
||||||
|
PrivateTmp=true
|
||||||
|
|
||||||
|
ProtectSystem=full
|
||||||
|
|
||||||
|
NoNewPrivileges=true
|
||||||
|
|
||||||
|
PrivateDevices=true
|
||||||
|
|
||||||
|
MemoryDenyWriteExecute=true
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
|
|
57
bitcoind.service
Normal file
57
bitcoind.service
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
# RaspiBolt: systemd unit for bitcoind
|
||||||
|
# /etc/systemd/system/bitcoind.service
|
||||||
|
|
||||||
|
[Unit]
|
||||||
|
Description=Bitcoin daemon
|
||||||
|
After=network.target
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
|
||||||
|
# Service execution
|
||||||
|
###################
|
||||||
|
|
||||||
|
ExecStart=/usr/local/bin/bitcoind -daemon \
|
||||||
|
-pid=/run/bitcoind/bitcoind.pid \
|
||||||
|
-conf=/home/bitcoin/.bitcoin/bitcoin.conf \
|
||||||
|
-datadir=/home/bitcoin/.bitcoin \
|
||||||
|
-startupnotify="chmod g+r /home/bitcoin/.bitcoin/.cookie"
|
||||||
|
|
||||||
|
# Process management
|
||||||
|
####################
|
||||||
|
Type=forking
|
||||||
|
PIDFile=/run/bitcoind/bitcoind.pid
|
||||||
|
Restart=on-failure
|
||||||
|
TimeoutSec=300
|
||||||
|
RestartSec=30
|
||||||
|
|
||||||
|
# Directory creation and permissions
|
||||||
|
####################################
|
||||||
|
User=bitcoin
|
||||||
|
UMask=0027
|
||||||
|
|
||||||
|
# /run/bitcoind
|
||||||
|
RuntimeDirectory=bitcoind
|
||||||
|
RuntimeDirectoryMode=0710
|
||||||
|
|
||||||
|
# Hardening measures
|
||||||
|
####################
|
||||||
|
# Provide a private /tmp and /var/tmp.
|
||||||
|
PrivateTmp=true
|
||||||
|
|
||||||
|
# Mount /usr, /boot/ and /etc read-only for the process.
|
||||||
|
ProtectSystem=full
|
||||||
|
|
||||||
|
# Disallow the process and all of its children to gain
|
||||||
|
# new privileges through execve().
|
||||||
|
NoNewPrivileges=true
|
||||||
|
|
||||||
|
# Use a new /dev namespace only populated with API pseudo devices
|
||||||
|
# such as /dev/null, /dev/zero and /dev/random.
|
||||||
|
PrivateDevices=true
|
||||||
|
|
||||||
|
# Deny the creation of writable and executable memory mappings.
|
||||||
|
MemoryDenyWriteExecute=true
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
|
|
416
src/main.rs
Normal file
416
src/main.rs
Normal file
@ -0,0 +1,416 @@
|
|||||||
|
extern crate bitcoincore_rpc;
|
||||||
|
extern crate zmq;
|
||||||
|
use bitcoin::Network;
|
||||||
|
|
||||||
|
use bitcoincore_rpc::{bitcoin, Auth, Client, Error, RpcApi};
|
||||||
|
use bitcoincore_rpc_json::GetBlockchainInfoResult;
|
||||||
|
|
||||||
|
use sqlite::{Value};
|
||||||
|
use serde::Serialize;
|
||||||
|
use serde::Deserialize;
|
||||||
|
use std::env;
|
||||||
|
use std::fs::OpenOptions;
|
||||||
|
use std::io::{ Write};
|
||||||
|
use log::{info,debug,warn,error};
|
||||||
|
use zmq::{Context, Socket};
|
||||||
|
use std::str;
|
||||||
|
use std::{thread, time::Duration};
|
||||||
|
//use byteorder::{LittleEndian, ReadBytesExt};
|
||||||
|
//use std::io::Cursor;
|
||||||
|
use hex;
|
||||||
|
use std::error::Error as StdError;
|
||||||
|
|
||||||
|
const LOCKTIME_THRESHOLD:i64 = 5000000;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone,Serialize, Deserialize)]
|
||||||
|
struct MyConfig {
|
||||||
|
zmq_listener: String,
|
||||||
|
requests_file: String,
|
||||||
|
db_file: String,
|
||||||
|
bitcoin_dir: String,
|
||||||
|
regtest: NetworkParams,
|
||||||
|
testnet: NetworkParams,
|
||||||
|
signet: NetworkParams,
|
||||||
|
mainnet: NetworkParams,
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for MyConfig {
|
||||||
|
fn default() -> Self {
|
||||||
|
MyConfig {
|
||||||
|
zmq_listener: "tcp://127.0.0.1:28332".to_string(),
|
||||||
|
requests_file: "rawrequests.log".to_string(),
|
||||||
|
db_file: "../bal.db".to_string(),
|
||||||
|
bitcoin_dir: "".to_string(),
|
||||||
|
regtest: get_network_params_default(Network::Regtest),
|
||||||
|
testnet: get_network_params_default(Network::Testnet),
|
||||||
|
signet: get_network_params_default(Network::Signet),
|
||||||
|
mainnet: get_network_params_default(Network::Bitcoin),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||||
|
struct NetworkParams {
|
||||||
|
host: String,
|
||||||
|
port: u16,
|
||||||
|
dir_path: String,
|
||||||
|
db_field: String,
|
||||||
|
cookie_file: String,
|
||||||
|
rpc_user: String,
|
||||||
|
rpc_pass: String,
|
||||||
|
}
|
||||||
|
fn get_network_params(cfg: &MyConfig,network:Network)-> &NetworkParams{
|
||||||
|
match network{
|
||||||
|
Network::Testnet => &cfg.testnet,
|
||||||
|
Network::Signet => &cfg.signet,
|
||||||
|
Network::Regtest => &cfg.regtest,
|
||||||
|
_ => &cfg.mainnet
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn get_network_params_default(network:Network) -> NetworkParams{
|
||||||
|
match network {
|
||||||
|
Network::Testnet => NetworkParams{
|
||||||
|
host: "http://localhost".to_string(),
|
||||||
|
port: 18332,
|
||||||
|
dir_path: "testnet3/".to_string(),
|
||||||
|
db_field: "testnet".to_string(),
|
||||||
|
cookie_file: "".to_string(),
|
||||||
|
rpc_user: "".to_string(),
|
||||||
|
rpc_pass: "".to_string(),
|
||||||
|
},
|
||||||
|
Network::Signet => NetworkParams{
|
||||||
|
host: "http://localhost".to_string(),
|
||||||
|
port: 18332,
|
||||||
|
dir_path: "signet/".to_string(),
|
||||||
|
db_field: "signet".to_string(),
|
||||||
|
cookie_file: "".to_string(),
|
||||||
|
rpc_user: "".to_string(),
|
||||||
|
rpc_pass: "".to_string(),
|
||||||
|
},
|
||||||
|
Network::Regtest => NetworkParams{
|
||||||
|
host: "http://localhost".to_string(),
|
||||||
|
port: 18443,
|
||||||
|
dir_path: "regtest/".to_string(),
|
||||||
|
db_field: "regtest".to_string(),
|
||||||
|
cookie_file: "".to_string(),
|
||||||
|
rpc_user: "".to_string(),
|
||||||
|
rpc_pass: "".to_string(),
|
||||||
|
},
|
||||||
|
_ => NetworkParams{
|
||||||
|
host: "http://localhost".to_string(),
|
||||||
|
port: 8332,
|
||||||
|
dir_path: "".to_string(),
|
||||||
|
db_field: "bitcoin".to_string(),
|
||||||
|
cookie_file: "".to_string(),
|
||||||
|
rpc_user: "".to_string(),
|
||||||
|
rpc_pass: "".to_string(),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_cookie_filename(network: &NetworkParams) ->Result<String,Box<dyn StdError>>{
|
||||||
|
if network.cookie_file !=""{
|
||||||
|
Ok(network.cookie_file.clone())
|
||||||
|
}else{
|
||||||
|
match env::var_os("HOME") {
|
||||||
|
Some(home) => {
|
||||||
|
info!("some home {}",home.to_str().unwrap());
|
||||||
|
match home.to_str(){
|
||||||
|
Some(home_str) => {
|
||||||
|
let cookie_file_path = format!("{}/.bitcoin/{}.cookie",home_str, network.dir_path);
|
||||||
|
|
||||||
|
Ok(cookie_file_path)
|
||||||
|
},
|
||||||
|
None => Err("wrong HOME value".into())
|
||||||
|
}
|
||||||
|
},
|
||||||
|
None => Err("Please Set HOME environment variable".into())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn get_client_from_username(url: &String, network: &NetworkParams) -> Result<(Client,GetBlockchainInfoResult),Box<dyn StdError>>{
|
||||||
|
if network.rpc_user != "" {
|
||||||
|
match Client::new(&url[..],Auth::UserPass(network.rpc_user.to_string(),network.rpc_pass.to_string())){
|
||||||
|
Ok(client) => match client.get_blockchain_info(){
|
||||||
|
Ok(bcinfo) => Ok((client,bcinfo)),
|
||||||
|
Err(err) => Err(err.into())
|
||||||
|
}
|
||||||
|
Err(err)=>Err(err.into())
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
Err("Failed".into())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn get_client_from_cookie(url: &String,network: &NetworkParams)->Result<(Client,GetBlockchainInfoResult),Box<dyn StdError>>{
|
||||||
|
match get_cookie_filename(network){
|
||||||
|
Ok(cookie) => {
|
||||||
|
match Client::new(&url[..], Auth::CookieFile(cookie.into())) {
|
||||||
|
Ok(client) => match client.get_blockchain_info(){
|
||||||
|
Ok(bcinfo) => Ok((client,bcinfo)),
|
||||||
|
Err(err) => Err(err.into())
|
||||||
|
},
|
||||||
|
Err(err)=>Err(err.into())
|
||||||
|
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Err(err)=>Err(err.into())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn get_client(network: &NetworkParams) -> Result<(Client,GetBlockchainInfoResult),Box<dyn StdError>>{
|
||||||
|
let url = format!("{}:{}",network.host,&network.port);
|
||||||
|
match get_client_from_username(&url,network){
|
||||||
|
Ok(client) =>{Ok(client)},
|
||||||
|
Err(_) =>{
|
||||||
|
match get_client_from_cookie(&url,&network){
|
||||||
|
Ok(client)=>{
|
||||||
|
Ok(client)
|
||||||
|
},
|
||||||
|
Err(err)=> Err(err.into())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn main_result(cfg: &MyConfig, network_params: &NetworkParams) -> Result<(), Error> {
|
||||||
|
|
||||||
|
|
||||||
|
/*let url = args.next().expect("Usage: <rpc_url> <username> <password>");
|
||||||
|
let user = args.next().expect("no user given");
|
||||||
|
let pass = args.next().expect("no pass given");
|
||||||
|
*/
|
||||||
|
//let network = Network::Regtest
|
||||||
|
match get_client(network_params){
|
||||||
|
Ok((rpc,bcinfo)) => {
|
||||||
|
info!("connected");
|
||||||
|
//let best_block_hash = rpc.get_best_block_hash()?;
|
||||||
|
//info!("best block hash: {}", best_block_hash);
|
||||||
|
//let bestblockcount = rpc.get_block_count()?;
|
||||||
|
//info!("best block height: {}", bestblockcount);
|
||||||
|
//let best_block_hash_by_height = rpc.get_block_hash(bestblockcount)?;
|
||||||
|
//info!("best block hash by height: {}", best_block_hash_by_height);
|
||||||
|
//assert_eq!(best_block_hash_by_height, best_block_hash);
|
||||||
|
//let from_block= std::cmp::max(0, bestblockcount - 11);
|
||||||
|
//let mut time_sum:u64=0;
|
||||||
|
//for i in from_block..bestblockcount{
|
||||||
|
// let hash = rpc.get_block_hash(i).unwrap();
|
||||||
|
// let block: bitcoin::Block = rpc.get_by_id(&hash).unwrap();
|
||||||
|
// time_sum += <u32 as Into<u64>>::into(block.header.time);
|
||||||
|
//}
|
||||||
|
//let average_time = time_sum/11;
|
||||||
|
info!("median time: {}",bcinfo.median_time);
|
||||||
|
let average_time = bcinfo.median_time;
|
||||||
|
let db = sqlite::open(&cfg.db_file).unwrap();
|
||||||
|
|
||||||
|
let query_tx = db.prepare("SELECT * FROM tbl_tx WHERE network = :network AND status = :status AND ( locktime < :bestblock_height OR locktime > :locktime_threshold AND locktime < :bestblock_time);").unwrap().into_iter();
|
||||||
|
//let query_tx = db.prepare("SELECT * FROM tbl_tx where status = :status").unwrap().into_iter();
|
||||||
|
let mut pushed_txs:Vec<String> = Vec::new();
|
||||||
|
let mut invalid_txs:Vec<String> = Vec::new();
|
||||||
|
for row in query_tx.bind::<&[(_, Value)]>(&[
|
||||||
|
(":locktime_threshold", (LOCKTIME_THRESHOLD as i64).into()),
|
||||||
|
(":bestblock_time", (average_time as i64).into()),
|
||||||
|
(":bestblock_height", (bcinfo.blocks as i64).into()),
|
||||||
|
(":network", network_params.db_field.clone().into()),
|
||||||
|
(":status", 0.into()),
|
||||||
|
][..])
|
||||||
|
.unwrap()
|
||||||
|
.map(|row| row.unwrap())
|
||||||
|
{
|
||||||
|
let tx = row.read::<&str, _>("tx");
|
||||||
|
let txid = row.read::<&str, _>("txid");
|
||||||
|
let locktime = row.read::<i64,_>("locktime");
|
||||||
|
info!("to be pushed: {}: {}",txid, locktime);
|
||||||
|
match rpc.send_raw_transaction(tx){
|
||||||
|
Ok(o) => {
|
||||||
|
let mut file = OpenOptions::new()
|
||||||
|
.append(true) // Set the append option
|
||||||
|
.create(true) // Create the file if it doesn't exist
|
||||||
|
.open("valid_txs")?;
|
||||||
|
let data = format!("{}\t:\t{}\t:\t{}\n",txid,average_time,locktime);
|
||||||
|
file.write_all(data.as_bytes())?;
|
||||||
|
drop(file);
|
||||||
|
|
||||||
|
info!("tx: {} pusshata PUSHED\n{}",txid,o);
|
||||||
|
pushed_txs.push(txid.to_string());
|
||||||
|
},
|
||||||
|
Err(err) => {
|
||||||
|
let mut file = OpenOptions::new()
|
||||||
|
.append(true) // Set the append option
|
||||||
|
.create(true) // Create the file if it doesn't exist
|
||||||
|
.open("invalid_txs")?;
|
||||||
|
let data = format!("{}:\t{}\t:\t{}\t:\t{}\n",txid,err,average_time,locktime);
|
||||||
|
file.write_all(data.as_bytes())?;
|
||||||
|
drop(file);
|
||||||
|
warn!("Error: {}\n{}",err,txid);
|
||||||
|
invalid_txs.push(txid.to_string());
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if pushed_txs.len() > 0 {
|
||||||
|
let _ = db.execute(format!("UPDATE tbl_tx SET status = 1 WHERE txid in ('{}');",pushed_txs.join("','")));
|
||||||
|
}
|
||||||
|
if invalid_txs.len() > 0 {
|
||||||
|
let _ = db.execute(format!("UPDATE tbl_tx SET status = 2 WHERE txid in ('{}');",invalid_txs.join("','")));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(_)=>{
|
||||||
|
panic!("impossible to get client")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_env(cfg: &mut MyConfig){
|
||||||
|
match env::var("BAL_PUSHER_ZMQ_LISTENER") {
|
||||||
|
Ok(value) => {
|
||||||
|
cfg.zmq_listener = value;},
|
||||||
|
Err(_) => {},
|
||||||
|
}
|
||||||
|
match env::var("BAL_PUSHER_REQUEST_FILE") {
|
||||||
|
Ok(value) => {
|
||||||
|
cfg.requests_file = value;},
|
||||||
|
Err(_) => {},
|
||||||
|
}
|
||||||
|
match env::var("BAL_PUSHER_DB_FILE") {
|
||||||
|
Ok(value) => {
|
||||||
|
cfg.db_file = value;},
|
||||||
|
Err(_) => {},
|
||||||
|
}
|
||||||
|
match env::var("BAL_PUSHER_BITCOIN_DIR") {
|
||||||
|
Ok(value) => {
|
||||||
|
cfg.bitcoin_dir = value;},
|
||||||
|
Err(_) => {},
|
||||||
|
}
|
||||||
|
cfg.regtest = parse_env_netconfig(cfg,"regtest");
|
||||||
|
cfg.signet = parse_env_netconfig(cfg,"signet");
|
||||||
|
cfg.testnet = parse_env_netconfig(cfg,"testnet");
|
||||||
|
drop(parse_env_netconfig(cfg,"bitcoin"));
|
||||||
|
|
||||||
|
}
|
||||||
|
fn parse_env_netconfig(cfg_lock: &mut MyConfig, chain: &str) -> NetworkParams{
|
||||||
|
//fn parse_env_netconfig(cfg_lock: &MutexGuard<MyConfig>, chain: &str) -> &NetworkParams{
|
||||||
|
let cfg = match chain{
|
||||||
|
"regtest" => &mut cfg_lock.regtest,
|
||||||
|
"signet" => &mut cfg_lock.signet,
|
||||||
|
"testnet" => &mut cfg_lock.testnet,
|
||||||
|
&_ => &mut cfg_lock.mainnet,
|
||||||
|
};
|
||||||
|
match env::var(format!("BAL_PUSHER_{}_HOST",chain.to_uppercase())) {
|
||||||
|
Ok(value) => { cfg.host= value; },
|
||||||
|
Err(_) => {},
|
||||||
|
}
|
||||||
|
match env::var(format!("BAL_PUSHER_{}_PORT",chain.to_uppercase())) {
|
||||||
|
Ok(value) => {
|
||||||
|
match value.parse::<u64>(){
|
||||||
|
Ok(value) =>{ cfg.port = value.try_into().unwrap(); },
|
||||||
|
Err(_) => {},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(_) => {},
|
||||||
|
}
|
||||||
|
match env::var(format!("BAL_PUSHER_{}_DIR_PATH",chain.to_uppercase())) {
|
||||||
|
Ok(value) => { cfg.dir_path = value; },
|
||||||
|
Err(_) => {},
|
||||||
|
}
|
||||||
|
match env::var(format!("BAL_PUSHER_{}_DB_FIELD",chain.to_uppercase())) {
|
||||||
|
Ok(value) => { cfg.db_field = value; },
|
||||||
|
Err(_) => {},
|
||||||
|
}
|
||||||
|
match env::var(format!("BAL_PUSHER_{}_COOKIE_FILE",chain.to_uppercase())) {
|
||||||
|
Ok(value) => {
|
||||||
|
cfg.cookie_file = value; },
|
||||||
|
Err(_) => {},
|
||||||
|
}
|
||||||
|
match env::var(format!("BAL_PUSHER_{}_RPC_USER",chain.to_uppercase())) {
|
||||||
|
Ok(value) => { cfg.rpc_user = value; },
|
||||||
|
Err(_) => {},
|
||||||
|
}
|
||||||
|
match env::var(format!("BAL_PUSHER_{}_RPC_PASSWORD",chain.to_uppercase())) {
|
||||||
|
Ok(value) => { cfg.rpc_pass = value; },
|
||||||
|
Err(_) => {},
|
||||||
|
}
|
||||||
|
cfg.clone()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_default_config()-> MyConfig {
|
||||||
|
let file = confy::get_configuration_file_path("bal-pusher",None).expect("Error while getting path");
|
||||||
|
info!("Default configuration file path is: {:#?}", file);
|
||||||
|
confy::load("bal-pusher",None).expect("cant_load")
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main(){
|
||||||
|
env_logger::init();
|
||||||
|
let mut cfg: MyConfig = match env::var("BAL_PUSHER_CONFIG_FILE") {
|
||||||
|
Ok(value) => {
|
||||||
|
match confy::load_path(value.to_string()){
|
||||||
|
Ok(val) => {
|
||||||
|
info!("The configuration file path is: {:#?}", value);
|
||||||
|
val
|
||||||
|
},
|
||||||
|
Err(err) => {
|
||||||
|
error!("{}",err);
|
||||||
|
get_default_config()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Err(_) => {
|
||||||
|
get_default_config()
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
parse_env(&mut cfg);
|
||||||
|
let mut args = std::env::args();
|
||||||
|
let _exe_name = args.next().unwrap();
|
||||||
|
let arg_network = match args.next(){
|
||||||
|
Some(nargs) => nargs,
|
||||||
|
None => "bitcoin".to_string()
|
||||||
|
};
|
||||||
|
let network = match arg_network.as_str(){
|
||||||
|
|
||||||
|
"testnet" => Network::Testnet,
|
||||||
|
"signet" => Network::Signet,
|
||||||
|
"regtest" => Network::Regtest,
|
||||||
|
_ => Network::Bitcoin,
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
debug!("Network: {}",arg_network);
|
||||||
|
let network_params = get_network_params(&cfg,network);
|
||||||
|
|
||||||
|
|
||||||
|
let context = Context::new();
|
||||||
|
let socket: Socket = context.socket(zmq::SUB).unwrap();
|
||||||
|
|
||||||
|
let zmq_address = cfg.zmq_listener.clone();
|
||||||
|
socket.connect(&zmq_address).unwrap();
|
||||||
|
|
||||||
|
socket.set_subscribe(b"").unwrap();
|
||||||
|
|
||||||
|
let _ = main_result(&cfg,&network_params);
|
||||||
|
info!("waiting new blocks..");
|
||||||
|
let mut last_seq:Vec<u8>=[0;4].to_vec();
|
||||||
|
loop {
|
||||||
|
let message = socket.recv_multipart(0).unwrap();
|
||||||
|
let topic = message[0].clone();
|
||||||
|
let body = message[1].clone();
|
||||||
|
let seq = message[2].clone();
|
||||||
|
if last_seq >= seq {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
last_seq = seq;
|
||||||
|
//let mut sequence_str = "Unknown".to_string();
|
||||||
|
/*if seq.len()==4{
|
||||||
|
let mut rdr = Cursor::new(seq);
|
||||||
|
let sequence = rdr.read_u32::<LittleEndian>().expect("Failed to read integer");
|
||||||
|
sequence_str = sequence.to_string();
|
||||||
|
}*/
|
||||||
|
if topic == b"hashblock" {
|
||||||
|
info!("NEW BLOCK{}", hex::encode(body));
|
||||||
|
//let cfg = cfg.clone();
|
||||||
|
let _ = main_result(&cfg,&network_params);
|
||||||
|
}
|
||||||
|
thread::sleep(Duration::from_millis(100)); // Sleep for 100ms
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
x
Reference in New Issue
Block a user