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
 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
		Reference in New Issue
	
	Block a user
	 bitcoinafterlife
						bitcoinafterlife