2022-11-23 04:21:46 +00:00
// Gupax - GUI Uniting P2Pool And XMRig
//
2023-02-26 16:45:58 +00:00
// Copyright (c) 2022-2023 hinto-janai
2022-11-23 04:21:46 +00:00
//
// 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
2022-12-11 01:55:44 +00:00
// found here, e.g: User clicks [Stop P2Pool] -> Arc<Mutex<ProcessSignal> is set
// indicating to this thread during its loop: "I should stop P2Pool!", e.g:
2022-11-30 03:38:01 +00:00
//
2022-12-29 03:03:45 +00:00
// if lock!(p2pool).signal == ProcessSignal::Stop {
2022-12-11 01:55:44 +00:00
// stop_p2pool(),
2022-11-30 03:38:01 +00:00
// }
//
// 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-14 03:41:05 +00:00
process ::Stdio ,
2022-12-03 21:02:34 +00:00
fmt ::Write ,
2022-11-30 22:21:55 +00:00
time ::* ,
2022-11-28 17:05:09 +00:00
thread ,
} ;
2022-12-08 23:31:20 +00:00
use crate ::{
constants ::* ,
SudoState ,
2022-12-26 18:56:37 +00:00
human ::* ,
2022-12-28 21:04:26 +00:00
GupaxP2poolApi ,
xmr ::* ,
2022-12-29 03:03:45 +00:00
macros ::* ,
2023-01-26 03:34:51 +00:00
RemoteNode ,
2022-12-08 23:31:20 +00:00
} ;
2022-12-11 20:49:01 +00:00
use sysinfo ::SystemExt ;
2022-12-02 18:15:26 +00:00
use serde ::{ Serialize , Deserialize } ;
2022-12-14 03:41:05 +00:00
use sysinfo ::{ CpuExt , ProcessExt } ;
2022-11-28 17:05:09 +00:00
use log ::* ;
2023-04-19 14:01:11 +00:00
use crate ::regex ::{
P2POOL_REGEX ,
XMRIG_REGEX ,
} ;
2022-11-28 17:05:09 +00:00
2022-12-04 01:12:40 +00:00
//---------------------------------------------------------------------------------------------------- Constants
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.
2022-12-11 01:55:44 +00:00
const MAX_GUI_OUTPUT_BYTES : usize = 500_000 ;
2022-12-14 22:37:29 +00:00
// Just a little leeway so a reset will go off before the [String] allocates more memory.
2022-12-08 17:29:38 +00:00
const GUI_OUTPUT_LEEWAY : usize = MAX_GUI_OUTPUT_BYTES - 1000 ;
2022-12-27 14:22:46 +00:00
// Some constants for generating hashrate/difficulty.
const MONERO_BLOCK_TIME_IN_SECONDS : u64 = 120 ;
const P2POOL_BLOCK_TIME_IN_SECONDS : u64 = 10 ;
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]
2022-12-11 20:49:01 +00:00
pub uptime : HumanTime , // Gupax uptime formatting for humans
pub pub_sys : Arc < Mutex < Sys > > , // The public API for [sysinfo] that the [Status] tab reads from
2022-12-03 21:02:34 +00:00
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)
2023-01-01 23:57:11 +00:00
pub gupax_p2pool_api : Arc < Mutex < GupaxP2poolApi > > , //
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-12-11 20:49:01 +00:00
//----------------------------------------------------------------------------------------------------
#[ derive(Debug,Clone) ]
pub struct Sys {
pub gupax_uptime : String ,
pub gupax_cpu_usage : String ,
pub gupax_memory_used_mb : String ,
pub system_cpu_model : String ,
pub system_memory : String ,
pub system_cpu_usage : String ,
}
impl Sys {
pub fn new ( ) -> Self {
Self {
gupax_uptime : " 0 seconds " . to_string ( ) ,
gupax_cpu_usage : " ???% " . to_string ( ) ,
gupax_memory_used_mb : " ??? megabytes " . to_string ( ) ,
system_cpu_usage : " ???% " . to_string ( ) ,
system_memory : " ???GB / ???GB " . to_string ( ) ,
system_cpu_model : " ??? " . to_string ( ) ,
}
}
}
2022-12-26 17:18:57 +00:00
impl Default for Sys { fn default ( ) -> Self { Self ::new ( ) } }
2022-12-14 03:41:05 +00:00
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.
2023-04-14 17:45:30 +00:00
#[ derive(Debug) ]
2022-11-28 17:05:09 +00:00
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.
2023-02-06 22:50:48 +00:00
// child: Option<Arc<Mutex<Box<dyn portable_pty::Child + Send + std::marker::Sync>>>>, // STDOUT/STDERR is combined automatically thanks to this PTY, nice
// stdin: Option<Box<dyn portable_pty::MasterPty + Send>>, // A handle to the process's MasterPTY/STDIN
2022-12-05 19:55:50 +00:00
// This is the process's private output [String], used by both [Simple] and [Advanced].
2022-12-14 22:37:29 +00:00
// "parse" contains the output that will be parsed, then tossed out. "pub" will be written to
// the same as parse, but it will be [swap()]'d by the "helper" thread into the GUIs [String].
2022-12-08 17:29:38 +00:00
// The "helper" thread synchronizes this swap so that the data in here is moved there
// roughly once a second. GUI thread never touches this.
2022-12-14 22:37:29 +00:00
output_parse : Arc < Mutex < String > > ,
output_pub : 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 {
2022-12-14 03:41:05 +00:00
pub fn new ( name : ProcessName , _args : String , _path : PathBuf ) -> Self {
2022-11-28 17:05:09 +00:00
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 ( ) ,
2023-02-06 22:50:48 +00:00
// stdin: Option::None,
// child: Option::None,
2022-12-29 03:03:45 +00:00
output_parse : arc_mut ! ( String ::with_capacity ( 500 ) ) ,
output_pub : arc_mut ! ( String ::with_capacity ( 500 ) ) ,
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 {
2023-04-14 16:12:35 +00:00
self . state = = ProcessState ::Alive | |
self . state = = ProcessState ::Middle | |
2023-04-19 14:01:11 +00:00
self . state = = ProcessState ::Syncing | |
self . state = = ProcessState ::NotMining
2022-12-06 03:33:35 +00:00
}
pub fn is_waiting ( & self ) -> bool {
self . state = = ProcessState ::Middle | | self . state = = ProcessState ::Waiting
}
2023-04-14 16:12:35 +00:00
pub fn is_syncing ( & self ) -> bool {
self . state = = ProcessState ::Syncing
}
2023-04-19 14:01:11 +00:00
pub fn is_not_mining ( & self ) -> bool {
self . state = = ProcessState ::NotMining
}
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!
2023-04-14 16:12:35 +00:00
2023-04-19 13:35:51 +00:00
// Only for P2Pool, ORANGE.
2023-04-14 16:12:35 +00:00
Syncing ,
2023-04-19 14:01:11 +00:00
// Only for XMRig, ORANGE.
NotMining ,
2022-11-28 17:05:09 +00:00
}
2023-01-01 19:51:10 +00:00
impl Default for ProcessState { fn default ( ) -> Self { Self ::Dead } }
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 ,
}
2023-01-01 19:51:10 +00:00
impl Default for ProcessSignal { fn default ( ) -> Self { Self ::None } }
2022-11-28 17:05:09 +00:00
#[ 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 ) } }
2022-12-14 03:41:05 +00:00
impl std ::fmt ::Display for ProcessName {
fn fmt ( & self , f : & mut std ::fmt ::Formatter ) -> std ::fmt ::Result {
match * self {
ProcessName ::P2pool = > write! ( f , " P2Pool " ) ,
ProcessName ::Xmrig = > write! ( f , " XMRig " ) ,
}
}
}
2022-11-30 03:38:01 +00:00
2022-12-05 19:55:50 +00:00
//---------------------------------------------------------------------------------------------------- [Helper]
impl Helper {
//---------------------------------------------------------------------------------------------------- General Functions
2022-12-28 21:04:26 +00:00
pub fn new ( instant : std ::time ::Instant , pub_sys : Arc < Mutex < Sys > > , 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 > > , gupax_p2pool_api : Arc < Mutex < GupaxP2poolApi > > ) -> Self {
2022-12-05 19:55:50 +00:00
Self {
instant ,
2022-12-11 20:49:01 +00:00
pub_sys ,
uptime : HumanTime ::into_human ( instant . elapsed ( ) ) ,
2022-12-29 03:03:45 +00:00
pub_api_p2pool : arc_mut ! ( PubP2poolApi ::new ( ) ) ,
pub_api_xmrig : arc_mut ! ( 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-28 21:04:26 +00:00
gupax_p2pool_api ,
2022-12-05 19:55:50 +00:00
}
}
2023-04-19 14:18:04 +00:00
fn read_pty_xmrig ( output_parse : Arc < Mutex < String > > , output_pub : 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 ( ) ;
2022-12-28 21:04:26 +00:00
while let Some ( Ok ( line ) ) = stdout . next ( ) {
// println!("{}", line); // For debugging.
2023-04-19 14:18:04 +00:00
if let Err ( e ) = writeln! ( lock! ( output_parse ) , " {} " , line ) { error! ( " XMRig PTY Parse | Output error: {} " , e ) ; }
if let Err ( e ) = writeln! ( lock! ( output_pub ) , " {} " , line ) { error! ( " XMRig PTY Pub | Output error: {} " , e ) ; }
2022-12-28 21:04:26 +00:00
}
}
2023-04-19 14:01:11 +00:00
fn read_pty_p2pool ( output_parse : Arc < Mutex < String > > , output_pub : Arc < Mutex < String > > , reader : Box < dyn std ::io ::Read + Send > , gupax_p2pool_api : Arc < Mutex < GupaxP2poolApi > > ) {
2022-12-28 21:04:26 +00:00
use std ::io ::BufRead ;
let mut stdout = std ::io ::BufReader ::new ( reader ) . lines ( ) ;
while let Some ( Ok ( line ) ) = stdout . next ( ) {
// println!("{}", line); // For debugging.
2023-04-19 14:01:11 +00:00
if P2POOL_REGEX . payout . is_match ( & line ) {
2022-12-28 21:04:26 +00:00
debug! ( " P2Pool PTY | Found payout, attempting write: {} " , line ) ;
2023-04-19 14:01:11 +00:00
let ( date , atomic_unit , block ) = PayoutOrd ::parse_raw_payout_line ( & line ) ;
2022-12-31 18:47:41 +00:00
let formatted_log_line = GupaxP2poolApi ::format_payout ( & date , & atomic_unit , & block ) ;
GupaxP2poolApi ::add_payout ( & mut lock! ( gupax_p2pool_api ) , & formatted_log_line , date , atomic_unit , block ) ;
if let Err ( e ) = GupaxP2poolApi ::write_to_all_files ( & lock! ( gupax_p2pool_api ) , & formatted_log_line ) { error! ( " P2Pool PTY GupaxP2poolApi | Write error: {} " , e ) ; }
2022-12-14 22:37:29 +00:00
}
2022-12-29 03:03:45 +00:00
if let Err ( e ) = writeln! ( lock! ( output_parse ) , " {} " , line ) { error! ( " P2Pool PTY Parse | Output error: {} " , e ) ; }
if let Err ( e ) = writeln! ( lock! ( output_pub ) , " {} " , line ) { error! ( " P2Pool PTY Pub | Output error: {} " , e ) ; }
2022-12-05 19:55:50 +00:00
}
}
2022-12-14 22:37:29 +00:00
// Reset output if larger than max bytes.
2022-12-07 02:13:37 +00:00
// This will also append a message showing it was reset.
2022-12-15 02:09:37 +00:00
fn check_reset_gui_output ( output : & mut String , name : ProcessName ) {
let len = output . len ( ) ;
if len > GUI_OUTPUT_LEEWAY {
info! ( " {} Watchdog | Output is nearing {} bytes, resetting! " , name , MAX_GUI_OUTPUT_BYTES ) ;
2022-12-17 14:52:33 +00:00
let text = format! ( " {} \n {} GUI log is exceeding the maximum: {} bytes! \n I've reset the logs for you! \n {} \n \n \n \n " , HORI_CONSOLE , name , MAX_GUI_OUTPUT_BYTES , HORI_CONSOLE ) ;
2022-12-15 02:09:37 +00:00
output . clear ( ) ;
output . push_str ( & text ) ;
debug! ( " {} Watchdog | Resetting GUI output ... OK " , name ) ;
} else {
debug! ( " {} Watchdog | GUI output reset not needed! Current byte length ... {} " , name , len ) ;
2022-12-07 02:13:37 +00:00
}
}
2022-12-26 17:18:57 +00:00
// Read P2Pool/XMRig's API file to a [String].
fn path_to_string ( path : & std ::path ::PathBuf , name : ProcessName ) -> std ::result ::Result < String , std ::io ::Error > {
match std ::fs ::read_to_string ( path ) {
Ok ( s ) = > Ok ( s ) ,
Err ( e ) = > { warn! ( " {} API | [{}] read error: {} " , name , path . display ( ) , e ) ; Err ( e ) } ,
}
}
2022-12-05 19:55:50 +00:00
//---------------------------------------------------------------------------------------------------- P2Pool specific
2022-12-06 03:33:35 +00:00
// 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-29 03:03:45 +00:00
lock2! ( helper , p2pool ) . signal = ProcessSignal ::Stop ;
lock2! ( helper , p2pool ) . state = ProcessState ::Middle ;
2022-12-06 03:33:35 +00:00
}
// 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-29 03:03:45 +00:00
lock2! ( helper , p2pool ) . signal = ProcessSignal ::Restart ;
lock2! ( helper , p2pool ) . state = ProcessState ::Middle ;
2022-12-06 03:33:35 +00:00
2022-12-14 03:41:05 +00:00
let helper = Arc ::clone ( helper ) ;
2022-12-06 03:33:35 +00:00
let state = state . clone ( ) ;
let path = path . clone ( ) ;
// This thread lives to wait, start p2pool then die.
thread ::spawn ( move | | {
2022-12-29 03:03:45 +00:00
while lock2! ( helper , p2pool ) . is_alive ( ) {
2022-12-07 02:33:24 +00:00
warn! ( " P2Pool | Want to restart but process is still alive, waiting... " ) ;
2022-12-29 03:03:45 +00:00
sleep! ( 1000 ) ;
2022-12-06 03:33:35 +00:00
}
// Ok, process is not alive, start the new one!
2022-12-12 19:34:17 +00:00
info! ( " P2Pool | Old process seems dead, starting 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-29 03:03:45 +00:00
lock2! ( helper , p2pool ) . state = ProcessState ::Middle ;
2022-12-06 03:33:35 +00:00
2022-12-26 17:18:57 +00:00
let ( args , api_path_local , api_path_network , api_path_pool ) = Self ::build_p2pool_args_and_mutate_img ( helper , state , path ) ;
2022-12-06 22:48:48 +00:00
// Print arguments & user settings to console
2022-12-26 17:18:57 +00:00
crate ::disk ::print_dash ( & format! (
" P2Pool | Launch arguments: {:#?} | Local API Path: {:#?} | Network API Path: {:#?} | Pool API Path: {:#?} " ,
args ,
api_path_local ,
api_path_network ,
api_path_pool ,
) ) ;
2022-12-06 22:48:48 +00:00
// Spawn watchdog thread
2022-12-29 03:03:45 +00:00
let process = Arc ::clone ( & lock! ( helper ) . p2pool ) ;
let gui_api = Arc ::clone ( & lock! ( helper ) . gui_api_p2pool ) ;
let pub_api = Arc ::clone ( & lock! ( helper ) . pub_api_p2pool ) ;
let gupax_p2pool_api = Arc ::clone ( & lock! ( helper ) . gupax_p2pool_api ) ;
2022-12-06 22:48:48 +00:00
let path = path . clone ( ) ;
thread ::spawn ( move | | {
2022-12-28 21:04:26 +00:00
Self ::spawn_p2pool_watchdog ( process , gui_api , pub_api , args , path , api_path_local , api_path_network , api_path_pool , gupax_p2pool_api ) ;
2022-12-06 22:48:48 +00:00
} ) ;
}
2022-12-23 18:33:40 +00:00
// Takes in a 95-char Monero address, returns the first and last
// 6 characters separated with dots like so: [4abcde...abcdef]
fn head_tail_of_monero_address ( address : & str ) -> String {
if address . len ( ) < 95 { return " ??? " . to_string ( ) }
2022-12-28 21:04:26 +00:00
let head = & address [ 0 .. 6 ] ;
2022-12-23 18:33:40 +00:00
let tail = & address [ 89 .. 95 ] ;
head . to_owned ( ) + " ... " + tail
}
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...
2022-12-26 17:18:57 +00:00
pub fn build_p2pool_args_and_mutate_img ( helper : & Arc < Mutex < Self > > , state : & crate ::disk ::P2pool , path : & std ::path ::PathBuf ) -> ( Vec < String > , PathBuf , PathBuf , PathBuf ) {
2022-12-05 19:55:50 +00:00
let mut args = Vec ::with_capacity ( 500 ) ;
let path = path . clone ( ) ;
2022-12-14 03:41:05 +00:00
let mut api_path = path ;
2022-12-05 19:55:50 +00:00
api_path . pop ( ) ;
// [Simple]
if state . simple {
// Build the p2pool argument
2023-01-26 03:34:51 +00:00
let ( ip , rpc , zmq ) = RemoteNode ::get_ip_rpc_zmq ( & 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-29 03:03:45 +00:00
* lock2! ( helper , img_p2pool ) = ImgP2pool {
2022-12-23 18:33:40 +00:00
mini : " P2Pool Mini " . to_string ( ) ,
address : Self ::head_tail_of_monero_address ( & state . address ) ,
2022-12-06 20:54:18 +00:00
host : ip . to_string ( ) ,
rpc : rpc . to_string ( ) ,
zmq : zmq . 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 = " " ;
2022-12-29 03:03:45 +00:00
let lock = lock! ( helper ) ;
let mut p2pool_image = lock! ( lock . img_p2pool ) ;
2022-12-23 18:33:40 +00:00
let mut mini = false ;
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 {
2022-12-23 18:33:40 +00:00
" --mini " = > { mini = true ; p2pool_image . mini = " P2Pool Mini " . to_string ( ) ; } ,
" --wallet " = > p2pool_image . address = Self ::head_tail_of_monero_address ( arg ) ,
2022-12-06 22:48:48 +00:00
" --host " = > p2pool_image . host = arg . to_string ( ) ,
" --rpc-port " = > p2pool_image . rpc = arg . to_string ( ) ,
" --zmq-port " = > p2pool_image . zmq = arg . to_string ( ) ,
" --out-peers " = > p2pool_image . out_peers = arg . to_string ( ) ,
" --in-peers " = > p2pool_image . in_peers = arg . to_string ( ) ,
2022-12-13 17:44:57 +00:00
" --data-api " = > api_path = PathBuf ::from ( arg ) ,
2022-12-06 22:48:48 +00:00
_ = > ( ) ,
}
2022-12-23 18:33:40 +00:00
if ! mini { p2pool_image . mini = " P2Pool Main " . to_string ( ) ; }
2023-02-06 14:17:09 +00:00
let arg = if arg = = " localhost " { " 127.0.0.1 " } else { arg } ;
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 {
2023-02-06 16:32:38 +00:00
let ip = if state . ip = = " localhost " { " 127.0.0.1 " } else { & state . ip } ;
2022-12-06 20:33:57 +00:00
args . push ( " --wallet " . to_string ( ) ) ; args . push ( state . address . clone ( ) ) ; // Wallet
2023-02-06 14:17:09 +00:00
args . push ( " --host " . to_string ( ) ) ; args . push ( ip . to_string ( ) ) ; // IP
2023-02-06 16:32:38 +00:00
args . push ( " --rpc-port " . to_string ( ) ) ; args . push ( state . rpc . to_string ( ) ) ; // RPC
args . push ( " --zmq-port " . to_string ( ) ) ; args . push ( state . zmq . to_string ( ) ) ; // ZMQ
2022-12-06 20:33:57 +00:00
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-29 03:03:45 +00:00
* lock2! ( helper , img_p2pool ) = ImgP2pool {
2022-12-23 18:33:40 +00:00
mini : if state . mini { " P2Pool Mini " . to_string ( ) } else { " P2Pool Main " . to_string ( ) } ,
address : Self ::head_tail_of_monero_address ( & state . address ) ,
2022-12-06 20:54:18 +00:00
host : state . selected_ip . to_string ( ) ,
rpc : state . selected_rpc . to_string ( ) ,
zmq : state . selected_zmq . to_string ( ) ,
out_peers : state . out_peers . to_string ( ) ,
in_peers : state . in_peers . to_string ( ) ,
2022-12-13 17:51:32 +00:00
} ;
2022-12-05 19:55:50 +00:00
}
}
2022-12-26 17:18:57 +00:00
let mut api_path_local = api_path . clone ( ) ;
let mut api_path_network = api_path . clone ( ) ;
let mut api_path_pool = api_path . clone ( ) ;
api_path_local . push ( P2POOL_API_PATH_LOCAL ) ;
api_path_network . push ( P2POOL_API_PATH_NETWORK ) ;
api_path_pool . push ( P2POOL_API_PATH_POOL ) ;
( args , api_path_local , api_path_network , api_path_pool )
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-28 21:04:26 +00:00
fn spawn_p2pool_watchdog ( process : Arc < Mutex < Process > > , gui_api : Arc < Mutex < PubP2poolApi > > , pub_api : Arc < Mutex < PubP2poolApi > > , args : Vec < String > , path : std ::path ::PathBuf , api_path_local : std ::path ::PathBuf , api_path_network : std ::path ::PathBuf , api_path_pool : std ::path ::PathBuf , gupax_p2pool_api : Arc < Mutex < GupaxP2poolApi > > ) {
2022-12-05 19:55:50 +00:00
// 1a. Create PTY
2022-12-12 19:34:17 +00:00
debug! ( " P2Pool | Creating PTY... " ) ;
2022-12-05 19:55:50 +00:00
let pty = portable_pty ::native_pty_system ( ) ;
2022-12-06 03:33:35 +00:00
let pair = pty . openpty ( portable_pty ::PtySize {
2022-12-10 20:35:20 +00:00
rows : 100 ,
cols : 1000 ,
2022-12-05 19:55:50 +00:00
pixel_width : 0 ,
pixel_height : 0 ,
} ) . unwrap ( ) ;
// 1b. Create command
2022-12-12 19:34:17 +00:00
debug! ( " P2Pool | Creating 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
2022-12-12 19:34:17 +00:00
debug! ( " P2Pool | Creating child... " ) ;
2022-12-29 03:03:45 +00:00
let child_pty = arc_mut! ( pair . slave . spawn_command ( cmd ) . unwrap ( ) ) ;
2023-02-06 22:50:48 +00:00
drop ( pair . slave ) ;
2022-12-05 19:55:50 +00:00
// 2. Set process state
2022-12-12 19:34:17 +00:00
debug! ( " P2Pool | Setting process state... " ) ;
2022-12-29 03:03:45 +00:00
let mut lock = lock! ( process ) ;
2023-04-14 16:12:35 +00:00
lock . state = ProcessState ::Syncing ;
2022-12-05 19:55:50 +00:00
lock . signal = ProcessSignal ::None ;
lock . start = Instant ::now ( ) ;
let reader = pair . master . try_clone_reader ( ) . unwrap ( ) ; // Get STDOUT/STDERR before moving the PTY
2023-02-06 22:50:48 +00:00
let mut stdin = pair . master ;
2022-12-05 19:55:50 +00:00
drop ( lock ) ;
// 3. Spawn PTY read thread
2022-12-12 19:34:17 +00:00
debug! ( " P2Pool | Spawning PTY read thread... " ) ;
2022-12-29 03:03:45 +00:00
let output_parse = Arc ::clone ( & lock! ( process ) . output_parse ) ;
let output_pub = Arc ::clone ( & lock! ( process ) . output_pub ) ;
2022-12-28 21:04:26 +00:00
let gupax_p2pool_api = Arc ::clone ( & gupax_p2pool_api ) ;
2022-12-05 19:55:50 +00:00
thread ::spawn ( move | | {
2023-04-19 14:01:11 +00:00
Self ::read_pty_p2pool ( output_parse , output_pub , reader , gupax_p2pool_api ) ;
2022-12-05 19:55:50 +00:00
} ) ;
2022-12-29 03:03:45 +00:00
let output_parse = Arc ::clone ( & lock! ( process ) . output_parse ) ;
let output_pub = Arc ::clone ( & lock! ( process ) . output_pub ) ;
2022-12-05 19:55:50 +00:00
2022-12-26 17:18:57 +00:00
debug! ( " P2Pool | Cleaning old [local] API files... " ) ;
2022-12-12 01:43:34 +00:00
// Attempt to remove stale API file
2022-12-26 17:18:57 +00:00
match std ::fs ::remove_file ( & api_path_local ) {
2022-12-12 01:43:34 +00:00
Ok ( _ ) = > info! ( " P2Pool | Attempting to remove stale API file ... OK " ) ,
Err ( e ) = > warn! ( " P2Pool | Attempting to remove stale API file ... FAIL ... {} " , e ) ,
}
// Attempt to create a default empty one.
use std ::io ::Write ;
2022-12-26 17:18:57 +00:00
if std ::fs ::File ::create ( & api_path_local ) . is_ok ( ) {
2022-12-12 01:43:34 +00:00
let text = r # "{"hashrate_15m":0,"hashrate_1h":0,"hashrate_24h":0,"shares_found":0,"average_effort":0.0,"current_effort":0.0,"connections":0}"# ;
2022-12-26 17:18:57 +00:00
match std ::fs ::write ( & api_path_local , text ) {
2022-12-12 01:43:34 +00:00
Ok ( _ ) = > info! ( " P2Pool | Creating default empty API file ... OK " ) ,
Err ( e ) = > warn! ( " P2Pool | Creating default empty API file ... FAIL ... {} " , e ) ,
}
}
2022-12-29 03:03:45 +00:00
let start = lock! ( process ) . start ;
2022-12-06 03:33:35 +00:00
2022-12-19 01:17:32 +00:00
// Reset stats before loop
2022-12-29 03:03:45 +00:00
* lock! ( pub_api ) = PubP2poolApi ::new ( ) ;
* lock! ( gui_api ) = PubP2poolApi ::new ( ) ;
2022-12-19 01:17:32 +00:00
2022-12-05 19:55:50 +00:00
// 4. Loop as watchdog
2022-12-08 17:29:38 +00:00
info! ( " P2Pool | Entering watchdog mode... woof! " ) ;
2022-12-05 19:55:50 +00:00
loop {
// Set timer
let now = Instant ::now ( ) ;
2022-12-12 19:34:17 +00:00
debug! ( " P2Pool Watchdog | ----------- Start of loop ----------- " ) ;
2023-01-01 14:46:23 +00:00
lock! ( gui_api ) . tick + = 1 ;
2022-12-12 19:34:17 +00:00
// Check if the process is secretly died without us knowing :)
2022-12-29 03:03:45 +00:00
if let Ok ( Some ( code ) ) = lock! ( child_pty ) . try_wait ( ) {
2022-12-12 19:34:17 +00:00
debug! ( " P2Pool Watchdog | Process secretly died! Getting exit status " ) ;
let exit_status = match code . success ( ) {
2022-12-29 03:03:45 +00:00
true = > { lock! ( process ) . state = ProcessState ::Dead ; " Successful " } ,
false = > { lock! ( process ) . state = ProcessState ::Failed ; " Failed " } ,
2022-12-12 19:34:17 +00:00
} ;
let uptime = HumanTime ::into_human ( start . elapsed ( ) ) ;
info! ( " P2Pool Watchdog | Stopped ... Uptime was: [{}], Exit status: [{}] " , uptime , exit_status ) ;
// This is written directly into the GUI, because sometimes the 900ms event loop can't catch it.
2022-12-16 19:33:04 +00:00
if let Err ( e ) = writeln! (
2022-12-29 03:03:45 +00:00
lock! ( gui_api ) . output ,
2022-12-16 19:33:04 +00:00
" {} \n P2Pool stopped | Uptime: [{}] | Exit status: [{}] \n {} \n \n \n \n " ,
HORI_CONSOLE ,
uptime ,
exit_status ,
HORI_CONSOLE
) {
error! ( " P2Pool Watchdog | GUI Uptime/Exit status write failed: {} " , e ) ;
}
2022-12-29 03:03:45 +00:00
lock! ( process ) . signal = ProcessSignal ::None ;
2022-12-12 19:34:17 +00:00
debug! ( " P2Pool Watchdog | Secret dead process reap OK, breaking " ) ;
break
}
2022-12-05 19:55:50 +00:00
// Check SIGNAL
2022-12-29 03:03:45 +00:00
if lock! ( process ) . signal = = ProcessSignal ::Stop {
2022-12-12 19:34:17 +00:00
debug! ( " P2Pool Watchdog | Stop SIGNAL caught " ) ;
2022-12-14 03:41:05 +00:00
// This actually sends a SIGHUP to p2pool (closes the PTY, hangs up on p2pool)
2022-12-29 03:03:45 +00:00
if let Err ( e ) = lock! ( child_pty ) . kill ( ) { error! ( " P2Pool Watchdog | Kill error: {} " , e ) ; }
2022-12-05 19:55:50 +00:00
// Wait to get the exit status
2022-12-29 03:03:45 +00:00
let exit_status = match lock! ( child_pty ) . wait ( ) {
2022-12-06 20:17:37 +00:00
Ok ( e ) = > {
if e . success ( ) {
2022-12-29 03:03:45 +00:00
lock! ( process ) . state = ProcessState ::Dead ; " Successful "
2022-12-06 20:17:37 +00:00
} else {
2022-12-29 03:03:45 +00:00
lock! ( process ) . state = ProcessState ::Failed ; " Failed "
2022-12-06 20:17:37 +00:00
}
} ,
2022-12-29 03:03:45 +00:00
_ = > { lock! ( process ) . 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-12 19:34:17 +00:00
info! ( " P2Pool Watchdog | 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.
2022-12-16 19:33:04 +00:00
if let Err ( e ) = writeln! (
2022-12-29 03:03:45 +00:00
lock! ( gui_api ) . output ,
2022-12-16 19:33:04 +00:00
" {} \n P2Pool stopped | Uptime: [{}] | Exit status: [{}] \n {} \n \n \n \n " ,
HORI_CONSOLE ,
uptime ,
exit_status ,
HORI_CONSOLE
) {
error! ( " P2Pool Watchdog | GUI Uptime/Exit status write failed: {} " , e ) ;
}
2022-12-29 03:03:45 +00:00
lock! ( process ) . signal = ProcessSignal ::None ;
2022-12-12 19:34:17 +00:00
debug! ( " P2Pool Watchdog | Stop SIGNAL done, breaking " ) ;
2022-12-06 03:33:35 +00:00
break
2022-12-07 02:13:37 +00:00
// Check RESTART
2022-12-29 03:03:45 +00:00
} else if lock! ( process ) . signal = = ProcessSignal ::Restart {
2022-12-12 19:34:17 +00:00
debug! ( " P2Pool Watchdog | Restart SIGNAL caught " ) ;
2022-12-14 03:41:05 +00:00
// This actually sends a SIGHUP to p2pool (closes the PTY, hangs up on p2pool)
2022-12-29 03:03:45 +00:00
if let Err ( e ) = lock! ( child_pty ) . kill ( ) { error! ( " P2Pool Watchdog | Kill error: {} " , e ) ; }
2022-12-06 03:33:35 +00:00
// Wait to get the exit status
2022-12-29 03:03:45 +00:00
let exit_status = match lock! ( child_pty ) . wait ( ) {
2022-12-05 19:55:50 +00:00
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-12 19:34:17 +00:00
info! ( " P2Pool Watchdog | 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.
2022-12-16 19:33:04 +00:00
if let Err ( e ) = writeln! (
2022-12-29 03:03:45 +00:00
lock! ( gui_api ) . output ,
2022-12-16 19:33:04 +00:00
" {} \n P2Pool stopped | Uptime: [{}] | Exit status: [{}] \n {} \n \n \n \n " ,
HORI_CONSOLE ,
uptime ,
exit_status ,
HORI_CONSOLE
) {
error! ( " P2Pool Watchdog | GUI Uptime/Exit status write failed: {} " , e ) ;
}
2022-12-29 03:03:45 +00:00
lock! ( process ) . state = ProcessState ::Waiting ;
2022-12-12 19:34:17 +00:00
debug! ( " P2Pool Watchdog | Restart SIGNAL done, breaking " ) ;
2022-12-05 19:55:50 +00:00
break
}
// Check vector of user input
2022-12-29 03:03:45 +00:00
let mut lock = lock! ( process ) ;
2022-12-05 19:55:50 +00:00
if ! lock . input . is_empty ( ) {
let input = std ::mem ::take ( & mut lock . input ) ;
for line in input {
2023-02-06 22:50:48 +00:00
if line . is_empty ( ) { continue }
2022-12-12 19:34:17 +00:00
debug! ( " P2Pool Watchdog | User input not empty, writing to STDIN: [{}] " , line ) ;
2023-02-06 22:50:48 +00:00
// Windows terminals (or at least the PTY abstraction I'm using, portable_pty)
// requires a [\r\n] to end a line, whereas Unix is okay with just a [\n].
//
// I have literally read all of [portable_pty]'s source code, dug into Win32 APIs,
// even rewrote some of the actual PTY code in order to understand why STDIN doesn't work
// on Windows. It's because of a fucking missing [\r]. Another reason to hate Windows :D
//
// XMRig did actually work before though, since it reads STDIN directly without needing a newline.
#[ cfg(target_os = " windows " ) ]
if let Err ( e ) = write! ( stdin , " {} \r \n " , line ) { error! ( " P2Pool Watchdog | STDIN error: {} " , e ) ; }
#[ cfg(target_family = " unix " ) ]
if let Err ( e ) = writeln! ( stdin , " {} " , line ) { error! ( " P2Pool Watchdog | STDIN error: {} " , e ) ; }
// Flush.
if let Err ( e ) = stdin . flush ( ) { error! ( " P2Pool Watchdog | STDIN flush error: {} " , e ) ; }
2022-12-05 19:55:50 +00:00
}
}
2023-02-07 00:43:17 +00:00
drop ( lock ) ;
2022-12-13 14:39:09 +00:00
// Check if logs need resetting
2022-12-14 22:37:29 +00:00
debug! ( " P2Pool Watchdog | Attempting GUI log reset check " ) ;
2022-12-29 03:03:45 +00:00
let mut lock = lock! ( gui_api ) ;
2022-12-15 02:09:37 +00:00
Self ::check_reset_gui_output ( & mut lock . output , ProcessName ::P2pool ) ;
drop ( lock ) ;
2022-12-13 14:39:09 +00:00
2022-12-06 18:18:40 +00:00
// Always update from output
2022-12-12 19:34:17 +00:00
debug! ( " P2Pool Watchdog | Starting [update_from_output()] " ) ;
2023-04-19 14:01:11 +00:00
PubP2poolApi ::update_from_output ( & pub_api , & output_parse , & output_pub , start . elapsed ( ) , & process ) ;
2022-12-06 18:18:40 +00:00
2022-12-26 17:18:57 +00:00
// Read [local] API
debug! ( " P2Pool Watchdog | Attempting [local] API file read " ) ;
if let Ok ( string ) = Self ::path_to_string ( & api_path_local , ProcessName ::P2pool ) {
2022-12-06 03:33:35 +00:00
// Deserialize
2022-12-27 14:22:46 +00:00
if let Ok ( local_api ) = PrivP2poolLocalApi ::from_str ( & string ) {
2022-12-06 03:33:35 +00:00
// Update the structs.
2022-12-27 14:22:46 +00:00
PubP2poolApi ::update_from_local ( & pub_api , local_api ) ;
2022-12-26 17:18:57 +00:00
}
}
// If more than 1 minute has passed, read the other API files.
2023-01-01 14:46:23 +00:00
if lock! ( gui_api ) . tick > = 60 {
2022-12-27 14:22:46 +00:00
debug! ( " P2Pool Watchdog | Attempting [network] & [pool] API file read " ) ;
if let ( Ok ( network_api ) , Ok ( pool_api ) ) = ( Self ::path_to_string ( & api_path_network , ProcessName ::P2pool ) , Self ::path_to_string ( & api_path_pool , ProcessName ::P2pool ) ) {
if let ( Ok ( network_api ) , Ok ( pool_api ) ) = ( PrivP2poolNetworkApi ::from_str ( & network_api ) , PrivP2poolPoolApi ::from_str ( & pool_api ) ) {
PubP2poolApi ::update_from_network_pool ( & pub_api , network_api , pool_api ) ;
2023-01-01 14:46:23 +00:00
lock! ( gui_api ) . tick = 0 ;
2022-12-26 17:18:57 +00:00
}
2022-12-06 03:33:35 +00:00
}
}
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
2022-12-12 19:34:17 +00:00
if elapsed < 900 {
let sleep = ( 900 - elapsed ) as u64 ;
2023-01-01 14:46:23 +00:00
debug! ( " P2Pool Watchdog | END OF LOOP - Tick: [{}/60] - Sleeping for [{}]ms... " , lock! ( gui_api ) . tick , sleep ) ;
2022-12-29 03:03:45 +00:00
sleep! ( sleep ) ;
2022-12-12 19:34:17 +00:00
} else {
2023-01-01 14:46:23 +00:00
debug! ( " P2Pool Watchdog | END OF LOOP - Tick: [{}/60] Not sleeping! " , lock! ( gui_api ) . tick ) ;
2022-12-12 19:34:17 +00:00
}
2022-12-05 19:55:50 +00:00
}
// 5. If loop broke, we must be done here.
2022-12-12 19:34:17 +00:00
info! ( " P2Pool Watchdog | Watchdog thread exiting... Goodbye! " ) ;
2022-12-05 19:55:50 +00:00
}
2022-12-08 01:50:14 +00:00
//---------------------------------------------------------------------------------------------------- XMRig specific, most functions are very similar to P2Pool's
2022-12-11 01:55:44 +00:00
// If processes are started with [sudo] on macOS, they must also
// be killed with [sudo] (even if I have a direct handle to it as the
// parent process...!). This is only needed on macOS, not Linux.
fn sudo_kill ( pid : u32 , sudo : & Arc < Mutex < SudoState > > ) -> bool {
// Spawn [sudo] to execute [kill] on the given [pid]
let mut child = std ::process ::Command ::new ( " sudo " )
. args ( [ " --stdin " , " kill " , " -9 " , & pid . to_string ( ) ] )
. stdin ( Stdio ::piped ( ) )
. spawn ( ) . unwrap ( ) ;
// Write the [sudo] password to STDIN.
let mut stdin = child . stdin . take ( ) . unwrap ( ) ;
use std ::io ::Write ;
2022-12-29 03:03:45 +00:00
if let Err ( e ) = writeln! ( stdin , " {} \n " , lock! ( sudo ) . pass ) { error! ( " Sudo Kill | STDIN error: {} " , e ) ; }
2022-12-11 01:55:44 +00:00
// Return exit code of [sudo/kill].
child . wait ( ) . unwrap ( ) . success ( )
}
2022-12-08 17:29:38 +00:00
// Just sets some signals for the watchdog thread to pick up on.
pub fn stop_xmrig ( helper : & Arc < Mutex < Self > > ) {
2022-12-09 03:37:50 +00:00
info! ( " XMRig | Attempting to stop... " ) ;
2022-12-29 03:03:45 +00:00
lock2! ( helper , xmrig ) . signal = ProcessSignal ::Stop ;
lock2! ( helper , xmrig ) . state = ProcessState ::Middle ;
2022-12-08 17:29:38 +00:00
}
2022-12-08 23:31:20 +00:00
// The "restart frontend" to a "frontend" function.
// Basically calls to kill the current xmrig, waits a little, then starts the below function in a a new thread, then exit.
pub fn restart_xmrig ( helper : & Arc < Mutex < Self > > , state : & crate ::disk ::Xmrig , path : & std ::path ::PathBuf , sudo : Arc < Mutex < SudoState > > ) {
info! ( " XMRig | Attempting to restart... " ) ;
2022-12-29 03:03:45 +00:00
lock2! ( helper , xmrig ) . signal = ProcessSignal ::Restart ;
lock2! ( helper , xmrig ) . state = ProcessState ::Middle ;
2022-12-08 23:31:20 +00:00
2022-12-14 03:41:05 +00:00
let helper = Arc ::clone ( helper ) ;
2022-12-08 23:31:20 +00:00
let state = state . clone ( ) ;
let path = path . clone ( ) ;
// This thread lives to wait, start xmrig then die.
thread ::spawn ( move | | {
2022-12-29 03:03:45 +00:00
while lock2! ( helper , xmrig ) . state ! = ProcessState ::Waiting {
2022-12-08 23:31:20 +00:00
warn! ( " XMRig | Want to restart but process is still alive, waiting... " ) ;
2022-12-29 03:03:45 +00:00
sleep! ( 1000 ) ;
2022-12-08 23:31:20 +00:00
}
// Ok, process is not alive, start the new one!
2022-12-12 19:34:17 +00:00
info! ( " XMRig | Old process seems dead, starting new one! " ) ;
2022-12-08 23:31:20 +00:00
Self ::start_xmrig ( & helper , & state , & path , sudo ) ;
} ) ;
info! ( " XMRig | Restart ... OK " ) ;
}
2022-12-08 01:50:14 +00:00
2022-12-08 23:31:20 +00:00
pub fn start_xmrig ( helper : & Arc < Mutex < Self > > , state : & crate ::disk ::Xmrig , path : & std ::path ::PathBuf , sudo : Arc < Mutex < SudoState > > ) {
2022-12-29 03:03:45 +00:00
lock2! ( helper , xmrig ) . state = ProcessState ::Middle ;
2022-12-08 01:50:14 +00:00
2022-12-11 20:49:01 +00:00
let ( args , api_ip_port ) = Self ::build_xmrig_args_and_mutate_img ( helper , state , path ) ;
2022-12-08 01:50:14 +00:00
// Print arguments & user settings to console
crate ::disk ::print_dash ( & format! ( " XMRig | Launch arguments: {:#?} " , args ) ) ;
2022-12-08 17:29:38 +00:00
info! ( " XMRig | Using path: [{}] " , path . display ( ) ) ;
2022-12-08 01:50:14 +00:00
// Spawn watchdog thread
2022-12-29 03:03:45 +00:00
let process = Arc ::clone ( & lock! ( helper ) . xmrig ) ;
let gui_api = Arc ::clone ( & lock! ( helper ) . gui_api_xmrig ) ;
let pub_api = Arc ::clone ( & lock! ( helper ) . pub_api_xmrig ) ;
2022-12-08 01:50:14 +00:00
let path = path . clone ( ) ;
thread ::spawn ( move | | {
2023-01-01 23:57:11 +00:00
Self ::spawn_xmrig_watchdog ( process , gui_api , pub_api , args , path , sudo , api_ip_port ) ;
2022-12-08 01:50:14 +00:00
} ) ;
}
2022-12-08 17:29:38 +00:00
// Takes in some [State/Xmrig] and parses it to build the actual command arguments.
// Returns the [Vec] of actual arguments, and mutates the [ImgXmrig] for the main GUI thread
2022-12-08 01:50:14 +00:00
// It returns a value... and mutates a deeply nested passed argument... this is some pretty bad code...
2022-12-11 20:49:01 +00:00
pub fn build_xmrig_args_and_mutate_img ( helper : & Arc < Mutex < Self > > , state : & crate ::disk ::Xmrig , path : & std ::path ::PathBuf ) -> ( Vec < String > , String ) {
2022-12-05 19:55:50 +00:00
let mut args = Vec ::with_capacity ( 500 ) ;
2022-12-13 17:44:57 +00:00
let mut api_ip = String ::with_capacity ( 15 ) ;
let mut api_port = String ::with_capacity ( 5 ) ;
2022-12-08 01:50:14 +00:00
let path = path . clone ( ) ;
2022-12-08 23:31:20 +00:00
// The actual binary we're executing is [sudo], technically
// the XMRig path is just an argument to sudo, so add it.
// Before that though, add the ["--prompt"] flag and set it
// to emptyness so that it doesn't show up in the output.
2022-12-11 01:55:44 +00:00
if cfg! ( unix ) {
args . push ( r # "--prompt="# . to_string ( ) ) ;
2022-12-11 02:48:25 +00:00
args . push ( " -- " . to_string ( ) ) ;
2022-12-11 01:55:44 +00:00
args . push ( path . display ( ) . to_string ( ) ) ;
}
2022-12-08 01:50:14 +00:00
// [Simple]
2022-12-05 19:55:50 +00:00
if state . simple {
2022-12-08 01:50:14 +00:00
// Build the xmrig argument
2022-12-12 03:01:37 +00:00
let rig = if state . simple_rig . is_empty ( ) { GUPAX_VERSION_UNDERSCORE . to_string ( ) } else { state . simple_rig . clone ( ) } ; // Rig name
2022-12-08 17:29:38 +00:00
args . push ( " --url " . to_string ( ) ) ; args . push ( " 127.0.0.1:3333 " . to_string ( ) ) ; // Local P2Pool (the default)
args . push ( " --threads " . to_string ( ) ) ; args . push ( state . current_threads . to_string ( ) ) ; // Threads
2022-12-14 03:41:05 +00:00
args . push ( " --user " . to_string ( ) ) ; args . push ( rig ) ; // Rig name
2022-12-08 17:29:38 +00:00
args . push ( " --no-color " . to_string ( ) ) ; // No color
args . push ( " --http-host " . to_string ( ) ) ; args . push ( " 127.0.0.1 " . to_string ( ) ) ; // HTTP API IP
args . push ( " --http-port " . to_string ( ) ) ; args . push ( " 18088 " . to_string ( ) ) ; // HTTP API Port
if state . pause ! = 0 { args . push ( " --pause-on-active " . to_string ( ) ) ; args . push ( state . pause . to_string ( ) ) ; } // Pause on active
2022-12-29 03:03:45 +00:00
* lock2! ( helper , img_xmrig ) = ImgXmrig {
2022-12-08 17:29:38 +00:00
threads : state . current_threads . to_string ( ) ,
url : " 127.0.0.1:3333 (Local P2Pool) " . to_string ( ) ,
2022-12-08 01:50:14 +00:00
} ;
2022-12-13 17:44:57 +00:00
api_ip = " 127.0.0.1 " . to_string ( ) ;
api_port = " 18088 " . to_string ( ) ;
2022-12-08 01:50:14 +00:00
// [Advanced]
2022-12-05 19:55:50 +00:00
} else {
2022-12-08 01:50:14 +00:00
// Overriding command arguments
2022-12-05 19:55:50 +00:00
if ! state . arguments . is_empty ( ) {
2022-12-08 01:50:14 +00:00
// This parses the input and attemps to fill out
2022-12-08 17:29:38 +00:00
// the [ImgXmrig]... This is pretty bad code...
2022-12-08 01:50:14 +00:00
let mut last = " " ;
2022-12-29 03:03:45 +00:00
let lock = lock! ( helper ) ;
let mut xmrig_image = lock! ( lock . img_xmrig ) ;
2022-12-05 19:55:50 +00:00
for arg in state . arguments . split_whitespace ( ) {
2022-12-08 01:50:14 +00:00
match last {
2022-12-13 17:44:57 +00:00
" --threads " = > xmrig_image . threads = arg . to_string ( ) ,
" --url " = > xmrig_image . url = arg . to_string ( ) ,
2022-12-19 14:31:11 +00:00
" --http-host " = > api_ip = if arg = = " localhost " { " 127.0.0.1 " . to_string ( ) } else { arg . to_string ( ) } ,
2022-12-13 17:44:57 +00:00
" --http-port " = > api_port = arg . to_string ( ) ,
2022-12-08 01:50:14 +00:00
_ = > ( ) ,
}
2022-12-19 14:31:11 +00:00
args . push ( if arg = = " localhost " { " 127.0.0.1 " . to_string ( ) } else { arg . to_string ( ) } ) ;
2022-12-08 01:50:14 +00:00
last = arg ;
2022-12-05 19:55:50 +00:00
}
2022-12-08 01:50:14 +00:00
// Else, build the argument
2022-12-05 19:55:50 +00:00
} else {
2022-12-13 17:44:57 +00:00
// XMRig doesn't understand [localhost]
2023-02-06 16:32:38 +00:00
let ip = if state . ip = = " localhost " | | state . ip . is_empty ( ) { " 127.0.0.1 " } else { & state . ip } ;
2022-12-13 17:44:57 +00:00
api_ip = if state . api_ip = = " localhost " | | state . api_ip . is_empty ( ) { " 127.0.0.1 " . to_string ( ) } else { state . api_ip . to_string ( ) } ;
api_port = if state . api_port . is_empty ( ) { " 18088 " . to_string ( ) } else { state . api_port . to_string ( ) } ;
2023-02-06 16:32:38 +00:00
let url = format! ( " {} : {} " , ip , state . port ) ; // Combine IP:Port into one string
2022-12-08 17:29:38 +00:00
args . push ( " --user " . to_string ( ) ) ; args . push ( state . address . clone ( ) ) ; // Wallet
args . push ( " --threads " . to_string ( ) ) ; args . push ( state . current_threads . to_string ( ) ) ; // Threads
2023-02-06 16:32:38 +00:00
args . push ( " --rig-id " . to_string ( ) ) ; args . push ( state . rig . to_string ( ) ) ; // Rig ID
2022-12-08 17:29:38 +00:00
args . push ( " --url " . to_string ( ) ) ; args . push ( url . clone ( ) ) ; // IP/Port
2022-12-09 01:24:37 +00:00
args . push ( " --http-host " . to_string ( ) ) ; args . push ( api_ip . to_string ( ) ) ; // HTTP API IP
args . push ( " --http-port " . to_string ( ) ) ; args . push ( api_port . to_string ( ) ) ; // HTTP API Port
2022-12-08 17:29:38 +00:00
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 ( " --pause-on-active " . to_string ( ) ) ; args . push ( state . pause . to_string ( ) ) ; } // Pause on active
2022-12-29 03:03:45 +00:00
* lock2! ( helper , img_xmrig ) = ImgXmrig {
2022-12-08 17:29:38 +00:00
url ,
threads : state . current_threads . to_string ( ) ,
} ;
2022-12-05 19:55:50 +00:00
}
}
2022-12-13 17:44:57 +00:00
( args , format! ( " {} : {} " , api_ip , api_port ) )
2022-12-08 01:50:14 +00:00
}
2022-12-05 19:55:50 +00:00
2022-12-10 02:00:33 +00:00
// We actually spawn [sudo] on Unix, with XMRig being the argument.
#[ cfg(target_family = " unix " ) ]
fn create_xmrig_cmd_unix ( args : Vec < String > , path : PathBuf ) -> portable_pty ::CommandBuilder {
let mut cmd = portable_pty ::cmdbuilder ::CommandBuilder ::new ( " sudo " ) ;
cmd . args ( args ) ;
cmd . cwd ( path . as_path ( ) . parent ( ) . unwrap ( ) ) ;
cmd
}
// Gupax should be admin on Windows, so just spawn XMRig normally.
#[ cfg(target_os = " windows " ) ]
fn create_xmrig_cmd_windows ( args : Vec < String > , path : PathBuf ) -> portable_pty ::CommandBuilder {
let mut cmd = portable_pty ::cmdbuilder ::CommandBuilder ::new ( path . clone ( ) ) ;
cmd . args ( args ) ;
cmd . cwd ( path . as_path ( ) . parent ( ) . unwrap ( ) ) ;
cmd
}
2022-12-11 20:49:01 +00:00
// The XMRig watchdog. Spawns 1 OS thread for reading a PTY (STDOUT+STDERR), and combines the [Child] with a PTY so STDIN actually works.
// This isn't actually async, a tokio runtime is unfortunately needed because [Hyper] is an async library (HTTP API calls)
#[ tokio::main ]
2023-01-01 23:57:11 +00:00
async fn spawn_xmrig_watchdog ( process : Arc < Mutex < Process > > , gui_api : Arc < Mutex < PubXmrigApi > > , pub_api : Arc < Mutex < PubXmrigApi > > , args : Vec < String > , path : std ::path ::PathBuf , sudo : Arc < Mutex < SudoState > > , mut api_ip_port : String ) {
2022-12-08 01:50:14 +00:00
// 1a. Create PTY
2022-12-12 19:34:17 +00:00
debug! ( " XMRig | Creating PTY... " ) ;
2022-12-08 01:50:14 +00:00
let pty = portable_pty ::native_pty_system ( ) ;
2022-12-11 02:48:25 +00:00
let mut pair = pty . openpty ( portable_pty ::PtySize {
2022-12-10 20:35:20 +00:00
rows : 100 ,
cols : 1000 ,
2022-12-08 01:50:14 +00:00
pixel_width : 0 ,
pixel_height : 0 ,
} ) . unwrap ( ) ;
// 1b. Create command
2022-12-12 19:34:17 +00:00
debug! ( " XMRig | Creating command... " ) ;
2022-12-10 02:00:33 +00:00
#[ cfg(target_os = " windows " ) ]
let cmd = Self ::create_xmrig_cmd_windows ( args , path ) ;
#[ cfg(target_family = " unix " ) ]
let cmd = Self ::create_xmrig_cmd_unix ( args , path ) ;
2022-12-08 01:50:14 +00:00
// 1c. Create child
2022-12-12 19:34:17 +00:00
debug! ( " XMRig | Creating child... " ) ;
2022-12-29 03:03:45 +00:00
let child_pty = arc_mut! ( pair . slave . spawn_command ( cmd ) . unwrap ( ) ) ;
2023-02-06 22:50:48 +00:00
drop ( pair . slave ) ;
2022-12-08 01:50:14 +00:00
2022-12-11 02:48:25 +00:00
// 2. Input [sudo] pass, wipe, then drop.
if cfg! ( unix ) {
2022-12-12 19:34:17 +00:00
debug! ( " XMRig | Inputting [sudo] and wiping... " ) ;
2023-03-01 17:04:07 +00:00
// a) Sleep to wait for [sudo]'s non-echo prompt (on Unix).
2022-12-11 02:48:25 +00:00
// this prevents users pass from showing up in the STDOUT.
2022-12-29 03:03:45 +00:00
sleep! ( 3000 ) ;
if let Err ( e ) = writeln! ( pair . master , " {} " , lock! ( sudo ) . pass ) { error! ( " XMRig | Sudo STDIN error: {} " , e ) ; } ;
2022-12-11 02:48:25 +00:00
SudoState ::wipe ( & sudo ) ;
2023-03-01 17:04:07 +00:00
// b) Reset GUI STDOUT just in case.
debug! ( " XMRig | Clearing GUI output... " ) ;
lock! ( gui_api ) . output . clear ( ) ;
2022-12-11 02:48:25 +00:00
}
2023-03-01 17:04:07 +00:00
2022-12-11 02:48:25 +00:00
// 3. Set process state
2022-12-12 19:34:17 +00:00
debug! ( " XMRig | Setting process state... " ) ;
2022-12-29 03:03:45 +00:00
let mut lock = lock! ( process ) ;
2023-04-19 14:18:04 +00:00
lock . state = ProcessState ::NotMining ;
2022-12-08 01:50:14 +00:00
lock . signal = ProcessSignal ::None ;
lock . start = Instant ::now ( ) ;
let reader = pair . master . try_clone_reader ( ) . unwrap ( ) ; // Get STDOUT/STDERR before moving the PTY
2023-02-06 22:50:48 +00:00
let mut stdin = pair . master ;
2022-12-08 01:50:14 +00:00
drop ( lock ) ;
2022-12-11 02:48:25 +00:00
// 4. Spawn PTY read thread
2022-12-12 19:34:17 +00:00
debug! ( " XMRig | Spawning PTY read thread... " ) ;
2022-12-29 03:03:45 +00:00
let output_parse = Arc ::clone ( & lock! ( process ) . output_parse ) ;
let output_pub = Arc ::clone ( & lock! ( process ) . output_pub ) ;
2022-12-05 19:55:50 +00:00
thread ::spawn ( move | | {
2022-12-28 21:04:26 +00:00
Self ::read_pty_xmrig ( output_parse , output_pub , reader ) ;
2022-12-05 19:55:50 +00:00
} ) ;
2023-04-19 14:18:04 +00:00
let output_parse = Arc ::clone ( & lock! ( process ) . output_parse ) ;
2022-12-29 03:03:45 +00:00
let output_pub = Arc ::clone ( & lock! ( process ) . output_pub ) ;
2022-12-05 19:55:50 +00:00
2022-12-11 20:49:01 +00:00
let client : hyper ::Client < hyper ::client ::HttpConnector > = hyper ::Client ::builder ( ) . build ( hyper ::client ::HttpConnector ::new ( ) ) ;
2022-12-29 03:03:45 +00:00
let start = lock! ( process ) . start ;
2022-12-19 14:31:11 +00:00
let api_uri = {
if ! api_ip_port . ends_with ( '/' ) { api_ip_port . push ( '/' ) ; }
" http:// " . to_owned ( ) + & api_ip_port + XMRIG_API_URI
} ;
info! ( " XMRig | Final API URI: {} " , api_uri ) ;
2022-12-08 01:50:14 +00:00
2022-12-19 01:17:32 +00:00
// Reset stats before loop
2022-12-29 03:03:45 +00:00
* lock! ( pub_api ) = PubXmrigApi ::new ( ) ;
* lock! ( gui_api ) = PubXmrigApi ::new ( ) ;
2022-12-19 01:17:32 +00:00
2022-12-11 02:48:25 +00:00
// 5. Loop as watchdog
2022-12-08 17:29:38 +00:00
info! ( " XMRig | Entering watchdog mode... woof! " ) ;
2022-12-08 01:50:14 +00:00
loop {
// Set timer
let now = Instant ::now ( ) ;
2022-12-12 19:34:17 +00:00
debug! ( " XMRig Watchdog | ----------- Start of loop ----------- " ) ;
2022-12-08 01:50:14 +00:00
2022-12-11 01:55:44 +00:00
// Check if the process secretly died without us knowing :)
2022-12-29 03:03:45 +00:00
if let Ok ( Some ( code ) ) = lock! ( child_pty ) . try_wait ( ) {
2022-12-12 19:34:17 +00:00
debug! ( " XMRig Watchdog | Process secretly died on us! Getting exit status... " ) ;
2022-12-11 01:55:44 +00:00
let exit_status = match code . success ( ) {
2022-12-29 03:03:45 +00:00
true = > { lock! ( process ) . state = ProcessState ::Dead ; " Successful " } ,
false = > { lock! ( process ) . state = ProcessState ::Failed ; " Failed " } ,
2022-12-11 01:55:44 +00:00
} ;
let uptime = HumanTime ::into_human ( start . elapsed ( ) ) ;
info! ( " XMRig | Stopped ... Uptime was: [{}], Exit status: [{}] " , uptime , exit_status ) ;
2022-12-16 19:33:04 +00:00
if let Err ( e ) = writeln! (
2022-12-29 03:03:45 +00:00
lock! ( gui_api ) . output ,
2022-12-16 19:33:04 +00:00
" {} \n XMRig stopped | Uptime: [{}] | Exit status: [{}] \n {} \n \n \n \n " ,
HORI_CONSOLE ,
uptime ,
exit_status ,
HORI_CONSOLE
) {
error! ( " XMRig Watchdog | GUI Uptime/Exit status write failed: {} " , e ) ;
}
2022-12-29 03:03:45 +00:00
lock! ( process ) . signal = ProcessSignal ::None ;
2022-12-12 19:34:17 +00:00
debug! ( " XMRig Watchdog | Secret dead process reap OK, breaking " ) ;
2022-12-11 01:55:44 +00:00
break
}
2022-12-08 23:31:20 +00:00
// Stop on [Stop/Restart] SIGNAL
2022-12-29 03:03:45 +00:00
let signal = lock! ( process ) . signal ;
2022-12-08 23:31:20 +00:00
if signal = = ProcessSignal ::Stop | | signal = = ProcessSignal ::Restart {
2022-12-12 19:34:17 +00:00
debug! ( " XMRig Watchdog | Stop/Restart SIGNAL caught " ) ;
2022-12-11 01:55:44 +00:00
// macOS requires [sudo] again to kill [XMRig]
if cfg! ( target_os = " macos " ) {
// If we're at this point, that means the user has
// entered their [sudo] pass again, after we wiped it.
// So, we should be able to find it in our [Arc<Mutex<SudoState>>].
2022-12-29 03:03:45 +00:00
Self ::sudo_kill ( lock! ( child_pty ) . process_id ( ) . unwrap ( ) , & sudo ) ;
2022-12-11 01:55:44 +00:00
// And... wipe it again (only if we're stopping full).
// If we're restarting, the next start will wipe it for us.
if signal ! = ProcessSignal ::Restart { SudoState ::wipe ( & sudo ) ; }
2022-12-29 03:03:45 +00:00
} else if let Err ( e ) = lock! ( child_pty ) . kill ( ) {
2022-12-17 15:09:50 +00:00
error! ( " XMRig Watchdog | Kill error: {} " , e ) ;
2022-12-11 01:55:44 +00:00
}
2022-12-29 03:03:45 +00:00
let exit_status = match lock! ( child_pty ) . wait ( ) {
2022-12-08 01:50:14 +00:00
Ok ( e ) = > {
2022-12-29 03:03:45 +00:00
let mut process = lock! ( process ) ;
2022-12-08 01:50:14 +00:00
if e . success ( ) {
2022-12-08 23:31:20 +00:00
if process . signal = = ProcessSignal ::Stop { process . state = ProcessState ::Dead ; }
" Successful "
2022-12-08 01:50:14 +00:00
} else {
2022-12-08 23:31:20 +00:00
if process . signal = = ProcessSignal ::Stop { process . state = ProcessState ::Failed ; }
" Failed "
2022-12-08 01:50:14 +00:00
}
} ,
2022-12-08 23:31:20 +00:00
_ = > {
2022-12-29 03:03:45 +00:00
let mut process = lock! ( process ) ;
2022-12-08 23:31:20 +00:00
if process . signal = = ProcessSignal ::Stop { process . state = ProcessState ::Failed ; }
" Unknown Error "
} ,
2022-12-08 01:50:14 +00:00
} ;
let uptime = HumanTime ::into_human ( start . elapsed ( ) ) ;
info! ( " XMRig | Stopped ... Uptime was: [{}], Exit status: [{}] " , uptime , exit_status ) ;
2022-12-16 19:33:04 +00:00
if let Err ( e ) = writeln! (
2022-12-29 03:03:45 +00:00
lock! ( gui_api ) . output ,
2022-12-16 19:33:04 +00:00
" {} \n XMRig stopped | Uptime: [{}] | Exit status: [{}] \n {} \n \n \n \n " ,
HORI_CONSOLE ,
uptime ,
exit_status ,
HORI_CONSOLE
) {
error! ( " XMRig Watchdog | GUI Uptime/Exit status write failed: {} " , e ) ;
}
2022-12-29 03:03:45 +00:00
let mut process = lock! ( process ) ;
2022-12-08 23:31:20 +00:00
match process . signal {
ProcessSignal ::Stop = > process . signal = ProcessSignal ::None ,
ProcessSignal ::Restart = > process . state = ProcessState ::Waiting ,
_ = > ( ) ,
}
2022-12-12 19:34:17 +00:00
debug! ( " XMRig Watchdog | Stop/Restart SIGNAL done, breaking " ) ;
2022-12-08 01:50:14 +00:00
break
}
// Check vector of user input
2022-12-29 03:03:45 +00:00
let mut lock = lock! ( process ) ;
2022-12-08 01:50:14 +00:00
if ! lock . input . is_empty ( ) {
let input = std ::mem ::take ( & mut lock . input ) ;
for line in input {
2023-02-06 22:50:48 +00:00
if line . is_empty ( ) { continue }
2022-12-12 19:34:17 +00:00
debug! ( " XMRig Watchdog | User input not empty, writing to STDIN: [{}] " , line ) ;
2023-02-06 22:50:48 +00:00
#[ cfg(target_os = " windows " ) ]
if let Err ( e ) = write! ( stdin , " {} \r \n " , line ) { error! ( " XMRig Watchdog | STDIN error: {} " , e ) ; }
#[ cfg(target_family = " unix " ) ]
if let Err ( e ) = writeln! ( stdin , " {} " , line ) { error! ( " XMRig Watchdog | STDIN error: {} " , e ) ; }
// Flush.
if let Err ( e ) = stdin . flush ( ) { error! ( " XMRig Watchdog | STDIN flush error: {} " , e ) ; }
2022-12-08 01:50:14 +00:00
}
}
2023-02-07 00:43:17 +00:00
drop ( lock ) ;
2022-12-08 01:50:14 +00:00
2022-12-13 14:39:09 +00:00
// Check if logs need resetting
2022-12-14 22:37:29 +00:00
debug! ( " XMRig Watchdog | Attempting GUI log reset check " ) ;
2022-12-29 03:03:45 +00:00
let mut lock = lock! ( gui_api ) ;
2022-12-15 02:09:37 +00:00
Self ::check_reset_gui_output ( & mut lock . output , ProcessName ::Xmrig ) ;
drop ( lock ) ;
2022-12-13 14:39:09 +00:00
2022-12-08 01:50:14 +00:00
// Always update from output
2022-12-12 19:34:17 +00:00
debug! ( " XMRig Watchdog | Starting [update_from_output()] " ) ;
2023-04-19 14:18:04 +00:00
PubXmrigApi ::update_from_output ( & pub_api , & output_pub , & output_parse , start . elapsed ( ) , & process ) ;
2022-12-11 20:49:01 +00:00
// Send an HTTP API request
2022-12-12 19:34:17 +00:00
debug! ( " XMRig Watchdog | Attempting HTTP API request... " ) ;
2022-12-19 14:31:11 +00:00
if let Ok ( priv_api ) = PrivXmrigApi ::request_xmrig_api ( client . clone ( ) , & api_uri ) . await {
2022-12-12 19:34:17 +00:00
debug! ( " XMRig Watchdog | HTTP API request OK, attempting [update_from_priv()] " ) ;
2022-12-12 03:01:37 +00:00
PubXmrigApi ::update_from_priv ( & pub_api , priv_api ) ;
} else {
2022-12-19 14:31:11 +00:00
warn! ( " XMRig Watchdog | Could not send HTTP API request to: {} " , api_uri ) ;
2022-12-11 20:49:01 +00:00
}
2022-12-08 01:50:14 +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
2022-12-12 19:34:17 +00:00
if elapsed < 900 {
let sleep = ( 900 - elapsed ) as u64 ;
debug! ( " XMRig Watchdog | END OF LOOP - Sleeping for [{}]ms... " , sleep ) ;
2022-12-29 03:03:45 +00:00
sleep! ( sleep ) ;
2022-12-12 19:34:17 +00:00
} else {
debug! ( " XMRig Watchdog | END OF LOOP - Not sleeping! " ) ;
}
2022-12-08 01:50:14 +00:00
}
// 5. If loop broke, we must be done here.
2022-12-12 19:34:17 +00:00
info! ( " XMRig Watchdog | Watchdog thread exiting... Goodbye! " ) ;
2022-12-05 19:55:50 +00:00
}
//---------------------------------------------------------------------------------------------------- The "helper"
2022-12-11 20:49:01 +00:00
fn update_pub_sys_from_sysinfo ( sysinfo : & sysinfo ::System , pub_sys : & mut Sys , pid : & sysinfo ::Pid , helper : & Helper , max_threads : usize ) {
let gupax_uptime = helper . uptime . to_string ( ) ;
let cpu = & sysinfo . cpus ( ) [ 0 ] ;
let gupax_cpu_usage = format! ( " {:.2} % " , sysinfo . process ( * pid ) . unwrap ( ) . cpu_usage ( ) / ( max_threads as f32 ) ) ;
let gupax_memory_used_mb = HumanNumber ::from_u64 ( sysinfo . process ( * pid ) . unwrap ( ) . memory ( ) / 1_000_000 ) ;
let gupax_memory_used_mb = format! ( " {} megabytes " , gupax_memory_used_mb ) ;
let system_cpu_model = format! ( " {} ( {} MHz) " , cpu . brand ( ) , cpu . frequency ( ) ) ;
let system_memory = {
let used = ( sysinfo . used_memory ( ) as f64 ) / 1_000_000_000.0 ;
let total = ( sysinfo . total_memory ( ) as f64 ) / 1_000_000_000.0 ;
format! ( " {:.3} GB / {:.3} GB " , used , total )
} ;
let system_cpu_usage = {
let mut total : f32 = 0.0 ;
for cpu in sysinfo . cpus ( ) {
total + = cpu . cpu_usage ( ) ;
}
format! ( " {:.2} % " , total / ( max_threads as f32 ) )
} ;
* pub_sys = Sys {
gupax_uptime ,
gupax_cpu_usage ,
gupax_memory_used_mb ,
system_cpu_usage ,
system_memory ,
system_cpu_model ,
} ;
}
2022-12-06 22:48:48 +00:00
// The "helper" thread. Syncs data between threads here and the GUI.
2022-12-11 20:49:01 +00:00
pub fn spawn_helper ( helper : & Arc < Mutex < Self > > , mut sysinfo : sysinfo ::System , pid : sysinfo ::Pid , max_threads : usize ) {
2022-12-13 14:39:09 +00:00
// The ordering of these locks is _very_ important. They MUST be in sync with how the main GUI thread locks stuff
// or a deadlock will occur given enough time. They will eventually both want to lock the [Arc<Mutex>] the other
// thread is already locking. Yes, I figured this out the hard way, hence the vast amount of debug!() messages.
// Example of different order (BAD!):
//
// GUI Main -> locks [p2pool] first
// Helper -> locks [gui_api_p2pool] first
// GUI Status Tab -> trys to lock [gui_api_p2pool] -> CAN'T
// Helper -> trys to lock [p2pool] -> CAN'T
//
// These two threads are now in a deadlock because both
// are trying to access locks the other one already has.
//
// The locking order here must be in the same chronological
// order as the main GUI thread (top to bottom).
let helper = Arc ::clone ( helper ) ;
2022-12-29 03:03:45 +00:00
let lock = lock! ( helper ) ;
2022-12-13 14:39:09 +00:00
let p2pool = Arc ::clone ( & lock . p2pool ) ;
let xmrig = Arc ::clone ( & lock . xmrig ) ;
let pub_sys = Arc ::clone ( & lock . pub_sys ) ;
let gui_api_p2pool = Arc ::clone ( & lock . gui_api_p2pool ) ;
let gui_api_xmrig = Arc ::clone ( & lock . gui_api_xmrig ) ;
let pub_api_p2pool = Arc ::clone ( & lock . pub_api_p2pool ) ;
let pub_api_xmrig = Arc ::clone ( & lock . pub_api_xmrig ) ;
drop ( lock ) ;
2022-12-11 20:49:01 +00:00
let sysinfo_cpu = sysinfo ::CpuRefreshKind ::everything ( ) ;
let sysinfo_processes = sysinfo ::ProcessRefreshKind ::new ( ) . with_cpu ( ) ;
2022-12-06 22:48:48 +00:00
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-12 19:34:17 +00:00
debug! ( " Helper | ----------- Start of loop ----------- " ) ;
// Ignore the invasive [debug!()] messages on the right side of the code.
// The reason why they are there are so that it's extremely easy to track
// down the culprit of an [Arc<Mutex>] deadlock. I know, they're ugly.
2022-12-05 19:55:50 +00:00
2022-12-06 22:48:48 +00:00
// 2. Lock... EVERYTHING!
2022-12-29 03:03:45 +00:00
let mut lock = lock! ( helper ) ; debug! ( " Helper | Locking (1/8) ... [helper] " ) ;
let p2pool = lock! ( p2pool ) ; debug! ( " Helper | Locking (2/8) ... [p2pool] " ) ;
let xmrig = lock! ( xmrig ) ; debug! ( " Helper | Locking (3/8) ... [xmrig] " ) ;
let mut lock_pub_sys = lock! ( pub_sys ) ; debug! ( " Helper | Locking (4/8) ... [pub_sys] " ) ;
let mut gui_api_p2pool = lock! ( gui_api_p2pool ) ; debug! ( " Helper | Locking (5/8) ... [gui_api_p2pool] " ) ;
let mut gui_api_xmrig = lock! ( gui_api_xmrig ) ; debug! ( " Helper | Locking (6/8) ... [gui_api_xmrig] " ) ;
let mut pub_api_p2pool = lock! ( pub_api_p2pool ) ; debug! ( " Helper | Locking (7/8) ... [pub_api_p2pool] " ) ;
let mut pub_api_xmrig = lock! ( pub_api_xmrig ) ; debug! ( " Helper | Locking (8/8) ... [pub_api_xmrig] " ) ;
2022-12-11 20:49:01 +00:00
// Calculate Gupax's uptime always.
lock . uptime = HumanTime ::into_human ( lock . instant . elapsed ( ) ) ;
2022-12-08 17:29:38 +00:00
// If [P2Pool] is alive...
2022-12-12 19:34:17 +00:00
if p2pool . is_alive ( ) {
debug! ( " Helper | P2Pool is alive! Running [combine_gui_pub_api()] " ) ;
PubP2poolApi ::combine_gui_pub_api ( & mut gui_api_p2pool , & mut pub_api_p2pool ) ;
} else {
debug! ( " Helper | P2Pool is dead! Skipping... " ) ;
}
2022-12-08 17:29:38 +00:00
// If [XMRig] is alive...
2022-12-12 19:34:17 +00:00
if xmrig . is_alive ( ) {
debug! ( " Helper | XMRig is alive! Running [combine_gui_pub_api()] " ) ;
PubXmrigApi ::combine_gui_pub_api ( & mut gui_api_xmrig , & mut pub_api_xmrig ) ;
} else {
debug! ( " Helper | XMRig is dead! Skipping... " ) ;
}
2022-12-06 22:48:48 +00:00
2022-12-11 20:49:01 +00:00
// 2. Selectively refresh [sysinfo] for only what we need (better performance).
2022-12-12 19:34:17 +00:00
sysinfo . refresh_cpu_specifics ( sysinfo_cpu ) ; debug! ( " Helper | Sysinfo refresh (1/3) ... [cpu] " ) ;
sysinfo . refresh_processes_specifics ( sysinfo_processes ) ; debug! ( " Helper | Sysinfo refresh (2/3) ... [processes] " ) ;
sysinfo . refresh_memory ( ) ; debug! ( " Helper | Sysinfo refresh (3/3) ... [memory] " ) ;
debug! ( " Helper | Sysinfo OK, running [update_pub_sys_from_sysinfo()] " ) ;
2022-12-11 20:49:01 +00:00
Self ::update_pub_sys_from_sysinfo ( & sysinfo , & mut lock_pub_sys , & pid , & lock , max_threads ) ;
// 3. Drop... (almost) EVERYTHING... IN REVERSE!
2022-12-12 19:34:17 +00:00
drop ( lock_pub_sys ) ; debug! ( " Helper | Unlocking (1/8) ... [pub_sys] " ) ;
drop ( xmrig ) ; debug! ( " Helper | Unlocking (2/8) ... [xmrig] " ) ;
drop ( p2pool ) ; debug! ( " Helper | Unlocking (3/8) ... [p2pool] " ) ;
drop ( pub_api_xmrig ) ; debug! ( " Helper | Unlocking (4/8) ... [pub_api_xmrig] " ) ;
drop ( pub_api_p2pool ) ; debug! ( " Helper | Unlocking (5/8) ... [pub_api_p2pool] " ) ;
drop ( gui_api_xmrig ) ; debug! ( " Helper | Unlocking (6/8) ... [gui_api_xmrig] " ) ;
drop ( gui_api_p2pool ) ; debug! ( " Helper | Unlocking (7/8) ... [gui_api_p2pool] " ) ;
drop ( lock ) ; debug! ( " Helper | Unlocking (8/8) ... [helper] " ) ;
2022-12-05 19:55:50 +00:00
2022-12-11 20:49:01 +00:00
// 4. 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.
2022-12-12 19:34:17 +00:00
let sleep = ( 1000 - elapsed ) as u64 ;
debug! ( " Helper | END OF LOOP - Sleeping for [{}]ms... " , sleep ) ;
2022-12-29 03:03:45 +00:00
sleep! ( sleep ) ;
2022-12-12 19:34:17 +00:00
} else {
debug! ( " Helper | END OF LOOP - Not sleeping! " ) ;
2022-12-05 19:55:50 +00:00
}
2022-12-11 20:49:01 +00:00
// 5. End loop
2022-12-05 19:55:50 +00:00
}
2022-12-06 22:48:48 +00:00
} ) ;
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-23 18:33:40 +00:00
pub mini : String , // Did the user start on the mini-chain?
2022-12-06 20:54:18 +00:00
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?
}
2022-12-18 01:51:50 +00:00
impl Default for ImgP2pool {
fn default ( ) -> Self {
Self ::new ( )
}
}
2022-12-06 22:48:48 +00:00
impl ImgP2pool {
pub fn new ( ) -> Self {
2022-12-06 20:54:18 +00:00
Self {
2022-12-23 18:33:40 +00:00
mini : String ::from ( " ??? " ) ,
address : String ::from ( " ??? " ) ,
host : String ::from ( " ??? " ) ,
rpc : String ::from ( " ??? " ) ,
zmq : String ::from ( " ??? " ) ,
out_peers : String ::from ( " ??? " ) ,
in_peers : String ::from ( " ??? " ) ,
2022-12-06 20:54:18 +00:00
}
}
}
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-18 17:07:09 +00:00
#[ derive(Debug,Clone,PartialEq) ]
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 ,
2022-12-26 17:18:57 +00:00
// Local API
2022-12-04 01:12:40 +00:00
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-12-31 00:22:43 +00:00
// The API needs a raw ints to go off of and
2022-12-27 14:22:46 +00:00
// there's not a good way to access it without doing weird
2022-12-31 00:22:43 +00:00
// [Arc<Mutex>] shenanigans, so some raw ints are stored here.
pub user_p2pool_hashrate_u64 : u64 ,
pub p2pool_difficulty_u64 : u64 ,
pub monero_difficulty_u64 : u64 ,
2023-01-01 20:33:55 +00:00
pub p2pool_hashrate_u64 : u64 ,
pub monero_hashrate_u64 : u64 ,
2023-01-01 14:46:23 +00:00
// Tick. Every loop this gets incremented.
// At 60, it indicated we should read the below API files.
pub tick : u8 ,
2022-12-26 17:18:57 +00:00
// Network API
2022-12-27 14:22:46 +00:00
pub monero_difficulty : HumanNumber , // e.g: [15,000,000]
pub monero_hashrate : HumanNumber , // e.g: [1.000 GH/s]
2022-12-31 00:22:43 +00:00
pub hash : String , // Current block hash
2022-12-27 14:22:46 +00:00
pub height : HumanNumber ,
2022-12-31 00:22:43 +00:00
pub reward : AtomicUnit ,
2022-12-26 17:18:57 +00:00
// Pool API
2022-12-27 14:22:46 +00:00
pub p2pool_difficulty : HumanNumber ,
pub p2pool_hashrate : HumanNumber ,
pub miners : HumanNumber , // Current amount of miners on P2Pool sidechain
// Mean (calcualted in functions, not serialized)
pub solo_block_mean : HumanTime , // Time it would take the user to find a solo block
pub p2pool_block_mean : HumanTime , // Time it takes the P2Pool sidechain to find a block
pub p2pool_share_mean : HumanTime , // Time it would take the user to find a P2Pool share
2022-12-27 16:15:14 +00:00
// Percent
pub p2pool_percent : HumanNumber , // Percentage of P2Pool hashrate capture of overall Monero hashrate.
pub user_p2pool_percent : HumanNumber , // How much percent the user's hashrate accounts for in P2Pool.
pub user_monero_percent : HumanNumber , // How much percent the user's hashrate accounts for in all of Monero hashrate.
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 ( ) ,
2023-01-01 14:46:23 +00:00
tick : 0 ,
2022-12-31 00:22:43 +00:00
user_p2pool_hashrate_u64 : 0 ,
p2pool_difficulty_u64 : 0 ,
monero_difficulty_u64 : 0 ,
2023-01-01 20:33:55 +00:00
p2pool_hashrate_u64 : 0 ,
monero_hashrate_u64 : 0 ,
2022-12-27 14:22:46 +00:00
monero_difficulty : HumanNumber ::unknown ( ) ,
monero_hashrate : HumanNumber ::unknown ( ) ,
hash : String ::from ( " ??? " ) ,
height : HumanNumber ::unknown ( ) ,
2022-12-31 00:22:43 +00:00
reward : AtomicUnit ::new ( ) ,
2022-12-27 14:22:46 +00:00
p2pool_difficulty : HumanNumber ::unknown ( ) ,
p2pool_hashrate : HumanNumber ::unknown ( ) ,
miners : HumanNumber ::unknown ( ) ,
solo_block_mean : HumanTime ::new ( ) ,
p2pool_block_mean : HumanTime ::new ( ) ,
p2pool_share_mean : HumanTime ::new ( ) ,
p2pool_percent : HumanNumber ::unknown ( ) ,
2022-12-27 16:15:14 +00:00
user_p2pool_percent : HumanNumber ::unknown ( ) ,
user_monero_percent : HumanNumber ::unknown ( ) ,
2022-12-04 01:12:40 +00:00
}
}
2022-12-08 17:29:38 +00:00
// The issue with just doing [gui_api = pub_api] is that values get overwritten.
// This doesn't matter for any of the values EXCEPT for the output, so we must
// manually append it instead of overwriting.
// This is used in the "helper" thread.
fn combine_gui_pub_api ( gui_api : & mut Self , pub_api : & mut Self ) {
2022-12-18 17:07:09 +00:00
let mut output = std ::mem ::take ( & mut gui_api . output ) ;
2022-12-08 17:29:38 +00:00
let buf = std ::mem ::take ( & mut pub_api . output ) ;
2022-12-18 17:07:09 +00:00
if ! buf . is_empty ( ) { output . push_str ( & buf ) ; }
2022-12-08 17:29:38 +00:00
* gui_api = Self {
output ,
2023-01-01 14:46:23 +00:00
tick : std ::mem ::take ( & mut gui_api . tick ) ,
2022-12-18 17:07:09 +00:00
.. pub_api . clone ( )
2022-12-08 17:29:38 +00:00
} ;
}
2022-12-06 20:17:37 +00:00
2022-12-27 14:22:46 +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.
2023-04-19 14:01:11 +00:00
fn calc_payouts_and_xmr ( output : & str ) -> ( u128 /* payout count */ , f64 /* total xmr */ ) {
let iter = P2POOL_REGEX . payout . find_iter ( output ) ;
2022-12-27 14:22:46 +00:00
let mut sum : f64 = 0.0 ;
let mut count : u128 = 0 ;
for i in iter {
2023-04-19 14:01:11 +00:00
if let Some ( word ) = P2POOL_REGEX . payout_float . find ( i . as_str ( ) ) {
2022-12-28 00:46:46 +00:00
match word . as_str ( ) . parse ::< f64 > ( ) {
Ok ( num ) = > { sum + = num ; count + = 1 ; } ,
Err ( e ) = > error! ( " P2Pool | Total XMR sum calculation error: [{}] " , e ) ,
}
2022-12-27 14:22:46 +00:00
}
}
( count , sum )
}
2022-12-08 17:29:38 +00:00
// Mutate "watchdog"'s [PubP2poolApi] with data the process output.
2023-04-14 16:12:35 +00:00
fn update_from_output (
public : & Arc < Mutex < Self > > ,
output_parse : & Arc < Mutex < String > > ,
output_pub : & Arc < Mutex < String > > ,
elapsed : std ::time ::Duration ,
process : & Arc < Mutex < Process > > ,
) {
2022-12-08 17:29:38 +00:00
// 1. Take the process's current output buffer and combine it with Pub (if not empty)
2022-12-29 03:03:45 +00:00
let mut output_pub = lock! ( output_pub ) ;
2022-12-14 22:37:29 +00:00
if ! output_pub . is_empty ( ) {
2022-12-29 03:03:45 +00:00
lock! ( public ) . output . push_str ( & std ::mem ::take ( & mut * output_pub ) ) ;
2022-12-08 17:29:38 +00:00
}
// 2. Parse the full STDOUT
2022-12-29 03:03:45 +00:00
let mut output_parse = lock! ( output_parse ) ;
2023-04-19 14:01:11 +00:00
let ( payouts_new , xmr_new ) = Self ::calc_payouts_and_xmr ( & output_parse ) ;
2023-04-14 15:29:45 +00:00
// Check for "SYNCHRONIZED" only if we aren't already.
2023-04-14 16:12:35 +00:00
if lock! ( process ) . state = = ProcessState ::Syncing {
2023-04-19 14:01:11 +00:00
if P2POOL_REGEX . synchronized . is_match ( & output_parse ) {
2023-04-14 16:12:35 +00:00
lock! ( process ) . state = ProcessState ::Alive ;
2023-04-14 15:29:45 +00:00
}
}
2022-12-14 22:37:29 +00:00
// 3. Throw away [output_parse]
output_parse . clear ( ) ;
drop ( output_parse ) ;
// 4. Add to current values
2022-12-29 03:03:45 +00:00
let mut public = lock! ( public ) ;
2022-12-18 17:07:09 +00:00
let ( payouts , xmr ) = ( public . payouts + payouts_new , public . xmr + xmr_new ) ;
2022-12-06 18:18:40 +00:00
2022-12-14 22:37:29 +00:00
// 5. Calculate hour/day/month given elapsed time
2022-12-06 20:17:37 +00:00
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 ;
2022-12-16 18:24:48 +00:00
let xmr_day = xmr_hour * 24.0 ;
let xmr_month = xmr_day * 30.0 ;
2022-12-06 18:18:40 +00:00
2022-12-19 01:17:32 +00:00
if payouts_new ! = 0 {
2022-12-18 17:07:09 +00:00
debug! ( " P2Pool Watchdog | New [Payout] found in output ... {} " , payouts_new ) ;
debug! ( " P2Pool Watchdog | Total [Payout] should be ... {} " , payouts ) ;
debug! ( " P2Pool Watchdog | Correct [Payout per] should be ... [{}/hour, {}/day, {}/month] " , payouts_hour , payouts_day , payouts_month ) ;
}
2022-12-19 01:17:32 +00:00
if xmr_new ! = 0.0 {
2022-12-18 17:07:09 +00:00
debug! ( " P2Pool Watchdog | New [XMR mined] found in output ... {} " , xmr_new ) ;
debug! ( " P2Pool Watchdog | Total [XMR mined] should be ... {} " , xmr ) ;
debug! ( " P2Pool Watchdog | Correct [XMR mined per] should be ... [{}/hour, {}/day, {}/month] " , xmr_hour , xmr_day , xmr_month ) ;
}
2022-12-14 22:37:29 +00:00
// 6. Mutate the struct with the new info
2022-12-06 18:18:40 +00:00
* public = Self {
2022-12-06 20:17:37 +00:00
uptime : HumanTime ::into_human ( elapsed ) ,
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 ,
2022-12-08 17:29:38 +00:00
.. std ::mem ::take ( & mut * public )
2022-12-05 19:55:50 +00:00
} ;
2022-12-06 18:18:40 +00:00
}
2022-12-26 17:18:57 +00:00
// Mutate [PubP2poolApi] with data from a [PrivP2poolLocalApi] and the process output.
2022-12-27 14:22:46 +00:00
fn update_from_local ( public : & Arc < Mutex < Self > > , local : PrivP2poolLocalApi ) {
2022-12-29 03:03:45 +00:00
let mut public = lock! ( public ) ;
2022-12-06 18:18:40 +00:00
* public = Self {
2022-12-27 14:22:46 +00:00
hashrate_15m : HumanNumber ::from_u64 ( local . hashrate_15m ) ,
hashrate_1h : HumanNumber ::from_u64 ( local . hashrate_1h ) ,
hashrate_24h : HumanNumber ::from_u64 ( local . hashrate_24h ) ,
shares_found : HumanNumber ::from_u64 ( local . shares_found ) ,
average_effort : HumanNumber ::to_percent ( local . average_effort ) ,
current_effort : HumanNumber ::to_percent ( local . current_effort ) ,
2023-03-30 14:17:04 +00:00
connections : HumanNumber ::from_u32 ( local . connections ) ,
2022-12-31 00:22:43 +00:00
user_p2pool_hashrate_u64 : local . hashrate_1h ,
2022-12-12 01:43:34 +00:00
.. std ::mem ::take ( & mut * public )
} ;
2022-12-05 19:55:50 +00:00
}
2022-12-27 14:22:46 +00:00
// Mutate [PubP2poolApi] with data from a [PrivP2pool(Network|Pool)Api].
fn update_from_network_pool ( public : & Arc < Mutex < Self > > , net : PrivP2poolNetworkApi , pool : PrivP2poolPoolApi ) {
2022-12-31 00:22:43 +00:00
let user_hashrate = lock! ( public ) . user_p2pool_hashrate_u64 ; // The user's total P2Pool hashrate
2022-12-27 14:22:46 +00:00
let monero_difficulty = net . difficulty ;
let monero_hashrate = monero_difficulty / MONERO_BLOCK_TIME_IN_SECONDS ;
let p2pool_hashrate = pool . pool_statistics . hashRate ;
let p2pool_difficulty = p2pool_hashrate * P2POOL_BLOCK_TIME_IN_SECONDS ;
// These [0] checks prevent dividing by 0 (it [panic!()]s)
2022-12-27 16:15:14 +00:00
let p2pool_block_mean ;
let user_p2pool_percent ;
if p2pool_hashrate = = 0 {
p2pool_block_mean = HumanTime ::new ( ) ;
user_p2pool_percent = HumanNumber ::unknown ( ) ;
2022-12-27 14:22:46 +00:00
} else {
2022-12-27 16:15:14 +00:00
p2pool_block_mean = HumanTime ::into_human ( std ::time ::Duration ::from_secs ( monero_difficulty / p2pool_hashrate ) ) ;
2023-01-01 20:33:55 +00:00
let f = ( user_hashrate as f64 / p2pool_hashrate as f64 ) * 100.0 ;
user_p2pool_percent = HumanNumber ::from_f64_to_percent_6_point ( f ) ;
2022-12-27 14:22:46 +00:00
} ;
2022-12-27 16:15:14 +00:00
let p2pool_percent ;
let user_monero_percent ;
if monero_hashrate = = 0 {
p2pool_percent = HumanNumber ::unknown ( ) ;
user_monero_percent = HumanNumber ::unknown ( ) ;
2022-12-27 14:22:46 +00:00
} else {
2023-01-01 20:33:55 +00:00
let f = ( p2pool_hashrate as f64 / monero_hashrate as f64 ) * 100.0 ;
p2pool_percent = HumanNumber ::from_f64_to_percent_6_point ( f ) ;
let f = ( user_hashrate as f64 / monero_hashrate as f64 ) * 100.0 ;
user_monero_percent = HumanNumber ::from_f64_to_percent_6_point ( f ) ;
2022-12-27 14:22:46 +00:00
} ;
let solo_block_mean ;
let p2pool_share_mean ;
2022-12-27 16:15:14 +00:00
if user_hashrate = = 0 {
2022-12-27 14:22:46 +00:00
solo_block_mean = HumanTime ::new ( ) ;
p2pool_share_mean = HumanTime ::new ( ) ;
} else {
2022-12-27 16:15:14 +00:00
solo_block_mean = HumanTime ::into_human ( std ::time ::Duration ::from_secs ( monero_difficulty / user_hashrate ) ) ;
p2pool_share_mean = HumanTime ::into_human ( std ::time ::Duration ::from_secs ( p2pool_difficulty / user_hashrate ) ) ;
2022-12-05 19:55:50 +00:00
}
2022-12-29 03:03:45 +00:00
let mut public = lock! ( public ) ;
2022-12-27 14:22:46 +00:00
* public = Self {
2022-12-31 00:22:43 +00:00
p2pool_difficulty_u64 : p2pool_difficulty ,
monero_difficulty_u64 : monero_difficulty ,
2023-01-01 20:33:55 +00:00
p2pool_hashrate_u64 : p2pool_hashrate ,
monero_hashrate_u64 : monero_hashrate ,
2022-12-27 14:22:46 +00:00
monero_difficulty : HumanNumber ::from_u64 ( monero_difficulty ) ,
monero_hashrate : HumanNumber ::from_u64_to_gigahash_3_point ( monero_hashrate ) ,
hash : net . hash ,
height : HumanNumber ::from_u32 ( net . height ) ,
2022-12-31 00:22:43 +00:00
reward : AtomicUnit ::from_u64 ( net . reward ) ,
2022-12-27 14:22:46 +00:00
p2pool_difficulty : HumanNumber ::from_u64 ( p2pool_difficulty ) ,
p2pool_hashrate : HumanNumber ::from_u64_to_megahash_3_point ( p2pool_hashrate ) ,
miners : HumanNumber ::from_u32 ( pool . pool_statistics . miners ) ,
solo_block_mean ,
p2pool_block_mean ,
p2pool_share_mean ,
p2pool_percent ,
2022-12-27 16:15:14 +00:00
user_p2pool_percent ,
user_monero_percent ,
2022-12-27 14:22:46 +00:00
.. std ::mem ::take ( & mut * public )
} ;
2022-12-05 19:55:50 +00:00
}
2022-12-31 00:22:43 +00:00
pub fn calculate_share_or_block_time ( hashrate : u64 , difficulty : u64 ) -> HumanTime {
if hashrate = = 0 {
HumanTime ::new ( )
} else {
HumanTime ::from_u64 ( difficulty / hashrate )
}
}
2023-01-01 14:46:23 +00:00
2023-01-01 20:33:55 +00:00
pub fn calculate_dominance ( my_hashrate : u64 , global_hashrate : u64 ) -> HumanNumber {
if global_hashrate = = 0 {
HumanNumber ::unknown ( )
} else {
let f = ( my_hashrate as f64 / global_hashrate as f64 ) * 100.0 ;
HumanNumber ::from_f64_to_percent_6_point ( f )
}
}
2023-01-01 14:46:23 +00:00
pub const fn calculate_tick_bar ( & self ) -> & 'static str {
// The stars are reduced by one because it takes a frame to render the stats.
// We want 0 stars at the same time stats are rendered, so it looks a little off here.
match self . tick {
1 = > " [ ] " ,
2 = > " [* ] " ,
3 = > " [** ] " ,
4 = > " [*** ] " ,
5 = > " [**** ] " ,
6 = > " [***** ] " ,
7 = > " [****** ] " ,
8 = > " [******* ] " ,
9 = > " [******** ] " ,
10 = > " [********* ] " ,
11 = > " [********** ] " ,
12 = > " [*********** ] " ,
13 = > " [************ ] " ,
14 = > " [************* ] " ,
15 = > " [************** ] " ,
16 = > " [*************** ] " ,
17 = > " [**************** ] " ,
18 = > " [***************** ] " ,
19 = > " [****************** ] " ,
20 = > " [******************* ] " ,
21 = > " [******************** ] " ,
22 = > " [********************* ] " ,
23 = > " [********************** ] " ,
24 = > " [*********************** ] " ,
25 = > " [************************ ] " ,
26 = > " [************************* ] " ,
27 = > " [************************** ] " ,
28 = > " [*************************** ] " ,
29 = > " [**************************** ] " ,
30 = > " [***************************** ] " ,
31 = > " [****************************** ] " ,
32 = > " [******************************* ] " ,
33 = > " [******************************** ] " ,
34 = > " [********************************* ] " ,
35 = > " [********************************** ] " ,
36 = > " [*********************************** ] " ,
37 = > " [************************************ ] " ,
38 = > " [************************************* ] " ,
39 = > " [************************************** ] " ,
40 = > " [*************************************** ] " ,
41 = > " [**************************************** ] " ,
42 = > " [***************************************** ] " ,
43 = > " [****************************************** ] " ,
44 = > " [******************************************* ] " ,
45 = > " [******************************************** ] " ,
46 = > " [********************************************* ] " ,
47 = > " [********************************************** ] " ,
48 = > " [*********************************************** ] " ,
49 = > " [************************************************ ] " ,
50 = > " [************************************************* ] " ,
51 = > " [************************************************** ] " ,
52 = > " [*************************************************** ] " ,
53 = > " [**************************************************** ] " ,
54 = > " [***************************************************** ] " ,
55 = > " [****************************************************** ] " ,
56 = > " [******************************************************* ] " ,
57 = > " [******************************************************** ] " ,
58 = > " [********************************************************* ] " ,
59 = > " [********************************************************** ] " ,
60 = > " [*********************************************************** ] " ,
_ = > " [************************************************************] " ,
}
}
2022-11-30 22:21:55 +00:00
}
2022-12-26 17:18:57 +00:00
//---------------------------------------------------------------------------------------------------- Private P2Pool "Local" Api
2023-03-30 13:45:21 +00:00
// This matches directly to P2Pool's [local/stratum] JSON API file (excluding a few stats).
2022-12-02 04:13:53 +00:00
// 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-26 17:18:57 +00:00
struct PrivP2poolLocalApi {
2022-12-27 14:22:46 +00:00
hashrate_15m : u64 ,
hashrate_1h : u64 ,
hashrate_24h : u64 ,
shares_found : u64 ,
2022-12-04 01:12:40 +00:00
average_effort : f32 ,
current_effort : f32 ,
2023-03-30 13:45:21 +00:00
connections : u32 , // This is a `uint32_t` in `p2pool`
2022-12-02 04:13:53 +00:00
}
2022-11-30 22:21:55 +00:00
2022-12-26 17:18:57 +00:00
impl Default for PrivP2poolLocalApi { fn default ( ) -> Self { Self ::new ( ) } }
impl PrivP2poolLocalApi {
2022-12-02 04:13:53 +00:00
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-12-11 20:49:01 +00:00
2022-12-26 17:18:57 +00:00
// Deserialize the above [String] into a [PrivP2poolApi]
fn from_str ( string : & str ) -> std ::result ::Result < Self , serde_json ::Error > {
match serde_json ::from_str ::< Self > ( string ) {
Ok ( a ) = > Ok ( a ) ,
Err ( e ) = > { warn! ( " P2Pool Local API | Could not deserialize API data: {} " , e ) ; Err ( e ) } ,
2022-12-11 20:49:01 +00:00
}
}
2022-12-26 17:18:57 +00:00
}
2022-12-11 20:49:01 +00:00
2022-12-26 17:18:57 +00:00
//---------------------------------------------------------------------------------------------------- Private P2Pool "Network" API
// This matches P2Pool's [network/stats] JSON API file.
#[ derive(Debug, Serialize, Deserialize, Clone) ]
struct PrivP2poolNetworkApi {
2022-12-27 14:22:46 +00:00
difficulty : u64 ,
2022-12-26 17:18:57 +00:00
hash : String ,
height : u32 ,
2022-12-27 14:22:46 +00:00
reward : u64 ,
2022-12-26 17:18:57 +00:00
timestamp : u32 ,
}
impl Default for PrivP2poolNetworkApi { fn default ( ) -> Self { Self ::new ( ) } }
impl PrivP2poolNetworkApi {
fn new ( ) -> Self {
Self {
difficulty : 0 ,
hash : String ::from ( " ??? " ) ,
height : 0 ,
reward : 0 ,
timestamp : 0 ,
}
}
fn from_str ( string : & str ) -> std ::result ::Result < Self , serde_json ::Error > {
2022-12-11 20:49:01 +00:00
match serde_json ::from_str ::< Self > ( string ) {
Ok ( a ) = > Ok ( a ) ,
2022-12-26 17:18:57 +00:00
Err ( e ) = > { warn! ( " P2Pool Network API | Could not deserialize API data: {} " , e ) ; Err ( e ) } ,
2022-12-11 20:49:01 +00:00
}
}
2022-11-30 22:21:55 +00:00
}
2022-12-26 17:18:57 +00:00
//---------------------------------------------------------------------------------------------------- Private P2Pool "Pool" API
// This matches P2Pool's [pool/stats] JSON API file.
#[ derive(Debug, Serialize, Deserialize, Clone, Copy) ]
struct PrivP2poolPoolApi {
pool_statistics : PoolStatistics ,
}
impl Default for PrivP2poolPoolApi { fn default ( ) -> Self { Self ::new ( ) } }
impl PrivP2poolPoolApi {
fn new ( ) -> Self {
Self {
pool_statistics : PoolStatistics ::new ( ) ,
}
}
fn from_str ( string : & str ) -> std ::result ::Result < Self , serde_json ::Error > {
match serde_json ::from_str ::< Self > ( string ) {
Ok ( a ) = > Ok ( a ) ,
Err ( e ) = > { warn! ( " P2Pool Pool API | Could not deserialize API data: {} " , e ) ; Err ( e ) } ,
}
}
}
2022-12-26 22:37:45 +00:00
#[ allow(non_snake_case) ]
2022-12-26 17:18:57 +00:00
#[ derive(Debug, Serialize, Deserialize, Clone, Copy) ]
struct PoolStatistics {
2022-12-27 14:22:46 +00:00
hashRate : u64 ,
2022-12-26 17:18:57 +00:00
miners : u32 ,
}
impl Default for PoolStatistics { fn default ( ) -> Self { Self ::new ( ) } }
2022-12-26 19:22:05 +00:00
impl PoolStatistics { fn new ( ) -> Self { Self { hashRate : 0 , miners : 0 } } }
2022-12-26 17:18:57 +00:00
2022-12-06 22:48:48 +00:00
//---------------------------------------------------------------------------------------------------- [ImgXmrig]
#[ derive(Debug, Clone) ]
2022-12-08 17:29:38 +00:00
pub struct ImgXmrig {
pub threads : String ,
pub url : String ,
}
2022-12-18 01:51:50 +00:00
impl Default for ImgXmrig {
fn default ( ) -> Self {
Self ::new ( )
}
}
2022-12-06 22:48:48 +00:00
impl ImgXmrig {
pub fn new ( ) -> Self {
2022-12-08 17:29:38 +00:00
Self {
2022-12-23 18:33:40 +00:00
threads : " ??? " . to_string ( ) ,
url : " ??? " . to_string ( ) ,
2022-12-08 17:29:38 +00:00
}
2022-12-06 22:48:48 +00:00
}
}
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-08 17:29:38 +00:00
pub output : String ,
pub uptime : HumanTime ,
pub worker_id : String ,
pub resources : HumanNumber ,
pub hashrate : HumanNumber ,
pub diff : HumanNumber ,
pub accepted : HumanNumber ,
pub rejected : HumanNumber ,
2023-03-17 20:12:06 +00:00
pub hashrate_raw : f32 ,
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-08 17:29:38 +00:00
uptime : HumanTime ::new ( ) ,
2022-12-04 01:12:40 +00:00
worker_id : " ??? " . to_string ( ) ,
resources : HumanNumber ::unknown ( ) ,
hashrate : HumanNumber ::unknown ( ) ,
diff : HumanNumber ::unknown ( ) ,
accepted : HumanNumber ::unknown ( ) ,
rejected : HumanNumber ::unknown ( ) ,
2023-03-17 20:12:06 +00:00
hashrate_raw : 0.0 ,
2022-12-04 01:12:40 +00:00
}
}
2022-12-08 17:29:38 +00:00
fn combine_gui_pub_api ( gui_api : & mut Self , pub_api : & mut Self ) {
let output = std ::mem ::take ( & mut gui_api . output ) ;
let buf = std ::mem ::take ( & mut pub_api . output ) ;
* gui_api = Self {
output ,
2022-12-11 20:49:01 +00:00
.. std ::mem ::take ( pub_api )
2022-12-08 17:29:38 +00:00
} ;
if ! buf . is_empty ( ) { gui_api . output . push_str ( & buf ) ; }
}
2022-12-14 22:37:29 +00:00
// This combines the buffer from the PTY thread [output_pub]
// with the actual [PubApiXmrig] output field.
2023-04-19 14:18:04 +00:00
fn update_from_output (
public : & Arc < Mutex < Self > > ,
output_parse : & Arc < Mutex < String > > ,
output_pub : & Arc < Mutex < String > > ,
elapsed : std ::time ::Duration ,
process : & Arc < Mutex < Process > > ,
) {
// 1. Take the process's current output buffer and combine it with Pub (if not empty)
2022-12-29 03:03:45 +00:00
let mut output_pub = lock! ( output_pub ) ;
2022-12-14 22:37:29 +00:00
if ! output_pub . is_empty ( ) {
2023-04-19 14:18:04 +00:00
let mut public = lock! ( public ) ;
2022-12-14 22:37:29 +00:00
public . output . push_str ( & std ::mem ::take ( & mut * output_pub ) ) ;
2023-04-19 14:18:04 +00:00
// Update uptime
public . uptime = HumanTime ::into_human ( elapsed ) ;
}
// 2. Check for "new job"/"no active...".
let mut output_parse = lock! ( output_parse ) ;
if XMRIG_REGEX . new_job . is_match ( & output_parse ) {
lock! ( process ) . state = ProcessState ::Alive ;
} else if XMRIG_REGEX . not_mining . is_match ( & output_parse ) {
lock! ( process ) . state = ProcessState ::NotMining ;
2022-12-08 17:29:38 +00:00
}
2023-04-19 14:18:04 +00:00
// 3. Throw away [output_parse]
output_parse . clear ( ) ;
drop ( output_parse ) ;
2022-12-08 17:29:38 +00:00
}
2022-12-04 01:12:40 +00:00
// Formats raw private data into ready-to-print human readable version.
2022-12-12 03:01:37 +00:00
fn update_from_priv ( public : & Arc < Mutex < Self > > , private : PrivXmrigApi ) {
2022-12-29 03:03:45 +00:00
let mut public = lock! ( public ) ;
2023-03-17 20:12:06 +00:00
let hashrate_raw = match private . hashrate . total . get ( 0 ) {
Some ( Some ( h ) ) = > * h ,
_ = > 0.0 ,
} ;
2022-12-11 20:49:01 +00:00
* public = Self {
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 ) ,
diff : HumanNumber ::from_u128 ( private . connection . diff ) ,
accepted : HumanNumber ::from_u128 ( private . connection . accepted ) ,
rejected : HumanNumber ::from_u128 ( private . connection . rejected ) ,
2023-03-17 20:12:06 +00:00
hashrate_raw ,
2022-12-12 03:01:37 +00:00
.. std ::mem ::take ( & mut * public )
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-11 20:49:01 +00:00
// Send an HTTP request to XMRig's API, serialize it into [Self] and return it
2022-12-25 18:30:27 +00:00
async fn request_xmrig_api ( client : hyper ::Client < hyper ::client ::HttpConnector > , api_uri : & str ) -> std ::result ::Result < Self , anyhow ::Error > {
2022-12-11 20:49:01 +00:00
let request = hyper ::Request ::builder ( )
. method ( " GET " )
2022-12-19 14:31:11 +00:00
. uri ( api_uri )
2022-12-11 20:49:01 +00:00
. body ( hyper ::Body ::empty ( ) ) ? ;
2022-12-14 03:41:05 +00:00
let response = tokio ::time ::timeout ( std ::time ::Duration ::from_millis ( 500 ) , client . request ( request ) ) . await ? ;
2022-12-11 20:49:01 +00:00
let body = hyper ::body ::to_bytes ( response ? . body_mut ( ) ) . await ? ;
Ok ( serde_json ::from_slice ::< Self > ( & body ) ? )
}
2022-12-02 04:13:53 +00:00
}
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 {
diff : u128 ,
accepted : u128 ,
rejected : u128 ,
}
impl Connection {
fn new ( ) -> Self {
Self {
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 ) ] ,
}
}
}
2022-12-17 22:17:26 +00:00
//---------------------------------------------------------------------------------------------------- TESTS
#[ cfg(test) ]
mod test {
2023-04-14 17:45:30 +00:00
use super ::* ;
2022-12-18 02:42:30 +00:00
#[ test ]
fn reset_gui_output ( ) {
let max = crate ::helper ::GUI_OUTPUT_LEEWAY ;
let mut string = String ::with_capacity ( max ) ;
for _ in 0 ..= max {
string . push ( '0' ) ;
}
crate ::Helper ::check_reset_gui_output ( & mut string , crate ::ProcessName ::P2pool ) ;
// Some text gets added, so just check for less than 500 bytes.
assert! ( string . len ( ) < 500 ) ;
}
2022-12-18 17:07:09 +00:00
#[ test ]
fn combine_gui_pub_p2pool_api ( ) {
2023-04-19 14:01:11 +00:00
use crate ::helper ::PubP2poolApi ;
let mut gui_api = PubP2poolApi ::new ( ) ;
let mut pub_api = PubP2poolApi ::new ( ) ;
pub_api . payouts = 1 ;
pub_api . payouts_hour = 2.0 ;
pub_api . payouts_day = 3.0 ;
pub_api . payouts_month = 4.0 ;
pub_api . xmr = 1.0 ;
pub_api . xmr_hour = 2.0 ;
pub_api . xmr_day = 3.0 ;
pub_api . xmr_month = 4.0 ;
println! ( " BEFORE - GUI_API: {:#?} \n PUB_API: {:#?} " , gui_api , pub_api ) ;
assert_ne! ( gui_api , pub_api ) ;
PubP2poolApi ::combine_gui_pub_api ( & mut gui_api , & mut pub_api ) ;
println! ( " AFTER - GUI_API: {:#?} \n PUB_API: {:#?} " , gui_api , pub_api ) ;
assert_eq! ( gui_api , pub_api ) ;
pub_api . xmr = 2.0 ;
PubP2poolApi ::combine_gui_pub_api ( & mut gui_api , & mut pub_api ) ;
assert_eq! ( gui_api , pub_api ) ;
assert_eq! ( gui_api . xmr , 2.0 ) ;
assert_eq! ( pub_api . xmr , 2.0 ) ;
2022-12-18 01:51:50 +00:00
}
2022-12-17 22:17:26 +00:00
#[ test ]
fn calc_payouts_and_xmr_from_output_p2pool ( ) {
2023-04-19 14:01:11 +00:00
use crate ::helper ::{ PubP2poolApi } ;
2022-12-17 22:17:26 +00:00
use std ::sync ::{ Arc , Mutex } ;
let public = Arc ::new ( Mutex ::new ( PubP2poolApi ::new ( ) ) ) ;
let output_parse = Arc ::new ( Mutex ::new ( String ::from (
2022-12-19 14:31:11 +00:00
r #" payout of 5.000000000001 XMR in block 1111
payout of 5.000000000001 XMR in block 1112
payout of 5.000000000001 XMR in block 1113 " #
2022-12-17 22:17:26 +00:00
) ) ) ;
let output_pub = Arc ::new ( Mutex ::new ( String ::new ( ) ) ) ;
let elapsed = std ::time ::Duration ::from_secs ( 60 ) ;
2023-04-14 17:45:30 +00:00
let process = Arc ::new ( Mutex ::new ( Process ::new ( ProcessName ::P2pool , " " . to_string ( ) , PathBuf ::new ( ) ) ) ) ;
2023-04-19 14:01:11 +00:00
PubP2poolApi ::update_from_output ( & public , & output_parse , & output_pub , elapsed , & process ) ;
2022-12-17 22:17:26 +00:00
let public = public . lock ( ) . unwrap ( ) ;
println! ( " {:#?} " , public ) ;
assert_eq! ( public . payouts , 3 ) ;
assert_eq! ( public . payouts_hour , 180.0 ) ;
assert_eq! ( public . payouts_day , 4320.0 ) ;
assert_eq! ( public . payouts_month , 129600.0 ) ;
assert_eq! ( public . xmr , 15.000000000003 ) ;
assert_eq! ( public . xmr_hour , 900.00000000018 ) ;
assert_eq! ( public . xmr_day , 21600.00000000432 ) ;
assert_eq! ( public . xmr_month , 648000.0000001296 ) ;
}
2023-04-14 17:45:30 +00:00
#[ test ]
fn set_p2pool_synchronized ( ) {
2023-04-19 14:01:11 +00:00
use crate ::helper ::{ PubP2poolApi } ;
2023-04-14 17:45:30 +00:00
use std ::sync ::{ Arc , Mutex } ;
let public = Arc ::new ( Mutex ::new ( PubP2poolApi ::new ( ) ) ) ;
let output_parse = Arc ::new ( Mutex ::new ( String ::from (
r #" payout of 5.000000000001 XMR in block 1111
NOTICE 2021 - 12 - 27 21 :42 :17.2008 SideChain SYNCHRONIZED
payout of 5.000000000001 XMR in block 1113 " #
) ) ) ;
let output_pub = Arc ::new ( Mutex ::new ( String ::new ( ) ) ) ;
let elapsed = std ::time ::Duration ::from_secs ( 60 ) ;
let process = Arc ::new ( Mutex ::new ( Process ::new ( ProcessName ::P2pool , " " . to_string ( ) , PathBuf ::new ( ) ) ) ) ;
// It only gets checked if we're `Syncing`.
process . lock ( ) . unwrap ( ) . state = ProcessState ::Syncing ;
2023-04-19 14:01:11 +00:00
PubP2poolApi ::update_from_output ( & public , & output_parse , & output_pub , elapsed , & process ) ;
2023-04-14 17:45:30 +00:00
println! ( " {:#?} " , process ) ;
assert! ( process . lock ( ) . unwrap ( ) . state = = ProcessState ::Alive ) ;
}
2022-12-27 14:22:46 +00:00
#[ test ]
fn update_pub_p2pool_from_local_network_pool ( ) {
use std ::sync ::{ Arc , Mutex } ;
use crate ::helper ::PubP2poolApi ;
use crate ::helper ::PrivP2poolLocalApi ;
use crate ::helper ::PrivP2poolNetworkApi ;
use crate ::helper ::PrivP2poolPoolApi ;
use crate ::helper ::PoolStatistics ;
let public = Arc ::new ( Mutex ::new ( PubP2poolApi ::new ( ) ) ) ;
let local = PrivP2poolLocalApi {
hashrate_15m : 10_000 ,
hashrate_1h : 20_000 ,
hashrate_24h : 30_000 ,
shares_found : 1000 ,
average_effort : 100.000 ,
current_effort : 200.000 ,
connections : 1234 ,
} ;
let network = PrivP2poolNetworkApi {
difficulty : 300_000_000_000 ,
hash : " asdf " . to_string ( ) ,
height : 1234 ,
reward : 2345 ,
timestamp : 3456 ,
} ;
let pool = PrivP2poolPoolApi {
pool_statistics : PoolStatistics {
hashRate : 1_000_000 , // 1 MH/s
miners : 1_000 ,
}
} ;
// Update Local
PubP2poolApi ::update_from_local ( & public , local ) ;
let p = public . lock ( ) . unwrap ( ) ;
println! ( " AFTER LOCAL: {:#?} " , p ) ;
assert_eq! ( p . hashrate_15m . to_string ( ) , " 10,000 " ) ;
assert_eq! ( p . hashrate_1h . to_string ( ) , " 20,000 " ) ;
assert_eq! ( p . hashrate_24h . to_string ( ) , " 30,000 " ) ;
assert_eq! ( p . shares_found . to_string ( ) , " 1,000 " ) ;
assert_eq! ( p . average_effort . to_string ( ) , " 100.00% " ) ;
assert_eq! ( p . current_effort . to_string ( ) , " 200.00% " ) ;
assert_eq! ( p . connections . to_string ( ) , " 1,234 " ) ;
2022-12-31 00:22:43 +00:00
assert_eq! ( p . user_p2pool_hashrate_u64 , 20000 ) ;
2022-12-27 14:22:46 +00:00
drop ( p ) ;
// Update Network + Pool
PubP2poolApi ::update_from_network_pool ( & public , network , pool ) ;
let p = public . lock ( ) . unwrap ( ) ;
println! ( " AFTER NETWORK+POOL: {:#?} " , p ) ;
2022-12-27 16:15:14 +00:00
assert_eq! ( p . monero_difficulty . to_string ( ) , " 300,000,000,000 " ) ;
assert_eq! ( p . monero_hashrate . to_string ( ) , " 2.500 GH/s " ) ;
assert_eq! ( p . hash . to_string ( ) , " asdf " ) ;
assert_eq! ( p . height . to_string ( ) , " 1,234 " ) ;
2022-12-31 00:22:43 +00:00
assert_eq! ( p . reward . to_u64 ( ) , 2345 ) ;
2022-12-27 16:15:14 +00:00
assert_eq! ( p . p2pool_difficulty . to_string ( ) , " 10,000,000 " ) ;
assert_eq! ( p . p2pool_hashrate . to_string ( ) , " 1.000 MH/s " ) ;
assert_eq! ( p . miners . to_string ( ) , " 1,000 " ) ;
assert_eq! ( p . solo_block_mean . to_string ( ) , " 5 months, 21 days, 9 hours, 52 minutes " ) ;
assert_eq! ( p . p2pool_block_mean . to_string ( ) , " 3 days, 11 hours, 20 minutes " ) ;
assert_eq! ( p . p2pool_share_mean . to_string ( ) , " 8 minutes, 20 seconds " ) ;
2023-01-01 23:57:11 +00:00
assert_eq! ( p . p2pool_percent . to_string ( ) , " 0.040000% " ) ;
assert_eq! ( p . user_p2pool_percent . to_string ( ) , " 2.000000% " ) ;
assert_eq! ( p . user_monero_percent . to_string ( ) , " 0.000800% " ) ;
2022-12-27 14:22:46 +00:00
drop ( p ) ;
}
2023-04-19 15:31:36 +00:00
#[ test ]
fn set_xmrig_mining ( ) {
use crate ::helper ::PubXmrigApi ;
use std ::sync ::{ Arc , Mutex } ;
let public = Arc ::new ( Mutex ::new ( PubXmrigApi ::new ( ) ) ) ;
let output_parse = Arc ::new ( Mutex ::new ( String ::from ( " [2022-02-12 12:49:30.311] net no active pools, stop mining " ) ) ) ;
let output_pub = Arc ::new ( Mutex ::new ( String ::new ( ) ) ) ;
let elapsed = std ::time ::Duration ::from_secs ( 60 ) ;
let process = Arc ::new ( Mutex ::new ( Process ::new ( ProcessName ::Xmrig , " " . to_string ( ) , PathBuf ::new ( ) ) ) ) ;
process . lock ( ) . unwrap ( ) . state = ProcessState ::Alive ;
PubXmrigApi ::update_from_output ( & public , & output_parse , & output_pub , elapsed , & process ) ;
println! ( " {:#?} " , process ) ;
assert! ( process . lock ( ) . unwrap ( ) . state = = ProcessState ::NotMining ) ;
let output_parse = Arc ::new ( Mutex ::new ( String ::from ( " [2022-02-12 12:49:30.311] net new job from 192.168.2.1:3333 diff 402K algo rx/0 height 2241142 (11 tx) " ) ) ) ;
PubXmrigApi ::update_from_output ( & public , & output_parse , & output_pub , elapsed , & process ) ;
assert! ( process . lock ( ) . unwrap ( ) . state = = ProcessState ::Alive ) ;
}
2022-12-17 22:17:26 +00:00
#[ test ]
2022-12-26 17:18:57 +00:00
fn serde_priv_p2pool_local_api ( ) {
2022-12-17 22:17:26 +00:00
let data =
r #" {
" hashrate_15m " : 12 ,
" hashrate_1h " : 11111 ,
" hashrate_24h " : 468967 ,
" total_hashes " : 2019283840922394082390 ,
" shares_found " : 289037 ,
" average_effort " : 915.563 ,
" current_effort " : 129.297 ,
" connections " : 123 ,
" incoming_connections " : 96
} " #;
2022-12-26 19:22:05 +00:00
let priv_api = crate ::helper ::PrivP2poolLocalApi ::from_str ( data ) . unwrap ( ) ;
2022-12-17 22:17:26 +00:00
let json = serde_json ::ser ::to_string_pretty ( & priv_api ) . unwrap ( ) ;
println! ( " {} " , json ) ;
let data_after_ser =
r #" {
" hashrate_15m " : 12 ,
" hashrate_1h " : 11111 ,
" hashrate_24h " : 468967 ,
" shares_found " : 289037 ,
" average_effort " : 915.563 ,
" current_effort " : 129.297 ,
" connections " : 123
} " #;
assert_eq! ( data_after_ser , json )
}
2022-12-26 19:22:05 +00:00
#[ test ]
fn serde_priv_p2pool_network_api ( ) {
let data =
r #" {
" difficulty " : 319028180924 ,
" hash " : " 22ae1b83d727bb2ff4efc17b485bc47bc8bf5e29a7b3af65baf42213ac70a39b " ,
" height " : 2776576 ,
" reward " : 600499860000 ,
" timestamp " : 1670953659
} " #;
let priv_api = crate ::helper ::PrivP2poolNetworkApi ::from_str ( data ) . unwrap ( ) ;
let json = serde_json ::ser ::to_string_pretty ( & priv_api ) . unwrap ( ) ;
println! ( " {} " , json ) ;
let data_after_ser =
r #" {
" difficulty " : 319028180924 ,
" hash " : " 22ae1b83d727bb2ff4efc17b485bc47bc8bf5e29a7b3af65baf42213ac70a39b " ,
" height " : 2776576 ,
" reward " : 600499860000 ,
" timestamp " : 1670953659
} " #;
assert_eq! ( data_after_ser , json )
}
#[ test ]
fn serde_priv_p2pool_pool_api ( ) {
let data =
r #" {
" pool_list " : [ " pplns " ] ,
" pool_statistics " : {
" hashRate " : 10225772 ,
" miners " : 713 ,
" totalHashes " : 487463929193948 ,
" lastBlockFoundTime " : 1670453228 ,
" lastBlockFound " : 2756570 ,
" totalBlocksFound " : 4
}
} " #;
let priv_api = crate ::helper ::PrivP2poolPoolApi ::from_str ( data ) . unwrap ( ) ;
let json = serde_json ::ser ::to_string_pretty ( & priv_api ) . unwrap ( ) ;
println! ( " {} " , json ) ;
let data_after_ser =
r #" {
" pool_statistics " : {
" hashRate " : 10225772 ,
" miners " : 713
}
} " #;
assert_eq! ( data_after_ser , json )
}
2022-12-17 22:17:26 +00:00
#[ test ]
fn serde_priv_xmrig_api ( ) {
let data =
r #" {
" id " : " 6226e3sd0cd1a6es " ,
" worker_id " : " hinto " ,
" uptime " : 123 ,
" restricted " : true ,
" resources " : {
" memory " : {
" free " : 123 ,
" total " : 123123 ,
" resident_set_memory " : 123123123
} ,
" load_average " : [ 10.97 , 10.58 , 10.47 ] ,
" hardware_concurrency " : 12
} ,
" features " : [ " api " , " asm " , " http " , " hwloc " , " tls " , " opencl " , " cuda " ] ,
" results " : {
" diff_current " : 123 ,
" shares_good " : 123 ,
" shares_total " : 123 ,
" avg_time " : 123 ,
" avg_time_ms " : 123 ,
" hashes_total " : 123 ,
" best " : [ 123 , 123 , 123 , 13 , 123 , 123 , 123 , 123 , 123 , 123 ] ,
" error_log " : [ ]
} ,
" algo " : " rx/0 " ,
" connection " : {
" pool " : " localhost:3333 " ,
" ip " : " 127.0.0.1 " ,
" uptime " : 123 ,
" uptime_ms " : 123 ,
" ping " : 0 ,
" failures " : 0 ,
" tls " : null ,
" tls-fingerprint " : null ,
" algo " : " rx/0 " ,
" diff " : 123 ,
" accepted " : 123 ,
" rejected " : 123 ,
" avg_time " : 123 ,
" avg_time_ms " : 123 ,
" hashes_total " : 123 ,
" error_log " : [ ]
} ,
" version " : " 6.18.0 " ,
" kind " : " miner " ,
" ua " : " XMRig/6.18.0 (Linux x86_64) libuv/2.0.0-dev gcc/10.2.1 " ,
" cpu " : {
" brand " : " blah blah blah " ,
" family " : 1 ,
" model " : 2 ,
" stepping " : 0 ,
" proc_info " : 123 ,
" aes " : true ,
" avx2 " : true ,
" x64 " : true ,
" 64_bit " : true ,
" l2 " : 123123 ,
" l3 " : 123123 ,
" cores " : 12 ,
" threads " : 24 ,
" packages " : 1 ,
" nodes " : 1 ,
" backend " : " hwloc/2.8.0a1-git " ,
" msr " : " ryzen_19h " ,
" assembly " : " ryzen " ,
" arch " : " x86_64 " ,
" flags " : [ " aes " , " vaes " , " avx " , " avx2 " , " bmi2 " , " osxsave " , " pdpe1gb " , " sse2 " , " ssse3 " , " sse4.1 " , " popcnt " , " cat_l3 " ]
} ,
" donate_level " : 0 ,
" paused " : false ,
" algorithms " : [ " cn/1 " , " cn/2 " , " cn/r " , " cn/fast " , " cn/half " , " cn/xao " , " cn/rto " , " cn/rwz " , " cn/zls " , " cn/double " , " cn/ccx " , " cn-lite/1 " , " cn-heavy/0 " , " cn-heavy/tube " , " cn-heavy/xhv " , " cn-pico " , " cn-pico/tlo " , " cn/upx2 " , " rx/0 " , " rx/wow " , " rx/arq " , " rx/graft " , " rx/sfx " , " rx/keva " , " argon2/chukwa " , " argon2/chukwav2 " , " argon2/ninja " , " astrobwt " , " astrobwt/v2 " , " ghostrider " ] ,
" hashrate " : {
" total " : [ 111.11 , 111.11 , 111.11 ] ,
" highest " : 111.11 ,
" threads " : [
[ 111.11 , 111.11 , 111.11 ]
]
} ,
" hugepages " : true
} " #;
use crate ::helper ::PrivXmrigApi ;
let priv_api = serde_json ::from_str ::< PrivXmrigApi > ( & data ) . unwrap ( ) ;
let json = serde_json ::ser ::to_string_pretty ( & priv_api ) . unwrap ( ) ;
println! ( " {} " , json ) ;
let data_after_ser =
r #" {
" worker_id " : " hinto " ,
" resources " : {
" load_average " : [
10.97 ,
10.58 ,
10.47
]
} ,
" connection " : {
" diff " : 123 ,
" accepted " : 123 ,
" rejected " : 123
} ,
" hashrate " : {
" total " : [
111.11 ,
111.11 ,
111.11
]
}
} " #;
assert_eq! ( data_after_ser , json )
}
}