2022-11-23 04:21:46 +00:00
// Gupax - GUI Uniting P2Pool And XMRig
//
// Copyright (c) 2022 hinto-janaiyo
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
2022-11-28 17:05:09 +00:00
2022-11-30 03:38:01 +00:00
// This file represents the "helper" thread, which is the full separate thread
// that runs alongside the main [App] GUI thread. It exists for the entire duration
// of Gupax so that things can be handled without locking up the GUI thread.
//
// This thread is a continual 1 second loop, collecting available jobs on the
// way down and (if possible) asynchronously executing them at the very end.
//
// The main GUI thread will interface with this thread by mutating the Arc<Mutex>'s
// found here, e.g: User clicks [Start P2Pool] -> Arc<Mutex<ProcessSignal> is set
// indicating to this thread during its loop: "I should start P2Pool!", e.g:
//
// match p2pool.lock().unwrap().signal {
// ProcessSignal::Start => start_p2pool(),
// ...
// }
//
// This also includes all things related to handling the child processes (P2Pool/XMRig):
// piping their stdout/stderr/stdin, accessing their APIs (HTTP + disk files), etc.
2022-11-28 17:05:09 +00:00
//---------------------------------------------------------------------------------------------------- Import
use std ::{
sync ::{ Arc , Mutex } ,
path ::PathBuf ,
2022-12-03 21:02:34 +00:00
process ::{ Command , Stdio } ,
fmt ::Write ,
2022-11-30 22:21:55 +00:00
time ::* ,
2022-11-28 17:05:09 +00:00
thread ,
} ;
2022-12-02 18:15:26 +00:00
use serde ::{ Serialize , Deserialize } ;
2022-11-28 17:05:09 +00:00
use crate ::constants ::* ;
2022-12-04 01:12:40 +00:00
use num_format ::{ Buffer , Locale } ;
2022-11-28 17:05:09 +00:00
use log ::* ;
2022-12-04 01:12:40 +00:00
//---------------------------------------------------------------------------------------------------- Constants
2022-12-07 03:01:36 +00:00
// The locale numbers are formatting in is English, which looks like: [1,000]
2022-12-04 01:12:40 +00:00
const LOCALE : num_format ::Locale = num_format ::Locale ::en ;
2022-12-07 03:01:36 +00:00
// The max amount of bytes of process output we are willing to
// hold in memory before it's too much and we need to reset.
const MAX_PROCESS_OUTPUT_BYTES : usize = 56_000_000 ;
// Just a little leeway so a reset will go off before the [String] allocates more memory.
const PROCESS_OUTPUT_LEEWAY : usize = MAX_PROCESS_OUTPUT_BYTES - 1000 ;
2022-12-02 18:15:26 +00:00
2022-11-30 22:21:55 +00:00
//---------------------------------------------------------------------------------------------------- [Helper] Struct
// A meta struct holding all the data that gets processed in this thread
pub struct Helper {
2022-12-03 21:02:34 +00:00
pub instant : Instant , // Gupax start as an [Instant]
pub human_time : HumanTime , // Gupax uptime formatting for humans
pub p2pool : Arc < Mutex < Process > > , // P2Pool process state
pub xmrig : Arc < Mutex < Process > > , // XMRig process state
2022-12-06 22:48:48 +00:00
pub gui_api_p2pool : Arc < Mutex < PubP2poolApi > > , // P2Pool API state (for GUI thread)
pub gui_api_xmrig : Arc < Mutex < PubXmrigApi > > , // XMRig API state (for GUI thread)
pub img_p2pool : Arc < Mutex < ImgP2pool > > , // A static "image" of the data P2Pool started with
pub img_xmrig : Arc < Mutex < ImgXmrig > > , // A static "image" of the data XMRig started with
pub_api_p2pool : Arc < Mutex < PubP2poolApi > > , // P2Pool API state (for Helper/P2Pool thread)
pub_api_xmrig : Arc < Mutex < PubXmrigApi > > , // XMRig API state (for Helper/XMRig thread)
2022-12-03 21:02:34 +00:00
priv_api_p2pool : Arc < Mutex < PrivP2poolApi > > , // For "watchdog" thread
priv_api_xmrig : Arc < Mutex < PrivXmrigApi > > , // For "watchdog" thread
2022-11-30 22:21:55 +00:00
}
2022-12-06 22:48:48 +00:00
// The communication between the data here and the GUI thread goes as follows:
// [GUI] <---> [Helper] <---> [Watchdog] <---> [Private Data only available here]
//
// Both [GUI] and [Helper] own their separate [Pub*Api] structs.
// Since P2Pool & XMRig will be updating their information out of sync,
// it's the helpers job to lock everything, and move the watchdog [Pub*Api]s
// on a 1-second interval into the [GUI]'s [Pub*Api] struct, atomically.
2022-11-28 17:05:09 +00:00
//---------------------------------------------------------------------------------------------------- [Process] Struct
// This holds all the state of a (child) process.
// The main GUI thread will use this to display console text, online state, etc.
pub struct Process {
2022-12-04 01:12:40 +00:00
pub name : ProcessName , // P2Pool or XMRig?
pub state : ProcessState , // The state of the process (alive, dead, etc)
pub signal : ProcessSignal , // Did the user click [Start/Stop/Restart]?
2022-11-28 17:05:09 +00:00
// STDIN Problem:
// - User can input many many commands in 1 second
// - The process loop only processes every 1 second
// - If there is only 1 [String] holding the user input,
// the user could overwrite their last input before
// the loop even has a chance to process their last command
// STDIN Solution:
// - When the user inputs something, push it to a [Vec]
// - In the process loop, loop over every [Vec] element and
// send each one individually to the process stdin
2022-12-05 19:55:50 +00:00
//
2022-12-04 01:12:40 +00:00
pub input : Vec < String > ,
2022-12-05 19:55:50 +00:00
// The below are the handles to the actual child process.
// [Simple] has no STDIN, but [Advanced] does. A PTY (pseudo-terminal) is
2022-12-06 03:33:35 +00:00
// required for P2Pool/XMRig to open their STDIN pipe.
child : Option < Arc < Mutex < Box < dyn portable_pty ::Child + Send + std ::marker ::Sync > > > > , // STDOUT/STDERR is combined automatically thanks to this PTY, nice
2022-12-05 19:55:50 +00:00
stdin : Option < Box < dyn portable_pty ::MasterPty + Send > > , // A handle to the process's MasterPTY/STDIN
// This is the process's private output [String], used by both [Simple] and [Advanced].
// The "watchdog" threads mutate this, the "helper" thread synchronizes the [Pub*Api] structs
// so that the data in here is cloned there roughly once a second. GUI thread never touches this.
2022-12-06 03:33:35 +00:00
output : Arc < Mutex < String > > ,
2022-12-06 20:17:37 +00:00
// Start time of process.
start : std ::time ::Instant ,
2022-11-28 17:05:09 +00:00
}
//---------------------------------------------------------------------------------------------------- [Process] Impl
impl Process {
pub fn new ( name : ProcessName , args : String , path : PathBuf ) -> Self {
Self {
name ,
2022-11-30 03:38:01 +00:00
state : ProcessState ::Dead ,
2022-11-28 17:05:09 +00:00
signal : ProcessSignal ::None ,
2022-12-06 20:17:37 +00:00
start : Instant ::now ( ) ,
2022-12-01 20:48:47 +00:00
stdin : Option ::None ,
2022-12-04 16:24:38 +00:00
child : Option ::None ,
2022-11-29 22:33:24 +00:00
// P2Pool log level 1 produces a bit less than 100,000 lines a day.
// Assuming each line averages 80 UTF-8 scalars (80 bytes), then this
// initial buffer should last around a week (56MB) before resetting.
2022-12-07 03:01:36 +00:00
output : Arc ::new ( Mutex ::new ( String ::with_capacity ( MAX_PROCESS_OUTPUT_BYTES ) ) ) ,
2022-11-28 17:05:09 +00:00
input : vec ! [ String ::new ( ) ] ,
}
}
// Borrow a [&str], return an owned split collection
pub fn parse_args ( args : & str ) -> Vec < String > {
args . split_whitespace ( ) . map ( | s | s . to_owned ( ) ) . collect ( )
}
2022-12-06 03:33:35 +00:00
// Convenience functions
pub fn is_alive ( & self ) -> bool {
self . state = = ProcessState ::Alive | | self . state = = ProcessState ::Middle
}
pub fn is_waiting ( & self ) -> bool {
self . state = = ProcessState ::Middle | | self . state = = ProcessState ::Waiting
}
2022-11-30 03:38:01 +00:00
}
2022-11-28 17:05:09 +00:00
2022-11-30 03:38:01 +00:00
//---------------------------------------------------------------------------------------------------- [Process*] Enum
#[ derive(Copy,Clone,Eq,PartialEq,Debug) ]
pub enum ProcessState {
2022-12-05 19:55:50 +00:00
Alive , // Process is online, GREEN!
Dead , // Process is dead, BLACK!
Failed , // Process is dead AND exited with a bad code, RED!
2022-12-06 03:33:35 +00:00
Middle , // Process is in the middle of something ([re]starting/stopping), YELLOW!
Waiting , // Process was successfully killed by a restart, and is ready to be started again, YELLOW!
2022-11-28 17:05:09 +00:00
}
#[ derive(Copy,Clone,Eq,PartialEq,Debug) ]
pub enum ProcessSignal {
None ,
2022-11-30 03:38:01 +00:00
Start ,
2022-11-28 17:05:09 +00:00
Stop ,
Restart ,
}
#[ derive(Copy,Clone,Eq,PartialEq,Debug) ]
pub enum ProcessName {
2022-11-30 22:21:55 +00:00
P2pool ,
Xmrig ,
2022-11-28 17:05:09 +00:00
}
2022-11-30 03:38:01 +00:00
impl std ::fmt ::Display for ProcessState { fn fmt ( & self , f : & mut std ::fmt ::Formatter ) -> std ::fmt ::Result { write! ( f , " {:#?} " , self ) } }
impl std ::fmt ::Display for ProcessSignal { fn fmt ( & self , f : & mut std ::fmt ::Formatter ) -> std ::fmt ::Result { write! ( f , " {:#?} " , self ) } }
impl std ::fmt ::Display for ProcessName { fn fmt ( & self , f : & mut std ::fmt ::Formatter ) -> std ::fmt ::Result { write! ( f , " {:#?} " , self ) } }
2022-12-05 19:55:50 +00:00
//---------------------------------------------------------------------------------------------------- [Helper]
use tokio ::io ::{ BufReader , AsyncBufReadExt } ;
impl Helper {
//---------------------------------------------------------------------------------------------------- General Functions
2022-12-06 22:48:48 +00:00
pub fn new ( instant : std ::time ::Instant , p2pool : Arc < Mutex < Process > > , xmrig : Arc < Mutex < Process > > , gui_api_p2pool : Arc < Mutex < PubP2poolApi > > , gui_api_xmrig : Arc < Mutex < PubXmrigApi > > , img_p2pool : Arc < Mutex < ImgP2pool > > , img_xmrig : Arc < Mutex < ImgXmrig > > ) -> Self {
2022-12-05 19:55:50 +00:00
Self {
instant ,
human_time : HumanTime ::into_human ( instant . elapsed ( ) ) ,
priv_api_p2pool : Arc ::new ( Mutex ::new ( PrivP2poolApi ::new ( ) ) ) ,
priv_api_xmrig : Arc ::new ( Mutex ::new ( PrivXmrigApi ::new ( ) ) ) ,
2022-12-06 22:48:48 +00:00
pub_api_p2pool : Arc ::new ( Mutex ::new ( PubP2poolApi ::new ( ) ) ) ,
pub_api_xmrig : Arc ::new ( Mutex ::new ( PubXmrigApi ::new ( ) ) ) ,
2022-12-05 19:55:50 +00:00
// These are created when initializing [App], since it needs a handle to it as well
2022-12-06 03:33:35 +00:00
p2pool ,
xmrig ,
2022-12-06 22:48:48 +00:00
gui_api_p2pool ,
gui_api_xmrig ,
img_p2pool ,
img_xmrig ,
2022-12-05 19:55:50 +00:00
}
}
// Reads a PTY which combines STDOUT/STDERR for me, yay
2022-12-06 03:33:35 +00:00
fn read_pty ( output : Arc < Mutex < String > > , reader : Box < dyn std ::io ::Read + Send > ) {
2022-12-05 19:55:50 +00:00
use std ::io ::BufRead ;
let mut stdout = std ::io ::BufReader ::new ( reader ) . lines ( ) ;
while let Some ( Ok ( line ) ) = stdout . next ( ) {
2022-12-06 03:33:35 +00:00
// println!("{}", line); // For debugging.
writeln! ( output . lock ( ) . unwrap ( ) , " {} " , line ) ;
2022-12-05 19:55:50 +00:00
}
}
2022-12-07 02:13:37 +00:00
// Reset output if larger than 55_999_000 bytes (around 1 week of logs).
// The actual [String] holds 56_000_000, but this allows for some leeway so it doesn't allocate more memory.
// This will also append a message showing it was reset.
fn check_reset_output ( output : & Arc < Mutex < String > > , name : ProcessName ) {
let mut output = output . lock ( ) . unwrap ( ) ;
2022-12-07 03:01:36 +00:00
if output . len ( ) > PROCESS_OUTPUT_LEEWAY {
2022-12-07 02:13:37 +00:00
let name = match name {
ProcessName ::P2pool = > " P2Pool " ,
ProcessName ::Xmrig = > " XMRig " ,
} ;
info! ( " {} | Output is nearing 56,000,000 bytes, resetting! " , name ) ;
let text = format! ( " {} \n {} logs are exceeding the maximum: 56,000,000 bytes! \n I've reset the logs for you, your stats may now be inaccurate since they depend on these logs! \n I think you rather have that than have it hogging your memory, though! \n {} " , HORI_CONSOLE , name , HORI_CONSOLE ) ;
output . clear ( ) ;
output . push_str ( & text ) ;
}
}
2022-12-05 19:55:50 +00:00
//---------------------------------------------------------------------------------------------------- P2Pool specific
2022-12-06 03:33:35 +00:00
// Read P2Pool's API file.
fn read_p2pool_api ( path : & std ::path ::PathBuf ) -> Result < String , std ::io ::Error > {
match std ::fs ::read_to_string ( path ) {
Ok ( s ) = > Ok ( s ) ,
Err ( e ) = > { warn! ( " P2Pool API | [{}] read error: {} " , path . display ( ) , e ) ; Err ( e ) } ,
}
}
// Deserialize the above [String] into a [PrivP2poolApi]
fn str_to_priv_p2pool_api ( string : & str ) -> Result < PrivP2poolApi , serde_json ::Error > {
match serde_json ::from_str ::< PrivP2poolApi > ( string ) {
Ok ( a ) = > Ok ( a ) ,
Err ( e ) = > { warn! ( " P2Pool API | Could not deserialize API data: {} " , e ) ; Err ( e ) } ,
}
}
// Just sets some signals for the watchdog thread to pick up on.
pub fn stop_p2pool ( helper : & Arc < Mutex < Self > > ) {
2022-12-07 02:33:24 +00:00
info! ( " P2Pool | Attempting to stop... " ) ;
2022-12-06 03:33:35 +00:00
helper . lock ( ) . unwrap ( ) . p2pool . lock ( ) . unwrap ( ) . signal = ProcessSignal ::Stop ;
helper . lock ( ) . unwrap ( ) . p2pool . lock ( ) . unwrap ( ) . state = ProcessState ::Middle ;
}
// The "restart frontend" to a "frontend" function.
// Basically calls to kill the current p2pool, waits a little, then starts the below function in a a new thread, then exit.
2022-12-06 22:48:48 +00:00
pub fn restart_p2pool ( helper : & Arc < Mutex < Self > > , state : & crate ::disk ::P2pool , path : & std ::path ::PathBuf ) {
2022-12-07 02:33:24 +00:00
info! ( " P2Pool | Attempting to restart... " ) ;
2022-12-06 03:33:35 +00:00
helper . lock ( ) . unwrap ( ) . p2pool . lock ( ) . unwrap ( ) . signal = ProcessSignal ::Restart ;
helper . lock ( ) . unwrap ( ) . p2pool . lock ( ) . unwrap ( ) . state = ProcessState ::Middle ;
let helper = Arc ::clone ( & helper ) ;
let state = state . clone ( ) ;
let path = path . clone ( ) ;
// This thread lives to wait, start p2pool then die.
thread ::spawn ( move | | {
while helper . lock ( ) . unwrap ( ) . p2pool . lock ( ) . unwrap ( ) . is_alive ( ) {
2022-12-07 02:33:24 +00:00
warn! ( " P2Pool | Want to restart but process is still alive, waiting... " ) ;
2022-12-06 03:33:35 +00:00
thread ::sleep ( SECOND ) ;
}
// Ok, process is not alive, start the new one!
2022-12-06 22:48:48 +00:00
Self ::start_p2pool ( & helper , & state , & path ) ;
2022-12-06 03:33:35 +00:00
} ) ;
info! ( " P2Pool | Restart ... OK " ) ;
}
// The "frontend" function that parses the arguments, and spawns either the [Simple] or [Advanced] P2Pool watchdog thread.
2022-12-06 22:48:48 +00:00
pub fn start_p2pool ( helper : & Arc < Mutex < Self > > , state : & crate ::disk ::P2pool , path : & std ::path ::PathBuf ) {
2022-12-06 03:33:35 +00:00
helper . lock ( ) . unwrap ( ) . p2pool . lock ( ) . unwrap ( ) . state = ProcessState ::Middle ;
2022-12-06 22:48:48 +00:00
let args = Self ::build_p2pool_args_and_mutate_img ( helper , state , path ) ;
// Print arguments & user settings to console
2022-12-07 02:33:24 +00:00
crate ::disk ::print_dash ( & format! ( " P2Pool | Launch arguments: {:#?} " , args ) ) ;
2022-12-06 22:48:48 +00:00
// Spawn watchdog thread
let process = Arc ::clone ( & helper . lock ( ) . unwrap ( ) . p2pool ) ;
2022-12-07 02:33:24 +00:00
let gui_api = Arc ::clone ( & helper . lock ( ) . unwrap ( ) . gui_api_p2pool ) ;
2022-12-06 22:48:48 +00:00
let pub_api = Arc ::clone ( & helper . lock ( ) . unwrap ( ) . pub_api_p2pool ) ;
let priv_api = Arc ::clone ( & helper . lock ( ) . unwrap ( ) . priv_api_p2pool ) ;
let path = path . clone ( ) ;
thread ::spawn ( move | | {
2022-12-07 02:33:24 +00:00
Self ::spawn_p2pool_watchdog ( process , gui_api , pub_api , priv_api , args , path ) ;
2022-12-06 22:48:48 +00:00
} ) ;
}
// Takes in some [State/P2pool] and parses it to build the actual command arguments.
// Returns the [Vec] of actual arguments, and mutates the [ImgP2pool] for the main GUI thread
// It returns a value... and mutates a deeply nested passed argument... this is some pretty bad code...
pub fn build_p2pool_args_and_mutate_img ( helper : & Arc < Mutex < Self > > , state : & crate ::disk ::P2pool , path : & std ::path ::PathBuf ) -> Vec < String > {
2022-12-05 19:55:50 +00:00
let mut args = Vec ::with_capacity ( 500 ) ;
let path = path . clone ( ) ;
let mut api_path = path . clone ( ) ;
api_path . pop ( ) ;
// [Simple]
if state . simple {
// Build the p2pool argument
let ( ip , rpc , zmq ) = crate ::node ::enum_to_ip_rpc_zmq_tuple ( state . node ) ; // Get: (IP, RPC, ZMQ)
2022-12-06 20:33:57 +00:00
args . push ( " --wallet " . to_string ( ) ) ; args . push ( state . address . clone ( ) ) ; // Wallet address
args . push ( " --host " . to_string ( ) ) ; args . push ( ip . to_string ( ) ) ; // IP Address
2022-12-05 19:55:50 +00:00
args . push ( " --rpc-port " . to_string ( ) ) ; args . push ( rpc . to_string ( ) ) ; // RPC Port
args . push ( " --zmq-port " . to_string ( ) ) ; args . push ( zmq . to_string ( ) ) ; // ZMQ Port
args . push ( " --data-api " . to_string ( ) ) ; args . push ( api_path . display ( ) . to_string ( ) ) ; // API Path
args . push ( " --local-api " . to_string ( ) ) ; // Enable API
args . push ( " --no-color " . to_string ( ) ) ; // Remove color escape sequences, Gupax terminal can't parse it :(
args . push ( " --mini " . to_string ( ) ) ; // P2Pool Mini
2022-12-06 22:48:48 +00:00
* helper . lock ( ) . unwrap ( ) . img_p2pool . lock ( ) . unwrap ( ) = ImgP2pool {
2022-12-06 20:54:18 +00:00
mini : true ,
address : state . address . clone ( ) ,
host : ip . to_string ( ) ,
rpc : rpc . to_string ( ) ,
zmq : zmq . to_string ( ) ,
log_level : " 3 " . to_string ( ) ,
out_peers : " 10 " . to_string ( ) ,
in_peers : " 10 " . to_string ( ) ,
} ;
2022-12-05 19:55:50 +00:00
// [Advanced]
} else {
// Overriding command arguments
if ! state . arguments . is_empty ( ) {
2022-12-06 22:48:48 +00:00
// This parses the input and attemps to fill out
// the [ImgP2pool]... This is pretty bad code...
let mut last = " " ;
let lock = helper . lock ( ) . unwrap ( ) ;
let mut p2pool_image = lock . img_p2pool . lock ( ) . unwrap ( ) ;
2022-12-05 19:55:50 +00:00
for arg in state . arguments . split_whitespace ( ) {
2022-12-06 22:48:48 +00:00
match last {
" --mini " = > p2pool_image . mini = true ,
" --wallet " = > p2pool_image . address = arg . to_string ( ) ,
" --host " = > p2pool_image . host = arg . to_string ( ) ,
" --rpc-port " = > p2pool_image . rpc = arg . to_string ( ) ,
" --zmq-port " = > p2pool_image . zmq = arg . to_string ( ) ,
" --loglevel " = > p2pool_image . log_level = arg . to_string ( ) ,
" --out-peers " = > p2pool_image . out_peers = arg . to_string ( ) ,
" --in-peers " = > p2pool_image . in_peers = arg . to_string ( ) ,
_ = > ( ) ,
}
2022-12-05 19:55:50 +00:00
args . push ( arg . to_string ( ) ) ;
2022-12-06 22:48:48 +00:00
last = arg ;
2022-12-05 19:55:50 +00:00
}
// Else, build the argument
} else {
2022-12-06 20:33:57 +00:00
args . push ( " --wallet " . to_string ( ) ) ; args . push ( state . address . clone ( ) ) ; // Wallet
args . push ( " --host " . to_string ( ) ) ; args . push ( state . selected_ip . to_string ( ) ) ; // IP
args . push ( " --rpc-port " . to_string ( ) ) ; args . push ( state . selected_rpc . to_string ( ) ) ; // RPC
args . push ( " --zmq-port " . to_string ( ) ) ; args . push ( state . selected_zmq . to_string ( ) ) ; // ZMQ
args . push ( " --loglevel " . to_string ( ) ) ; args . push ( state . log_level . to_string ( ) ) ; // Log Level
args . push ( " --out-peers " . to_string ( ) ) ; args . push ( state . out_peers . to_string ( ) ) ; // Out Peers
args . push ( " --in-peers " . to_string ( ) ) ; args . push ( state . in_peers . to_string ( ) ) ; // In Peers
args . push ( " --data-api " . to_string ( ) ) ; args . push ( api_path . display ( ) . to_string ( ) ) ; // API Path
args . push ( " --local-api " . to_string ( ) ) ; // Enable API
args . push ( " --no-color " . to_string ( ) ) ; // Remove color escape sequences
if state . mini { args . push ( " --mini " . to_string ( ) ) ; } ; // Mini
2022-12-06 22:48:48 +00:00
* helper . lock ( ) . unwrap ( ) . img_p2pool . lock ( ) . unwrap ( ) = ImgP2pool {
2022-12-06 20:54:18 +00:00
mini : state . mini ,
address : state . address . clone ( ) ,
host : state . selected_ip . to_string ( ) ,
rpc : state . selected_rpc . to_string ( ) ,
zmq : state . selected_zmq . to_string ( ) ,
log_level : state . log_level . to_string ( ) ,
out_peers : state . out_peers . to_string ( ) ,
in_peers : state . in_peers . to_string ( ) ,
2022-12-06 22:48:48 +00:00
}
2022-12-05 19:55:50 +00:00
}
}
2022-12-06 22:48:48 +00:00
args
2022-12-05 19:55:50 +00:00
}
2022-12-06 03:33:35 +00:00
// The P2Pool watchdog. Spawns 1 OS thread for reading a PTY (STDOUT+STDERR), and combines the [Child] with a PTY so STDIN actually works.
2022-12-05 19:55:50 +00:00
#[ tokio::main ]
2022-12-07 02:33:24 +00:00
async fn spawn_p2pool_watchdog ( process : Arc < Mutex < Process > > , gui_api : Arc < Mutex < PubP2poolApi > > , pub_api : Arc < Mutex < PubP2poolApi > > , priv_api : Arc < Mutex < PrivP2poolApi > > , args : Vec < String > , mut path : std ::path ::PathBuf ) {
2022-12-05 19:55:50 +00:00
// 1a. Create PTY
let pty = portable_pty ::native_pty_system ( ) ;
2022-12-06 03:33:35 +00:00
let pair = pty . openpty ( portable_pty ::PtySize {
2022-12-05 19:55:50 +00:00
rows : 24 ,
cols : 80 ,
pixel_width : 0 ,
pixel_height : 0 ,
} ) . unwrap ( ) ;
// 1b. Create command
2022-12-06 03:33:35 +00:00
let mut cmd = portable_pty ::CommandBuilder ::new ( path . as_path ( ) ) ;
2022-12-05 19:55:50 +00:00
cmd . args ( args ) ;
2022-12-06 03:33:35 +00:00
cmd . cwd ( path . as_path ( ) . parent ( ) . unwrap ( ) ) ;
2022-12-05 19:55:50 +00:00
// 1c. Create child
let child_pty = Arc ::new ( Mutex ::new ( pair . slave . spawn_command ( cmd ) . unwrap ( ) ) ) ;
// 2. Set process state
let mut lock = process . lock ( ) . unwrap ( ) ;
lock . state = ProcessState ::Alive ;
lock . signal = ProcessSignal ::None ;
lock . start = Instant ::now ( ) ;
2022-12-06 03:33:35 +00:00
lock . child = Some ( Arc ::clone ( & child_pty ) ) ;
2022-12-05 19:55:50 +00:00
let reader = pair . master . try_clone_reader ( ) . unwrap ( ) ; // Get STDOUT/STDERR before moving the PTY
lock . stdin = Some ( pair . master ) ;
drop ( lock ) ;
// 3. Spawn PTY read thread
2022-12-06 03:33:35 +00:00
let output_clone = Arc ::clone ( & process . lock ( ) . unwrap ( ) . output ) ;
2022-12-05 19:55:50 +00:00
thread ::spawn ( move | | {
2022-12-06 03:33:35 +00:00
Self ::read_pty ( output_clone , reader ) ;
2022-12-05 19:55:50 +00:00
} ) ;
2022-12-06 03:33:35 +00:00
path . pop ( ) ;
path . push ( P2POOL_API_PATH ) ;
let regex = P2poolRegex ::new ( ) ;
let output = Arc ::clone ( & process . lock ( ) . unwrap ( ) . output ) ;
2022-12-06 20:17:37 +00:00
let start = process . lock ( ) . unwrap ( ) . start ;
2022-12-06 03:33:35 +00:00
2022-12-05 19:55:50 +00:00
// 4. Loop as watchdog
loop {
// Set timer
let now = Instant ::now ( ) ;
// Check SIGNAL
if process . lock ( ) . unwrap ( ) . signal = = ProcessSignal ::Stop {
child_pty . lock ( ) . unwrap ( ) . kill ( ) ; // This actually sends a SIGHUP to p2pool (closes the PTY, hangs up on p2pool)
// Wait to get the exit status
2022-12-06 03:33:35 +00:00
let exit_status = match child_pty . lock ( ) . unwrap ( ) . wait ( ) {
2022-12-06 20:17:37 +00:00
Ok ( e ) = > {
if e . success ( ) {
process . lock ( ) . unwrap ( ) . state = ProcessState ::Dead ; " Successful "
} else {
process . lock ( ) . unwrap ( ) . state = ProcessState ::Failed ; " Failed "
}
} ,
_ = > { process . lock ( ) . unwrap ( ) . state = ProcessState ::Failed ; " Unknown Error " } ,
2022-12-06 03:33:35 +00:00
} ;
2022-12-06 20:17:37 +00:00
let uptime = HumanTime ::into_human ( start . elapsed ( ) ) ;
2022-12-06 03:33:35 +00:00
info! ( " P2Pool | Stopped ... Uptime was: [{}], Exit status: [{}] " , uptime , exit_status ) ;
2022-12-07 02:33:24 +00:00
// This is written directly into the GUI API, because sometimes the 900ms event loop can't catch it.
writeln! ( gui_api . lock ( ) . unwrap ( ) . output , " {} \n P2Pool stopped | Uptime: [{}] | Exit status: [{}] \n {} \n \n " , HORI_CONSOLE , uptime , exit_status , HORI_CONSOLE ) ;
2022-12-06 20:17:37 +00:00
process . lock ( ) . unwrap ( ) . signal = ProcessSignal ::None ;
2022-12-06 03:33:35 +00:00
break
2022-12-07 02:13:37 +00:00
// Check RESTART
2022-12-06 03:33:35 +00:00
} else if process . lock ( ) . unwrap ( ) . signal = = ProcessSignal ::Restart {
child_pty . lock ( ) . unwrap ( ) . kill ( ) ; // This actually sends a SIGHUP to p2pool (closes the PTY, hangs up on p2pool)
// Wait to get the exit status
2022-12-05 19:55:50 +00:00
let exit_status = match child_pty . lock ( ) . unwrap ( ) . wait ( ) {
Ok ( e ) = > if e . success ( ) { " Successful " } else { " Failed " } ,
_ = > " Unknown Error " ,
} ;
2022-12-06 20:17:37 +00:00
let uptime = HumanTime ::into_human ( start . elapsed ( ) ) ;
2022-12-06 03:33:35 +00:00
info! ( " P2Pool | Stopped ... Uptime was: [{}], Exit status: [{}] " , uptime , exit_status ) ;
2022-12-07 02:33:24 +00:00
// This is written directly into the GUI API, because sometimes the 900ms event loop can't catch it.
writeln! ( gui_api . lock ( ) . unwrap ( ) . output , " {} \n P2Pool stopped | Uptime: [{}] | Exit status: [{}] \n {} \n \n " , HORI_CONSOLE , uptime , exit_status , HORI_CONSOLE ) ;
2022-12-06 20:17:37 +00:00
process . lock ( ) . unwrap ( ) . state = ProcessState ::Waiting ;
2022-12-06 03:33:35 +00:00
break
// Check if the process is secretly died without us knowing :)
} else if let Ok ( Some ( code ) ) = child_pty . lock ( ) . unwrap ( ) . try_wait ( ) {
let exit_status = match code . success ( ) {
2022-12-06 20:17:37 +00:00
true = > { process . lock ( ) . unwrap ( ) . state = ProcessState ::Dead ; " Successful " } ,
false = > { process . lock ( ) . unwrap ( ) . state = ProcessState ::Failed ; " Failed " } ,
2022-12-06 03:33:35 +00:00
} ;
2022-12-06 20:17:37 +00:00
let uptime = HumanTime ::into_human ( start . elapsed ( ) ) ;
2022-12-05 19:55:50 +00:00
info! ( " P2Pool | Stopped ... Uptime was: [{}], Exit status: [{}] " , uptime , exit_status ) ;
2022-12-07 02:33:24 +00:00
// This is written directly into the GUI, because sometimes the 900ms event loop can't catch it.
writeln! ( gui_api . lock ( ) . unwrap ( ) . output , " {} \n P2Pool stopped | Uptime: [{}] | Exit status: [{}] \n {} \n \n " , HORI_CONSOLE , uptime , exit_status , HORI_CONSOLE ) ;
2022-12-06 20:17:37 +00:00
process . lock ( ) . unwrap ( ) . signal = ProcessSignal ::None ;
2022-12-05 19:55:50 +00:00
break
}
// Check vector of user input
let mut lock = process . lock ( ) . unwrap ( ) ;
if ! lock . input . is_empty ( ) {
let input = std ::mem ::take ( & mut lock . input ) ;
for line in input {
writeln! ( lock . stdin . as_mut ( ) . unwrap ( ) , " {} " , line ) ;
}
}
drop ( lock ) ;
2022-12-06 18:18:40 +00:00
// Always update from output
2022-12-06 20:17:37 +00:00
PubP2poolApi ::update_from_output ( & pub_api , & output , start . elapsed ( ) , & regex ) ;
2022-12-06 18:18:40 +00:00
2022-12-06 03:33:35 +00:00
// Read API file into string
if let Ok ( string ) = Self ::read_p2pool_api ( & path ) {
// Deserialize
if let Ok ( s ) = Self ::str_to_priv_p2pool_api ( & string ) {
// Update the structs.
2022-12-06 18:18:40 +00:00
PubP2poolApi ::update_from_priv ( & pub_api , & priv_api ) ;
2022-12-06 03:33:35 +00:00
}
}
2022-12-07 02:13:37 +00:00
// Check if logs need resetting
Self ::check_reset_output ( & output , ProcessName ::P2pool ) ;
2022-12-05 19:55:50 +00:00
// Sleep (only if 900ms hasn't passed)
let elapsed = now . elapsed ( ) . as_millis ( ) ;
// Since logic goes off if less than 1000, casting should be safe
if elapsed < 900 { std ::thread ::sleep ( std ::time ::Duration ::from_millis ( ( 900 - elapsed ) as u64 ) ) ; }
}
// 5. If loop broke, we must be done here.
2022-12-06 20:54:18 +00:00
info! ( " P2Pool | Watchdog thread exiting... Goodbye! " ) ;
2022-12-05 19:55:50 +00:00
}
//---------------------------------------------------------------------------------------------------- XMRig specific
// Intermediate function that parses the arguments, and spawns the XMRig watchdog thread.
2022-12-06 03:33:35 +00:00
pub fn spawn_xmrig ( helper : & Arc < Mutex < Self > > , state : & crate ::disk ::Xmrig , path : std ::path ::PathBuf ) {
2022-12-05 19:55:50 +00:00
let mut args = Vec ::with_capacity ( 500 ) ;
if state . simple {
let rig_name = if state . simple_rig . is_empty ( ) { GUPAX_VERSION . to_string ( ) } else { state . simple_rig . clone ( ) } ; // Rig name
args . push ( format! ( " --threads {} " , state . current_threads ) ) ; // Threads
args . push ( format! ( " --user {} " , state . simple_rig ) ) ; // Rig name
args . push ( format! ( " --url 127.0.0.1:3333 " ) ) ; // Local P2Pool (the default)
args . push ( " --no-color " . to_string ( ) ) ; // No color escape codes
if state . pause ! = 0 { args . push ( format! ( " --pause-on-active {} " , state . pause ) ) ; } // Pause on active
} else {
if ! state . arguments . is_empty ( ) {
for arg in state . arguments . split_whitespace ( ) {
args . push ( arg . to_string ( ) ) ;
}
} else {
args . push ( format! ( " --user {} " , state . address . clone ( ) ) ) ; // Wallet
args . push ( format! ( " --threads {} " , state . current_threads ) ) ; // Threads
args . push ( format! ( " --rig-id {} " , state . selected_rig ) ) ; // Rig ID
args . push ( format! ( " --url {} : {} " , state . selected_ip . clone ( ) , state . selected_port . clone ( ) ) ) ; // IP/Port
args . push ( format! ( " --http-host {} " , state . api_ip ) . to_string ( ) ) ; // HTTP API IP
args . push ( format! ( " --http-port {} " , state . api_port ) . to_string ( ) ) ; // HTTP API Port
args . push ( " --no-color " . to_string ( ) ) ; // No color escape codes
if state . tls { args . push ( " --tls " . to_string ( ) ) ; } // TLS
if state . keepalive { args . push ( " --keepalive " . to_string ( ) ) ; } // Keepalive
if state . pause ! = 0 { args . push ( format! ( " --pause-on-active {} " , state . pause ) ) ; } // Pause on active
}
}
// Print arguments to console
crate ::disk ::print_dash ( & format! ( " XMRig | Launch arguments ... {:#?} " , args ) ) ;
// Spawn watchdog thread
thread ::spawn ( move | | {
Self ::spawn_xmrig_watchdog ( args ) ;
} ) ;
}
// The actual XMRig watchdog tokio runtime.
#[ tokio::main ]
pub async fn spawn_xmrig_watchdog ( args : Vec < String > ) {
}
//---------------------------------------------------------------------------------------------------- The "helper"
2022-12-06 22:48:48 +00:00
// The "helper" thread. Syncs data between threads here and the GUI.
2022-12-05 19:55:50 +00:00
pub fn spawn_helper ( helper : & Arc < Mutex < Self > > ) {
2022-12-06 22:48:48 +00:00
let mut helper = Arc ::clone ( helper ) ;
thread ::spawn ( move | | {
2022-12-07 02:13:37 +00:00
info! ( " Helper | Hello from helper thread! Entering loop where I will spend the rest of my days... " ) ;
2022-12-05 19:55:50 +00:00
// Begin loop
loop {
2022-12-06 22:48:48 +00:00
// 1. Loop init timestamp
2022-12-05 19:55:50 +00:00
let start = Instant ::now ( ) ;
2022-12-06 22:48:48 +00:00
// 2. Lock... EVERYTHING!
let mut lock = helper . lock ( ) . unwrap ( ) ;
let mut gui_api_p2pool = lock . gui_api_p2pool . lock ( ) . unwrap ( ) ;
let mut gui_api_xmrig = lock . gui_api_xmrig . lock ( ) . unwrap ( ) ;
let mut pub_api_p2pool = lock . pub_api_p2pool . lock ( ) . unwrap ( ) ;
let mut pub_api_xmrig = lock . pub_api_xmrig . lock ( ) . unwrap ( ) ;
let p2pool = lock . p2pool . lock ( ) . unwrap ( ) ;
let xmrig = lock . xmrig . lock ( ) . unwrap ( ) ;
// Calculate Gupax's uptime always.
let human_time = HumanTime ::into_human ( lock . instant . elapsed ( ) ) ;
// If both [P2Pool/XMRig] are alive...
if p2pool . is_alive ( ) & & xmrig . is_alive ( ) {
* gui_api_p2pool = std ::mem ::take ( & mut pub_api_p2pool ) ;
* gui_api_xmrig = std ::mem ::take ( & mut pub_api_xmrig ) ;
// If only [P2Pool] is alive...
} else if p2pool . is_alive ( ) {
* gui_api_p2pool = std ::mem ::take ( & mut pub_api_p2pool ) ;
// If only [XMRig] is alive...
} else if xmrig . is_alive ( ) {
* gui_api_xmrig = std ::mem ::take ( & mut pub_api_xmrig ) ;
}
// 2. Drop... (almost) EVERYTHING... IN REVERSE!
drop ( xmrig ) ;
drop ( p2pool ) ;
drop ( pub_api_xmrig ) ;
drop ( pub_api_p2pool ) ;
drop ( gui_api_xmrig ) ;
drop ( gui_api_p2pool ) ;
// Update the time... then drop :)
lock . human_time = human_time ;
drop ( lock ) ;
2022-12-05 19:55:50 +00:00
2022-12-06 22:48:48 +00:00
// 3. Calculate if we should sleep or not.
2022-12-05 19:55:50 +00:00
// If we should sleep, how long?
let elapsed = start . elapsed ( ) . as_millis ( ) ;
if elapsed < 1000 {
// Casting from u128 to u64 should be safe here, because [elapsed]
// is less than 1000, meaning it can fit into a u64 easy.
std ::thread ::sleep ( std ::time ::Duration ::from_millis ( ( 1000 - elapsed ) as u64 ) ) ;
}
2022-12-06 22:48:48 +00:00
// 4. End loop
2022-12-05 19:55:50 +00:00
}
2022-12-06 22:48:48 +00:00
// 5. Something has gone terribly wrong if the helper exited this loop.
let text = " HELPER THREAD HAS ESCAPED THE LOOP...! " ;
error! ( " {} " , text ) ; error! ( " {} " , text ) ; error! ( " {} " , text ) ; panic! ( " {} " , text ) ;
} ) ;
2022-12-05 19:55:50 +00:00
}
}
2022-11-30 22:21:55 +00:00
//---------------------------------------------------------------------------------------------------- [HumanTime]
// This converts a [std::time::Duration] into something more readable.
// Used for uptime display purposes: [7 years, 8 months, 15 days, 23 hours, 35 minutes, 1 second]
// Code taken from [https://docs.rs/humantime/] and edited to remove sub-second time, change spacing and some words.
use std ::time ::Duration ;
#[ derive(Debug, Clone) ]
pub struct HumanTime ( Duration ) ;
impl HumanTime {
2022-12-06 20:17:37 +00:00
pub fn new ( ) -> HumanTime {
HumanTime ( ZERO_SECONDS )
}
2022-11-30 22:21:55 +00:00
pub fn into_human ( d : Duration ) -> HumanTime {
HumanTime ( d )
}
fn plural ( f : & mut std ::fmt ::Formatter , started : & mut bool , name : & str , value : u64 ) -> std ::fmt ::Result {
if value > 0 {
2022-12-06 20:17:37 +00:00
if * started {
f . write_str ( " , " ) ? ;
}
write! ( f , " {} {} " , value , name ) ? ;
if value > 1 {
f . write_str ( " s " ) ? ;
}
* started = true ;
2022-11-30 22:21:55 +00:00
}
Ok ( ( ) )
}
}
impl std ::fmt ::Display for HumanTime {
fn fmt ( & self , f : & mut std ::fmt ::Formatter ) -> std ::fmt ::Result {
let secs = self . 0. as_secs ( ) ;
if secs = = 0 {
2022-12-06 20:17:37 +00:00
f . write_str ( " 0 seconds " ) ? ;
2022-11-30 22:21:55 +00:00
return Ok ( ( ) ) ;
}
let years = secs / 31_557_600 ; // 365.25d
let ydays = secs % 31_557_600 ;
let months = ydays / 2_630_016 ; // 30.44d
let mdays = ydays % 2_630_016 ;
let days = mdays / 86400 ;
let day_secs = mdays % 86400 ;
let hours = day_secs / 3600 ;
let minutes = day_secs % 3600 / 60 ;
let seconds = day_secs % 60 ;
let ref mut started = false ;
2022-12-06 20:17:37 +00:00
Self ::plural ( f , started , " year " , years ) ? ;
Self ::plural ( f , started , " month " , months ) ? ;
Self ::plural ( f , started , " day " , days ) ? ;
Self ::plural ( f , started , " hour " , hours ) ? ;
Self ::plural ( f , started , " minute " , minutes ) ? ;
Self ::plural ( f , started , " second " , seconds ) ? ;
2022-11-30 22:21:55 +00:00
Ok ( ( ) )
}
}
2022-12-04 01:12:40 +00:00
//---------------------------------------------------------------------------------------------------- [HumanNumber]
// Human readable numbers.
// Float | [1234.57] -> [1,234] | Casts as u64/u128, adds comma
// Unsigned | [1234567] -> [1,234,567] | Adds comma
// Percent | [99.123] -> [99.12%] | Truncates to 2 after dot, adds percent
// Percent | [0.001] -> [0%] | Rounds down, removes redundant zeros
// Hashrate | [123.0, 311.2, null] -> [123, 311, ???] | Casts, replaces null with [???]
// CPU Load | [12.0, 11.4, null] -> [12.0, 11.4, ???] | No change, just into [String] form
#[ derive(Debug, Clone) ]
pub struct HumanNumber ( String ) ;
impl std ::fmt ::Display for HumanNumber {
fn fmt ( & self , f : & mut std ::fmt ::Formatter ) -> std ::fmt ::Result {
write! ( f , " {} " , & self . 0 )
}
}
impl HumanNumber {
fn unknown ( ) -> Self {
Self ( " ??? " . to_string ( ) )
}
fn to_percent ( f : f32 ) -> Self {
if f < 0.01 {
Self ( " 0% " . to_string ( ) )
} else {
Self ( format! ( " {:.2} % " , f ) )
}
}
fn from_f32 ( f : f32 ) -> Self {
let mut buf = num_format ::Buffer ::new ( ) ;
buf . write_formatted ( & ( f as u64 ) , & LOCALE ) ;
Self ( buf . as_str ( ) . to_string ( ) )
}
fn from_f64 ( f : f64 ) -> Self {
let mut buf = num_format ::Buffer ::new ( ) ;
buf . write_formatted ( & ( f as u128 ) , & LOCALE ) ;
Self ( buf . as_str ( ) . to_string ( ) )
}
fn from_u8 ( u : u8 ) -> Self {
let mut buf = num_format ::Buffer ::new ( ) ;
buf . write_formatted ( & u , & LOCALE ) ;
Self ( buf . as_str ( ) . to_string ( ) )
}
fn from_u16 ( u : u16 ) -> Self {
let mut buf = num_format ::Buffer ::new ( ) ;
buf . write_formatted ( & u , & LOCALE ) ;
Self ( buf . as_str ( ) . to_string ( ) )
}
fn from_u32 ( u : u32 ) -> Self {
let mut buf = num_format ::Buffer ::new ( ) ;
buf . write_formatted ( & u , & LOCALE ) ;
Self ( buf . as_str ( ) . to_string ( ) )
}
fn from_u64 ( u : u64 ) -> Self {
let mut buf = num_format ::Buffer ::new ( ) ;
buf . write_formatted ( & u , & LOCALE ) ;
Self ( buf . as_str ( ) . to_string ( ) )
}
fn from_u128 ( u : u128 ) -> Self {
let mut buf = num_format ::Buffer ::new ( ) ;
buf . write_formatted ( & u , & LOCALE ) ;
Self ( buf . as_str ( ) . to_string ( ) )
}
fn from_hashrate ( array : [ Option < f32 > ; 3 ] ) -> Self {
let mut string = " [ " . to_string ( ) ;
let mut buf = num_format ::Buffer ::new ( ) ;
let mut n = 0 ;
for i in array {
match i {
Some ( f ) = > {
let f = f as u128 ;
buf . write_formatted ( & f , & LOCALE ) ;
string . push_str ( buf . as_str ( ) ) ;
string . push_str ( " H/s " ) ;
} ,
None = > string . push_str ( " ??? H/s " ) ,
}
if n ! = 2 {
string . push_str ( " , " ) ;
n + = 1 ;
} else {
string . push ( ']' ) ;
break
}
}
Self ( string )
}
fn from_load ( array : [ Option < f32 > ; 3 ] ) -> Self {
let mut string = " [ " . to_string ( ) ;
let mut n = 0 ;
for i in array {
match i {
Some ( f ) = > string . push_str ( format! ( " {} " , f ) . as_str ( ) ) ,
None = > string . push_str ( " ??? " ) ,
}
if n ! = 2 {
string . push_str ( " , " ) ;
n + = 1 ;
} else {
string . push ( ']' ) ;
break
}
}
Self ( string )
}
}
2022-12-05 19:55:50 +00:00
//---------------------------------------------------------------------------------------------------- Regexes
// Not to be confused with the [Regexes] struct in [main.rs], this one is meant
// for parsing the output of P2Pool and finding payouts and total XMR found.
// Why Regex instead of the standard library?
// 1. I'm already using Regex
// 2. It's insanely faster
//
// The following STDLIB implementation takes [0.003~] seconds to find all matches given a [String] with 30k lines:
// let mut n = 0;
// for line in P2POOL_OUTPUT.lines() {
2022-12-06 03:33:35 +00:00
// if line.contains("You received a payout of [0-9].[0-9]+ XMR") { n += 1; }
2022-12-05 19:55:50 +00:00
// }
//
// This regex function takes [0.0003~] seconds (10x faster):
2022-12-06 03:33:35 +00:00
// let regex = Regex::new("You received a payout of [0-9].[0-9]+ XMR").unwrap();
2022-12-05 19:55:50 +00:00
// let n = regex.find_iter(P2POOL_OUTPUT).count();
//
// Both are nominally fast enough where it doesn't matter too much but meh, why not use regex.
struct P2poolRegex {
2022-12-06 03:33:35 +00:00
payout : regex ::Regex ,
float : regex ::Regex ,
2022-12-05 19:55:50 +00:00
}
impl P2poolRegex {
fn new ( ) -> Self {
2022-12-06 03:33:35 +00:00
Self {
payout : regex ::Regex ::new ( " You received a payout of [0-9].[0-9]+ XMR " ) . unwrap ( ) ,
float : regex ::Regex ::new ( " [0-9].[0-9]+ " ) . unwrap ( ) ,
}
2022-12-05 19:55:50 +00:00
}
}
2022-12-06 22:48:48 +00:00
//---------------------------------------------------------------------------------------------------- [ImgP2pool]
// A static "image" of data that P2Pool started with.
// This is just a snapshot of the user data when they initially started P2Pool.
// Created by [start_p2pool()] and return to the main GUI thread where it will store it.
// No need for an [Arc<Mutex>] since the Helper thread doesn't need this information.
2022-12-06 20:54:18 +00:00
#[ derive(Debug, Clone) ]
2022-12-06 22:48:48 +00:00
pub struct ImgP2pool {
2022-12-06 20:54:18 +00:00
pub mini : bool , // Did the user start on the mini-chain?
pub address : String , // What address is the current p2pool paying out to? (This gets shortened to [4xxxxx...xxxxxx])
pub host : String , // What monerod are we using?
pub rpc : String , // What is the RPC port?
pub zmq : String , // What is the ZMQ port?
pub out_peers : String , // How many out-peers?
pub in_peers : String , // How many in-peers?
pub log_level : String , // What log level?
}
2022-12-06 22:48:48 +00:00
impl ImgP2pool {
pub fn new ( ) -> Self {
2022-12-06 20:54:18 +00:00
Self {
mini : true ,
address : String ::new ( ) ,
host : String ::new ( ) ,
rpc : String ::new ( ) ,
zmq : String ::new ( ) ,
out_peers : String ::new ( ) ,
in_peers : String ::new ( ) ,
log_level : String ::new ( ) ,
}
}
}
2022-12-06 22:48:48 +00:00
//---------------------------------------------------------------------------------------------------- Public P2Pool API
// Helper/GUI threads both have a copy of this, Helper updates
// the GUI's version on a 1-second interval from the private data.
2022-12-06 03:33:35 +00:00
#[ derive(Debug, Clone) ]
2022-12-02 04:13:53 +00:00
pub struct PubP2poolApi {
2022-12-05 19:55:50 +00:00
// Output
pub output : String ,
2022-12-06 20:17:37 +00:00
// Uptime
pub uptime : HumanTime ,
2022-12-04 01:12:40 +00:00
// These are manually parsed from the STDOUT.
2022-12-05 19:55:50 +00:00
pub payouts : u128 ,
2022-12-04 01:12:40 +00:00
pub payouts_hour : f64 ,
pub payouts_day : f64 ,
pub payouts_month : f64 ,
pub xmr : f64 ,
pub xmr_hour : f64 ,
pub xmr_day : f64 ,
pub xmr_month : f64 ,
// The rest are serialized from the API, then turned into [HumanNumber]s
pub hashrate_15m : HumanNumber ,
pub hashrate_1h : HumanNumber ,
pub hashrate_24h : HumanNumber ,
pub shares_found : HumanNumber ,
pub average_effort : HumanNumber ,
pub current_effort : HumanNumber ,
pub connections : HumanNumber ,
2022-11-30 22:21:55 +00:00
}
2022-12-06 22:48:48 +00:00
impl Default for PubP2poolApi {
fn default ( ) -> Self {
Self ::new ( )
}
}
2022-12-02 04:13:53 +00:00
impl PubP2poolApi {
2022-11-30 22:21:55 +00:00
pub fn new ( ) -> Self {
Self {
2022-12-06 22:48:48 +00:00
output : String ::new ( ) ,
2022-12-06 20:17:37 +00:00
uptime : HumanTime ::new ( ) ,
2022-12-05 19:55:50 +00:00
payouts : 0 ,
2022-12-04 01:12:40 +00:00
payouts_hour : 0.0 ,
payouts_day : 0.0 ,
payouts_month : 0.0 ,
xmr : 0.0 ,
xmr_hour : 0.0 ,
xmr_day : 0.0 ,
xmr_month : 0.0 ,
hashrate_15m : HumanNumber ::unknown ( ) ,
hashrate_1h : HumanNumber ::unknown ( ) ,
hashrate_24h : HumanNumber ::unknown ( ) ,
shares_found : HumanNumber ::unknown ( ) ,
average_effort : HumanNumber ::unknown ( ) ,
current_effort : HumanNumber ::unknown ( ) ,
connections : HumanNumber ::unknown ( ) ,
}
}
2022-12-06 18:18:40 +00:00
// Mutate [PubP2poolApi] with data the process output.
2022-12-06 20:17:37 +00:00
fn update_from_output ( public : & Arc < Mutex < Self > > , output : & Arc < Mutex < String > > , elapsed : std ::time ::Duration , regex : & P2poolRegex ) {
// 1. Clone output
2022-12-06 03:33:35 +00:00
let output = output . lock ( ) . unwrap ( ) . clone ( ) ;
2022-12-06 20:17:37 +00:00
// 2. Parse STDOUT
2022-12-05 19:55:50 +00:00
let ( payouts , xmr ) = Self ::calc_payouts_and_xmr ( & output , & regex ) ;
2022-12-06 18:18:40 +00:00
2022-12-06 20:17:37 +00:00
// 3. Calculate hour/day/month given elapsed time
let elapsed_as_secs_f64 = elapsed . as_secs_f64 ( ) ;
2022-12-06 18:18:40 +00:00
// Payouts
2022-12-06 20:17:37 +00:00
let per_sec = ( payouts as f64 ) / elapsed_as_secs_f64 ;
2022-12-06 18:18:40 +00:00
let payouts_hour = ( per_sec * 60.0 ) * 60.0 ;
let payouts_day = payouts_hour * 24.0 ;
let payouts_month = payouts_day * 30.0 ;
// Total XMR
2022-12-06 20:17:37 +00:00
let per_sec = xmr / elapsed_as_secs_f64 ;
2022-12-06 18:18:40 +00:00
let xmr_hour = ( per_sec * 60.0 ) * 60.0 ;
let xmr_day = payouts_hour * 24.0 ;
let xmr_month = payouts_day * 30.0 ;
2022-12-06 20:17:37 +00:00
// 4. Mutate the struct with the new info
2022-12-06 18:18:40 +00:00
let mut public = public . lock ( ) . unwrap ( ) ;
* public = Self {
2022-12-06 20:17:37 +00:00
uptime : HumanTime ::into_human ( elapsed ) ,
2022-12-06 03:33:35 +00:00
output ,
2022-12-05 19:55:50 +00:00
payouts ,
xmr ,
2022-12-06 18:18:40 +00:00
payouts_hour ,
payouts_day ,
payouts_month ,
xmr_hour ,
xmr_day ,
xmr_month ,
.. public . clone ( )
2022-12-05 19:55:50 +00:00
} ;
2022-12-06 18:18:40 +00:00
}
// Mutate [PubP2poolApi] with data from a [PrivP2poolApi] and the process output.
fn update_from_priv ( public : & Arc < Mutex < Self > > , private : & Arc < Mutex < PrivP2poolApi > > ) {
2022-12-06 20:17:37 +00:00
// priv -> pub conversion
2022-12-06 03:33:35 +00:00
let private = private . lock ( ) . unwrap ( ) ;
2022-12-06 18:18:40 +00:00
let mut public = public . lock ( ) . unwrap ( ) ;
* public = Self {
2022-12-05 19:55:50 +00:00
hashrate_15m : HumanNumber ::from_u128 ( private . hashrate_15m ) ,
hashrate_1h : HumanNumber ::from_u128 ( private . hashrate_1h ) ,
hashrate_24h : HumanNumber ::from_u128 ( private . hashrate_24h ) ,
shares_found : HumanNumber ::from_u128 ( private . shares_found ) ,
average_effort : HumanNumber ::to_percent ( private . average_effort ) ,
current_effort : HumanNumber ::to_percent ( private . current_effort ) ,
connections : HumanNumber ::from_u16 ( private . connections ) ,
2022-12-06 18:18:40 +00:00
.. public . clone ( )
2022-12-05 19:55:50 +00:00
}
}
// Essentially greps the output for [x.xxxxxxxxxxxx XMR] where x = a number.
// It sums each match and counts along the way, handling an error by not adding and printing to console.
fn calc_payouts_and_xmr ( output : & str , regex : & P2poolRegex ) -> ( u128 /* payout count */ , f64 /* total xmr */ ) {
2022-12-06 03:33:35 +00:00
let iter = regex . payout . find_iter ( output ) ;
2022-12-05 19:55:50 +00:00
let mut result : f64 = 0.0 ;
let mut count : u128 = 0 ;
for i in iter {
2022-12-06 03:33:35 +00:00
match regex . float . find ( i . as_str ( ) ) . unwrap ( ) . as_str ( ) . parse ::< f64 > ( ) {
Ok ( num ) = > { result + = num ; count + = 1 ; } ,
Err ( e ) = > error! ( " P2Pool | Total XMR sum calculation error: [{}] " , e ) ,
2022-12-05 19:55:50 +00:00
}
}
( count , result )
}
2022-11-30 22:21:55 +00:00
}
2022-12-02 04:13:53 +00:00
//---------------------------------------------------------------------------------------------------- Private P2Pool API
// This is the data the "watchdog" threads mutate.
// It matches directly to P2Pool's [local/stats] JSON API file (excluding a few stats).
// P2Pool seems to initialize all stats at 0 (or 0.0), so no [Option] wrapper seems needed.
2022-12-06 03:33:35 +00:00
#[ derive(Debug, Serialize, Deserialize, Clone, Copy) ]
2022-12-02 04:13:53 +00:00
struct PrivP2poolApi {
hashrate_15m : u128 ,
hashrate_1h : u128 ,
hashrate_24h : u128 ,
shares_found : u128 ,
2022-12-04 01:12:40 +00:00
average_effort : f32 ,
current_effort : f32 ,
2022-12-02 04:13:53 +00:00
connections : u16 , // No one will have more than 65535 connections... right?
}
2022-11-30 22:21:55 +00:00
2022-12-02 04:13:53 +00:00
impl PrivP2poolApi {
fn new ( ) -> Self {
Self {
hashrate_15m : 0 ,
hashrate_1h : 0 ,
hashrate_24h : 0 ,
shares_found : 0 ,
average_effort : 0.0 ,
current_effort : 0.0 ,
connections : 0 ,
}
}
2022-11-30 22:21:55 +00:00
}
2022-12-06 22:48:48 +00:00
//---------------------------------------------------------------------------------------------------- [ImgXmrig]
#[ derive(Debug, Clone) ]
pub struct ImgXmrig { }
impl ImgXmrig {
pub fn new ( ) -> Self {
Self { }
}
}
2022-12-02 04:13:53 +00:00
//---------------------------------------------------------------------------------------------------- Public XMRig API
2022-12-06 03:33:35 +00:00
#[ derive(Debug, Clone) ]
2022-12-02 04:13:53 +00:00
pub struct PubXmrigApi {
2022-12-05 19:55:50 +00:00
output : String ,
2022-12-04 01:12:40 +00:00
worker_id : String ,
resources : HumanNumber ,
hashrate : HumanNumber ,
pool : String ,
diff : HumanNumber ,
accepted : HumanNumber ,
rejected : HumanNumber ,
2022-12-02 04:13:53 +00:00
}
2022-12-06 22:48:48 +00:00
impl Default for PubXmrigApi {
fn default ( ) -> Self {
Self ::new ( )
}
}
2022-12-02 04:13:53 +00:00
impl PubXmrigApi {
2022-11-30 22:21:55 +00:00
pub fn new ( ) -> Self {
Self {
2022-12-06 22:48:48 +00:00
output : String ::new ( ) ,
2022-12-04 01:12:40 +00:00
worker_id : " ??? " . to_string ( ) ,
resources : HumanNumber ::unknown ( ) ,
hashrate : HumanNumber ::unknown ( ) ,
pool : " ??? " . to_string ( ) ,
diff : HumanNumber ::unknown ( ) ,
accepted : HumanNumber ::unknown ( ) ,
rejected : HumanNumber ::unknown ( ) ,
}
}
// Formats raw private data into ready-to-print human readable version.
2022-12-05 19:55:50 +00:00
fn from_priv ( private : PrivXmrigApi , output : String ) -> Self {
2022-12-04 01:12:40 +00:00
Self {
2022-12-05 19:55:50 +00:00
output : output . clone ( ) ,
2022-12-04 01:12:40 +00:00
worker_id : private . worker_id ,
resources : HumanNumber ::from_load ( private . resources . load_average ) ,
hashrate : HumanNumber ::from_hashrate ( private . hashrate . total ) ,
pool : private . connection . pool ,
diff : HumanNumber ::from_u128 ( private . connection . diff ) ,
accepted : HumanNumber ::from_u128 ( private . connection . accepted ) ,
rejected : HumanNumber ::from_u128 ( private . connection . rejected ) ,
2022-11-30 22:21:55 +00:00
}
}
}
2022-12-02 04:13:53 +00:00
//---------------------------------------------------------------------------------------------------- Private XMRig API
// This matches to some JSON stats in the HTTP call [summary],
// e.g: [wget -qO- localhost:18085/1/summary].
// XMRig doesn't initialize stats at 0 (or 0.0) and instead opts for [null]
// which means some elements need to be wrapped in an [Option] or else serde will [panic!].
2022-12-06 03:33:35 +00:00
#[ derive(Debug, Serialize, Deserialize, Clone) ]
2022-12-02 04:13:53 +00:00
struct PrivXmrigApi {
worker_id : String ,
resources : Resources ,
connection : Connection ,
hashrate : Hashrate ,
}
impl PrivXmrigApi {
fn new ( ) -> Self {
Self {
worker_id : String ::new ( ) ,
resources : Resources ::new ( ) ,
connection : Connection ::new ( ) ,
hashrate : Hashrate ::new ( ) ,
}
}
}
2022-12-06 03:33:35 +00:00
#[ derive(Debug, Serialize, Deserialize, Clone, Copy) ]
2022-12-02 04:13:53 +00:00
struct Resources {
load_average : [ Option < f32 > ; 3 ] ,
}
impl Resources {
fn new ( ) -> Self {
Self {
load_average : [ Some ( 0.0 ) , Some ( 0.0 ) , Some ( 0.0 ) ] ,
}
}
}
2022-12-06 03:33:35 +00:00
#[ derive(Debug, Serialize, Deserialize, Clone) ]
2022-12-02 04:13:53 +00:00
struct Connection {
pool : String ,
diff : u128 ,
accepted : u128 ,
rejected : u128 ,
}
impl Connection {
fn new ( ) -> Self {
Self {
pool : String ::new ( ) ,
diff : 0 ,
accepted : 0 ,
rejected : 0 ,
}
}
}
2022-12-06 03:33:35 +00:00
#[ derive(Debug, Serialize, Deserialize, Clone, Copy) ]
2022-12-02 04:13:53 +00:00
struct Hashrate {
total : [ Option < f32 > ; 3 ] ,
}
impl Hashrate {
fn new ( ) -> Self {
Self {
total : [ Some ( 0.0 ) , Some ( 0.0 ) , Some ( 0.0 ) ] ,
}
}
}