mirror of
https://github.com/Cyrix126/gupaxx.git
synced 2025-01-03 12:39:35 +00:00
feat: integrate algorithm of HR distribution
This commit is contained in:
parent
87eb738ca8
commit
74c6a92535
10 changed files with 496 additions and 123 deletions
|
@ -64,7 +64,7 @@ rand = "0.8.5"
|
||||||
regex = { version = "1.10.3", default-features = false, features = ["perf"] }
|
regex = { version = "1.10.3", default-features = false, features = ["perf"] }
|
||||||
rfd = "0.14.0"
|
rfd = "0.14.0"
|
||||||
serde = { version = "1.0.197", features = ["rc", "derive"] }
|
serde = { version = "1.0.197", features = ["rc", "derive"] }
|
||||||
serde_json = "1.0"
|
serde_json = "1.0.114"
|
||||||
sysinfo = { version = "0.30.5", default-features = false }
|
sysinfo = { version = "0.30.5", default-features = false }
|
||||||
tls-api = "0.9.0"
|
tls-api = "0.9.0"
|
||||||
tokio = { version = "1.36.0", features = ["rt", "time", "macros", "process"] }
|
tokio = { version = "1.36.0", features = ["rt", "time", "macros", "process"] }
|
||||||
|
|
|
@ -71,6 +71,27 @@ The mHR is calculated depending on the sidechain the p2pool is mining on.
|
||||||
The XvB process will check every ten minutes the last 15 minutes average HR and decide when to switch (in seconds) for the ten next minutes. (first p2pool then XvB).
|
The XvB process will check every ten minutes the last 15 minutes average HR and decide when to switch (in seconds) for the ten next minutes. (first p2pool then XvB).
|
||||||
*Need to see the time for Xmrig takes to set the new settings by API.*
|
*Need to see the time for Xmrig takes to set the new settings by API.*
|
||||||
When the time to switch arrives, XvB process will send a request to Xmrig to change the node used.
|
When the time to switch arrives, XvB process will send a request to Xmrig to change the node used.
|
||||||
|
### Modification of config of xmrig
|
||||||
|
|
||||||
|
The following 4 attributes must be applied to xmrig config when mining to XvB node.
|
||||||
|
|
||||||
|
```ignore
|
||||||
|
"url": "xvb node:4247"
|
||||||
|
"user": "user id",
|
||||||
|
"keepalive": true,
|
||||||
|
"tls": true,
|
||||||
|
```
|
||||||
|
Or to return back to p2pool
|
||||||
|
|
||||||
|
```ignore
|
||||||
|
"url": "127.0.0.1:3333"
|
||||||
|
"user": "Gupax_v1_3_5",
|
||||||
|
"keepalive": false,
|
||||||
|
"tls": false,
|
||||||
|
```
|
||||||
|
|
||||||
|
The HTTP API of xmrig requires to give a full config.
|
||||||
|
The current config will be requested, modified and sent back.
|
||||||
|
|
||||||
[^1]: https://p2pool.io/mini/api/pool/stats
|
[^1]: https://p2pool.io/mini/api/pool/stats
|
||||||
[^2]: https://github.com/SChernykh/p2pool?tab=readme-ov-file#how-payouts-work-in-p2pool
|
[^2]: https://github.com/SChernykh/p2pool?tab=readme-ov-file#how-payouts-work-in-p2pool
|
||||||
|
|
|
@ -535,7 +535,12 @@ impl crate::app::App {
|
||||||
.on_hover_text("Restart Xvb")
|
.on_hover_text("Restart Xvb")
|
||||||
.clicked()
|
.clicked()
|
||||||
{
|
{
|
||||||
Helper::restart_xvb(&self.helper, &self.state.xvb, &self.state.p2pool);
|
Helper::restart_xvb(
|
||||||
|
&self.helper,
|
||||||
|
&self.state.xvb,
|
||||||
|
&self.state.p2pool,
|
||||||
|
&self.state.xmrig,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
if key.is_down() && !wants_input
|
if key.is_down() && !wants_input
|
||||||
|| ui
|
|| ui
|
||||||
|
@ -569,7 +574,12 @@ impl crate::app::App {
|
||||||
.on_disabled_hover_text(XVB_NOT_CONFIGURED)
|
.on_disabled_hover_text(XVB_NOT_CONFIGURED)
|
||||||
.clicked()
|
.clicked()
|
||||||
{
|
{
|
||||||
Helper::start_xvb(&self.helper, &self.state.xvb, &self.state.p2pool);
|
Helper::start_xvb(
|
||||||
|
&self.helper,
|
||||||
|
&self.state.xvb,
|
||||||
|
&self.state.p2pool,
|
||||||
|
&self.state.xmrig,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -161,7 +161,7 @@ path_xmr: {:#?}\n
|
||||||
}
|
}
|
||||||
Tab::Xvb => {
|
Tab::Xvb => {
|
||||||
debug!("App | Entering [XvB] Tab");
|
debug!("App | Entering [XvB] Tab");
|
||||||
crate::disk::state::Xvb::show(&mut self.state.xvb, self.size, &self.state.p2pool.address, ctx, ui, &self.xvb_api, lock!(self.xvb).is_alive());
|
crate::disk::state::Xvb::show(&mut self.state.xvb, self.size, &self.state.p2pool.address, ctx, ui, &self.xvb_api, lock!(self.xvb).is_alive()&& !lock!(self.xvb).is_syncing() && !lock!(self.xvb).is_not_mining());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -3,6 +3,7 @@ use std::sync::{Arc, Mutex};
|
||||||
use egui::TextStyle::{self, Name};
|
use egui::TextStyle::{self, Name};
|
||||||
use egui::{vec2, Hyperlink, Image, RichText, TextEdit, Ui, Vec2};
|
use egui::{vec2, Hyperlink, Image, RichText, TextEdit, Ui, Vec2};
|
||||||
use log::debug;
|
use log::debug;
|
||||||
|
use readable::byte::Byte;
|
||||||
|
|
||||||
use crate::helper::xvb::PubXvbApi;
|
use crate::helper::xvb::PubXvbApi;
|
||||||
use crate::utils::constants::{
|
use crate::utils::constants::{
|
||||||
|
@ -26,7 +27,7 @@ impl crate::disk::state::Xvb {
|
||||||
_ctx: &egui::Context,
|
_ctx: &egui::Context,
|
||||||
ui: &mut egui::Ui,
|
ui: &mut egui::Ui,
|
||||||
api: &Arc<Mutex<PubXvbApi>>,
|
api: &Arc<Mutex<PubXvbApi>>,
|
||||||
xvb_is_alive: bool,
|
private_stats: bool,
|
||||||
) {
|
) {
|
||||||
let website_height = size.y / 10.0;
|
let website_height = size.y / 10.0;
|
||||||
let width = size.x;
|
let width = size.x;
|
||||||
|
@ -104,8 +105,8 @@ impl crate::disk::state::Xvb {
|
||||||
}
|
}
|
||||||
// private stats
|
// private stats
|
||||||
let priv_stats = &lock!(api).stats_priv;
|
let priv_stats = &lock!(api).stats_priv;
|
||||||
ui.set_enabled(xvb_is_alive);
|
|
||||||
// ui.vertical_centered(|ui| {
|
// ui.vertical_centered(|ui| {
|
||||||
|
ui.add_enabled_ui(private_stats, |ui| {
|
||||||
ui.add_space(SPACE * 2.0);
|
ui.add_space(SPACE * 2.0);
|
||||||
ui.horizontal(|ui| {
|
ui.horizontal(|ui| {
|
||||||
// widget takes a third less space for two separator.
|
// widget takes a third less space for two separator.
|
||||||
|
@ -135,7 +136,13 @@ impl crate::disk::state::Xvb {
|
||||||
ui.add_sized(size_stat, |ui: &mut Ui| {
|
ui.add_sized(size_stat, |ui: &mut Ui| {
|
||||||
ui.vertical_centered(|ui| {
|
ui.vertical_centered(|ui| {
|
||||||
ui.label(XVB_DONATED_1H_FIELD);
|
ui.label(XVB_DONATED_1H_FIELD);
|
||||||
ui.label(priv_stats.donor_1hr_avg.to_string());
|
ui.label(
|
||||||
|
[
|
||||||
|
Byte::from(priv_stats.donor_1hr_avg).to_string(),
|
||||||
|
"H/s".to_string(),
|
||||||
|
]
|
||||||
|
.concat(),
|
||||||
|
);
|
||||||
})
|
})
|
||||||
.response
|
.response
|
||||||
});
|
});
|
||||||
|
@ -143,7 +150,13 @@ impl crate::disk::state::Xvb {
|
||||||
ui.add_sized(size_stat, |ui: &mut Ui| {
|
ui.add_sized(size_stat, |ui: &mut Ui| {
|
||||||
ui.vertical_centered(|ui| {
|
ui.vertical_centered(|ui| {
|
||||||
ui.label(XVB_DONATED_24H_FIELD);
|
ui.label(XVB_DONATED_24H_FIELD);
|
||||||
ui.label(priv_stats.donor_24hr_avg.to_string());
|
ui.label(
|
||||||
|
[
|
||||||
|
Byte::from(priv_stats.donor_24hr_avg).to_string(),
|
||||||
|
"H/s".to_string(),
|
||||||
|
]
|
||||||
|
.concat(),
|
||||||
|
);
|
||||||
})
|
})
|
||||||
.response
|
.response
|
||||||
});
|
});
|
||||||
|
@ -156,7 +169,9 @@ impl crate::disk::state::Xvb {
|
||||||
})
|
})
|
||||||
.response
|
.response
|
||||||
})
|
})
|
||||||
.on_disabled_hover_text("You do not yet have a share in the PPLNS Window.");
|
.on_disabled_hover_text(
|
||||||
|
"You do not yet have a share in the PPLNS Window.",
|
||||||
|
);
|
||||||
});
|
});
|
||||||
ui.separator();
|
ui.separator();
|
||||||
ui.add_sized(size_stat, |ui: &mut Ui| {
|
ui.add_sized(size_stat, |ui: &mut Ui| {
|
||||||
|
@ -175,6 +190,7 @@ impl crate::disk::state::Xvb {
|
||||||
.response
|
.response
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
});
|
||||||
// Rules link help
|
// Rules link help
|
||||||
ui.add_space(ui.available_height() / 2.0);
|
ui.add_space(ui.available_height() / 2.0);
|
||||||
ui.vertical_centered(|ui| {
|
ui.vertical_centered(|ui| {
|
||||||
|
|
|
@ -255,6 +255,37 @@ pub enum XvbNode {
|
||||||
NorthAmerica,
|
NorthAmerica,
|
||||||
#[default]
|
#[default]
|
||||||
Europe,
|
Europe,
|
||||||
|
P2pool,
|
||||||
|
}
|
||||||
|
impl XvbNode {
|
||||||
|
pub fn url(&self) -> String {
|
||||||
|
match self {
|
||||||
|
Self::NorthAmerica => String::from(XVB_NODE_NA),
|
||||||
|
Self::Europe => String::from(XVB_NODE_EU),
|
||||||
|
Self::P2pool => String::from("127.0.0.1:3333"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn user(&self, address: &str) -> String {
|
||||||
|
match self {
|
||||||
|
Self::NorthAmerica => address.chars().take(8).collect(),
|
||||||
|
Self::Europe => address.chars().take(8).collect(),
|
||||||
|
Self::P2pool => GUPAX_VERSION_UNDERSCORE.to_string(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn tls(&self) -> bool {
|
||||||
|
match self {
|
||||||
|
Self::NorthAmerica => true,
|
||||||
|
Self::Europe => true,
|
||||||
|
Self::P2pool => false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn keepalive(&self) -> bool {
|
||||||
|
match self {
|
||||||
|
Self::NorthAmerica => true,
|
||||||
|
Self::Europe => true,
|
||||||
|
Self::P2pool => false,
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug, Deserialize, Serialize)]
|
#[derive(Clone, Debug, Deserialize, Serialize)]
|
||||||
|
|
|
@ -1,12 +1,15 @@
|
||||||
|
use crate::disk::state::XvbNode;
|
||||||
use crate::helper::{ProcessName, ProcessSignal, ProcessState};
|
use crate::helper::{ProcessName, ProcessSignal, ProcessState};
|
||||||
use crate::regex::XMRIG_REGEX;
|
use crate::regex::XMRIG_REGEX;
|
||||||
use crate::utils::human::HumanNumber;
|
use crate::utils::human::HumanNumber;
|
||||||
use crate::utils::sudo::SudoState;
|
use crate::utils::sudo::SudoState;
|
||||||
use crate::{constants::*, macros::*};
|
use crate::{constants::*, macros::*};
|
||||||
|
use anyhow::{anyhow, Result};
|
||||||
use log::*;
|
use log::*;
|
||||||
use readable::num::Unsigned;
|
use readable::num::Unsigned;
|
||||||
use readable::up::Uptime;
|
use readable::up::Uptime;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
use serde_json::Value;
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
use std::{
|
use std::{
|
||||||
fmt::Write,
|
fmt::Write,
|
||||||
|
@ -199,8 +202,6 @@ impl Helper {
|
||||||
args.push("127.0.0.1".to_string()); // HTTP API IP
|
args.push("127.0.0.1".to_string()); // HTTP API IP
|
||||||
args.push("--http-port".to_string());
|
args.push("--http-port".to_string());
|
||||||
args.push("18088".to_string()); // HTTP API Port
|
args.push("18088".to_string()); // HTTP API Port
|
||||||
args.push(format!("--http-access-token={}", state.token)); // HTTP API Port
|
|
||||||
args.push("--http-no-restricted".to_string());
|
|
||||||
if state.pause != 0 {
|
if state.pause != 0 {
|
||||||
args.push("--pause-on-active".to_string());
|
args.push("--pause-on-active".to_string());
|
||||||
args.push(state.pause.to_string());
|
args.push(state.pause.to_string());
|
||||||
|
@ -290,6 +291,8 @@ impl Helper {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
args.push(format!("--http-access-token={}", state.token)); // HTTP API Port
|
||||||
|
args.push("--http-no-restricted".to_string());
|
||||||
(args, format!("{}:{}", api_ip, api_port))
|
(args, format!("{}:{}", api_ip, api_port))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -622,8 +625,8 @@ pub struct PubXmrigApi {
|
||||||
pub diff: String,
|
pub diff: String,
|
||||||
pub accepted: String,
|
pub accepted: String,
|
||||||
pub rejected: String,
|
pub rejected: String,
|
||||||
|
|
||||||
pub hashrate_raw: f32,
|
pub hashrate_raw: f32,
|
||||||
|
pub hashrate_raw_15m: f32,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for PubXmrigApi {
|
impl Default for PubXmrigApi {
|
||||||
|
@ -644,6 +647,7 @@ impl PubXmrigApi {
|
||||||
accepted: UNKNOWN_DATA.to_string(),
|
accepted: UNKNOWN_DATA.to_string(),
|
||||||
rejected: UNKNOWN_DATA.to_string(),
|
rejected: UNKNOWN_DATA.to_string(),
|
||||||
hashrate_raw: 0.0,
|
hashrate_raw: 0.0,
|
||||||
|
hashrate_raw_15m: 0.0,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -701,6 +705,10 @@ impl PubXmrigApi {
|
||||||
Some(Some(h)) => *h,
|
Some(Some(h)) => *h,
|
||||||
_ => 0.0,
|
_ => 0.0,
|
||||||
};
|
};
|
||||||
|
let hashrate_raw_15m = match private.hashrate.total.last() {
|
||||||
|
Some(Some(h)) => *h,
|
||||||
|
_ => 0.0,
|
||||||
|
};
|
||||||
|
|
||||||
*public = Self {
|
*public = Self {
|
||||||
worker_id: private.worker_id,
|
worker_id: private.worker_id,
|
||||||
|
@ -710,6 +718,7 @@ impl PubXmrigApi {
|
||||||
accepted: Unsigned::from(private.connection.accepted as usize).to_string(),
|
accepted: Unsigned::from(private.connection.accepted as usize).to_string(),
|
||||||
rejected: Unsigned::from(private.connection.rejected as usize).to_string(),
|
rejected: Unsigned::from(private.connection.rejected as usize).to_string(),
|
||||||
hashrate_raw,
|
hashrate_raw,
|
||||||
|
hashrate_raw_15m,
|
||||||
..std::mem::take(&mut *public)
|
..std::mem::take(&mut *public)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -758,6 +767,61 @@ impl PrivXmrigApi {
|
||||||
let body = hyper::body::to_bytes(response?.body_mut()).await?;
|
let body = hyper::body::to_bytes(response?.body_mut()).await?;
|
||||||
Ok(serde_json::from_slice::<Self>(&body)?)
|
Ok(serde_json::from_slice::<Self>(&body)?)
|
||||||
}
|
}
|
||||||
|
// #[inline]
|
||||||
|
// // Replace config with new node
|
||||||
|
pub async fn update_xmrig_config(
|
||||||
|
client: &hyper::Client<hyper::client::HttpConnector>,
|
||||||
|
api_uri: &str,
|
||||||
|
token: &str,
|
||||||
|
node: XvbNode,
|
||||||
|
address: &str,
|
||||||
|
) -> Result<()> {
|
||||||
|
// get config
|
||||||
|
let request = hyper::Request::builder()
|
||||||
|
.method("GET")
|
||||||
|
.header("Authorization", ["Bearer ", token].concat())
|
||||||
|
.uri(api_uri)
|
||||||
|
.body(hyper::Body::empty())?;
|
||||||
|
let response = tokio::time::timeout(
|
||||||
|
std::time::Duration::from_millis(500),
|
||||||
|
client.request(request),
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
let body = hyper::body::to_bytes(response?.body_mut()).await?;
|
||||||
|
// deserialize to json
|
||||||
|
let mut config = serde_json::from_slice::<Value>(&body)?;
|
||||||
|
// modify node configuration
|
||||||
|
*config
|
||||||
|
.pointer_mut("/pools/0/url")
|
||||||
|
.ok_or_else(|| anyhow!("pools/0/url does not exist in xmrig config"))? =
|
||||||
|
node.url().into();
|
||||||
|
*config
|
||||||
|
.pointer_mut("/pools/0/user")
|
||||||
|
.ok_or_else(|| anyhow!("pools/0/user does not exist in xmrig config"))? =
|
||||||
|
node.user(&address).into();
|
||||||
|
*config
|
||||||
|
.pointer_mut("/pools/0/tls")
|
||||||
|
.ok_or_else(|| anyhow!("pools/0/tls does not exist in xmrig config"))? =
|
||||||
|
node.tls().into();
|
||||||
|
*config
|
||||||
|
.pointer_mut("/pools/0/keepalive")
|
||||||
|
.ok_or_else(|| anyhow!("pools/0/keepalive does not exist in xmrig config"))? =
|
||||||
|
node.keepalive().into();
|
||||||
|
// reconstruct body from new config
|
||||||
|
let body = hyper::body::Body::from(config.to_string());
|
||||||
|
// send new config
|
||||||
|
let request = hyper::Request::builder()
|
||||||
|
.method("PUT")
|
||||||
|
.header("Authorization", ["Bearer ", token].concat())
|
||||||
|
.uri(api_uri)
|
||||||
|
.body(body)?;
|
||||||
|
tokio::time::timeout(
|
||||||
|
std::time::Duration::from_millis(500),
|
||||||
|
client.request(request),
|
||||||
|
)
|
||||||
|
.await??;
|
||||||
|
anyhow::Ok(())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize, Clone, Copy)]
|
#[derive(Debug, Serialize, Deserialize, Clone, Copy)]
|
||||||
|
|
|
@ -2,19 +2,28 @@ use anyhow::{bail, Result};
|
||||||
use bytes::Bytes;
|
use bytes::Bytes;
|
||||||
use derive_more::Display;
|
use derive_more::Display;
|
||||||
use hyper::client::HttpConnector;
|
use hyper::client::HttpConnector;
|
||||||
use hyper::StatusCode;
|
use hyper::{Client, StatusCode};
|
||||||
use hyper_tls::HttpsConnector;
|
use hyper_tls::HttpsConnector;
|
||||||
use log::{debug, error, info, warn};
|
use log::{debug, error, info, warn};
|
||||||
use readable::up::Uptime;
|
use readable::up::Uptime;
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
use std::fmt::Write;
|
use std::fmt::Write;
|
||||||
|
use std::time::Duration;
|
||||||
use std::{
|
use std::{
|
||||||
sync::{Arc, Mutex},
|
sync::{Arc, Mutex},
|
||||||
thread,
|
thread,
|
||||||
time::Instant,
|
time::Instant,
|
||||||
};
|
};
|
||||||
|
use tokio::spawn;
|
||||||
|
use tokio::time::sleep_until;
|
||||||
|
|
||||||
use crate::utils::constants::{XVB_PUBLIC_ONLY, XVB_URL};
|
use crate::disk::state::XvbNode;
|
||||||
|
use crate::helper::xmrig::PrivXmrigApi;
|
||||||
|
use crate::utils::constants::{
|
||||||
|
BLOCK_PPLNS_WINDOW_MAIN, BLOCK_PPLNS_WINDOW_MINI, SECOND_PER_BLOCK_P2POOL, XMRIG_CONFIG_URI,
|
||||||
|
XVB_BUFFER, XVB_PUBLIC_ONLY, XVB_ROUND_DONOR_MEGA_MIN_HR, XVB_ROUND_DONOR_MIN_HR,
|
||||||
|
XVB_ROUND_DONOR_VIP_MIN_HR, XVB_ROUND_DONOR_WHALE_MIN_HR, XVB_TIME_ALGO, XVB_URL,
|
||||||
|
};
|
||||||
use crate::{
|
use crate::{
|
||||||
helper::{ProcessSignal, ProcessState},
|
helper::{ProcessSignal, ProcessState},
|
||||||
utils::{
|
utils::{
|
||||||
|
@ -24,6 +33,7 @@ use crate::{
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::p2pool::PubP2poolApi;
|
use super::p2pool::PubP2poolApi;
|
||||||
|
use super::xmrig::PubXmrigApi;
|
||||||
use super::{Helper, Process};
|
use super::{Helper, Process};
|
||||||
|
|
||||||
impl Helper {
|
impl Helper {
|
||||||
|
@ -37,6 +47,7 @@ impl Helper {
|
||||||
helper: &Arc<Mutex<Self>>,
|
helper: &Arc<Mutex<Self>>,
|
||||||
state_xvb: &crate::disk::state::Xvb,
|
state_xvb: &crate::disk::state::Xvb,
|
||||||
state_p2pool: &crate::disk::state::P2pool,
|
state_p2pool: &crate::disk::state::P2pool,
|
||||||
|
state_xmrig: &crate::disk::state::Xmrig,
|
||||||
) {
|
) {
|
||||||
info!("XvB | Attempting to restart...");
|
info!("XvB | Attempting to restart...");
|
||||||
lock2!(helper, xvb).signal = ProcessSignal::Restart;
|
lock2!(helper, xvb).signal = ProcessSignal::Restart;
|
||||||
|
@ -44,6 +55,7 @@ impl Helper {
|
||||||
let helper = helper.clone();
|
let helper = helper.clone();
|
||||||
let state_xvb = state_xvb.clone();
|
let state_xvb = state_xvb.clone();
|
||||||
let state_p2pool = state_p2pool.clone();
|
let state_p2pool = state_p2pool.clone();
|
||||||
|
let state_xmrig = state_xmrig.clone();
|
||||||
// This thread lives to wait, start xmrig then die.
|
// This thread lives to wait, start xmrig then die.
|
||||||
thread::spawn(move || {
|
thread::spawn(move || {
|
||||||
while lock2!(helper, xvb).state != ProcessState::Waiting {
|
while lock2!(helper, xvb).state != ProcessState::Waiting {
|
||||||
|
@ -52,7 +64,7 @@ impl Helper {
|
||||||
}
|
}
|
||||||
// Ok, process is not alive, start the new one!
|
// Ok, process is not alive, start the new one!
|
||||||
info!("XvB | Old process seems dead, starting new one!");
|
info!("XvB | Old process seems dead, starting new one!");
|
||||||
Self::start_xvb(&helper, &state_xvb, &state_p2pool);
|
Self::start_xvb(&helper, &state_xvb, &state_p2pool, &state_xmrig);
|
||||||
});
|
});
|
||||||
info!("XMRig | Restart ... OK");
|
info!("XMRig | Restart ... OK");
|
||||||
}
|
}
|
||||||
|
@ -60,6 +72,7 @@ impl Helper {
|
||||||
helper: &Arc<Mutex<Self>>,
|
helper: &Arc<Mutex<Self>>,
|
||||||
state_xvb: &crate::disk::state::Xvb,
|
state_xvb: &crate::disk::state::Xvb,
|
||||||
state_p2pool: &crate::disk::state::P2pool,
|
state_p2pool: &crate::disk::state::P2pool,
|
||||||
|
state_xmrig: &crate::disk::state::Xmrig,
|
||||||
) {
|
) {
|
||||||
info!("XvB | setting state to Middle");
|
info!("XvB | setting state to Middle");
|
||||||
lock2!(helper, xvb).state = ProcessState::Middle;
|
lock2!(helper, xvb).state = ProcessState::Middle;
|
||||||
|
@ -70,6 +83,8 @@ impl Helper {
|
||||||
// needed to see if it is alive. For XvB process to function completely, p2pool node must be alive to check the shares in the pplns window.
|
// needed to see if it is alive. For XvB process to function completely, p2pool node must be alive to check the shares in the pplns window.
|
||||||
let process_p2pool = Arc::clone(&lock!(helper).p2pool);
|
let process_p2pool = Arc::clone(&lock!(helper).p2pool);
|
||||||
let gui_api_p2pool = Arc::clone(&lock!(helper).gui_api_p2pool);
|
let gui_api_p2pool = Arc::clone(&lock!(helper).gui_api_p2pool);
|
||||||
|
let process_xmrig = Arc::clone(&lock!(helper).xmrig);
|
||||||
|
let gui_api_xmrig = Arc::clone(&lock!(helper).gui_api_xmrig);
|
||||||
info!("XvB | cloning of state");
|
info!("XvB | cloning of state");
|
||||||
// Reset before printing to output.
|
// Reset before printing to output.
|
||||||
// Need to reset because values of stats would stay otherwise which could bring confusion even if panel is with a disabled theme.
|
// Need to reset because values of stats would stay otherwise which could bring confusion even if panel is with a disabled theme.
|
||||||
|
@ -87,6 +102,7 @@ impl Helper {
|
||||||
// verify if token and address are existent on XvB server
|
// verify if token and address are existent on XvB server
|
||||||
let state_xvb = state_xvb.clone();
|
let state_xvb = state_xvb.clone();
|
||||||
let state_p2pool = state_p2pool.clone();
|
let state_p2pool = state_p2pool.clone();
|
||||||
|
let state_xmrig = state_xmrig.clone();
|
||||||
|
|
||||||
info!("XvB | spawn watchdog");
|
info!("XvB | spawn watchdog");
|
||||||
thread::spawn(move || {
|
thread::spawn(move || {
|
||||||
|
@ -96,8 +112,11 @@ impl Helper {
|
||||||
process,
|
process,
|
||||||
&state_xvb,
|
&state_xvb,
|
||||||
&state_p2pool,
|
&state_p2pool,
|
||||||
|
&state_xmrig,
|
||||||
gui_api_p2pool,
|
gui_api_p2pool,
|
||||||
process_p2pool,
|
process_p2pool,
|
||||||
|
gui_api_xmrig,
|
||||||
|
process_xmrig,
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -108,34 +127,30 @@ impl Helper {
|
||||||
process: Arc<Mutex<Process>>,
|
process: Arc<Mutex<Process>>,
|
||||||
state_xvb: &crate::disk::state::Xvb,
|
state_xvb: &crate::disk::state::Xvb,
|
||||||
state_p2pool: &crate::disk::state::P2pool,
|
state_p2pool: &crate::disk::state::P2pool,
|
||||||
|
state_xmrig: &crate::disk::state::Xmrig,
|
||||||
gui_api_p2pool: Arc<Mutex<PubP2poolApi>>,
|
gui_api_p2pool: Arc<Mutex<PubP2poolApi>>,
|
||||||
process_p2pool: Arc<Mutex<Process>>,
|
process_p2pool: Arc<Mutex<Process>>,
|
||||||
|
gui_api_xmrig: Arc<Mutex<PubXmrigApi>>,
|
||||||
|
process_xmrig: Arc<Mutex<Process>>,
|
||||||
) {
|
) {
|
||||||
let https = HttpsConnector::new();
|
let https = HttpsConnector::new();
|
||||||
let client = hyper::Client::builder().build(https);
|
let client = hyper::Client::builder().build(https);
|
||||||
let resp =
|
|
||||||
XvbPrivStats::request_api(&client, &state_p2pool.address, &state_xvb.token).await;
|
|
||||||
info!("XvB | verify address and token");
|
info!("XvB | verify address and token");
|
||||||
match resp {
|
if let Err(err) =
|
||||||
Ok(_) => {
|
XvbPrivStats::request_api(&client, &state_p2pool.address, &state_xvb.token).await
|
||||||
let mut lock = lock!(process);
|
{
|
||||||
lock.state = ProcessState::Alive;
|
|
||||||
}
|
|
||||||
Err(err) => {
|
|
||||||
// send to console: token non existent for address on XvB server
|
// send to console: token non existent for address on XvB server
|
||||||
warn!("Xvb | Start ... Partially failed because token and associated address are not existent on XvB server: {}\n", err);
|
warn!("Xvb | Start ... Partially failed because token and associated address are not existent on XvB server: {}\n", err);
|
||||||
// output the error to console
|
// output the error to console
|
||||||
if let Err(e) = writeln!(
|
if let Err(e) = writeln!(
|
||||||
lock!(gui_api).output,
|
lock!(gui_api).output,
|
||||||
"Failure to retrieve private stats from XvB server.\nError: {}\n{}\n",
|
"Token and associated address are not valid on XvB API.\nCheck if you are registered.\nError: {}\n",
|
||||||
err,
|
err,
|
||||||
XVB_PUBLIC_ONLY
|
|
||||||
) {
|
) {
|
||||||
error!("XvB Watchdog | GUI status write failed: {}", e);
|
error!("XvB Watchdog | GUI status write failed: {}", e);
|
||||||
}
|
}
|
||||||
lock!(process).state = ProcessState::NotMining;
|
lock!(process).state = ProcessState::NotMining;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
info!("XvB | verify p2pool node");
|
info!("XvB | verify p2pool node");
|
||||||
if !lock!(process_p2pool).is_alive() {
|
if !lock!(process_p2pool).is_alive() {
|
||||||
// send to console: p2pool process is not running
|
// send to console: p2pool process is not running
|
||||||
|
@ -143,7 +158,21 @@ impl Helper {
|
||||||
// output the error to console
|
// output the error to console
|
||||||
if let Err(e) = writeln!(
|
if let Err(e) = writeln!(
|
||||||
lock!(gui_api).output,
|
lock!(gui_api).output,
|
||||||
"Failure to completely start XvB process because p2pool instance is not running.\n",
|
"P2pool process is not running.\nCheck the P2pool Tab\n",
|
||||||
|
) {
|
||||||
|
error!("XvB Watchdog | GUI status write failed: {}", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
lock!(process).state = ProcessState::Syncing;
|
||||||
|
}
|
||||||
|
|
||||||
|
if !lock!(process_xmrig).is_alive() {
|
||||||
|
// send to console: p2pool process is not running
|
||||||
|
warn!("Xvb | Start ... Partially failed because Xmrig instance is not running.");
|
||||||
|
// output the error to console
|
||||||
|
if let Err(e) = writeln!(
|
||||||
|
lock!(gui_api).output,
|
||||||
|
"XMRig process is not running.\nCheck the Xmrig Tab.\n",
|
||||||
) {
|
) {
|
||||||
error!("XvB Watchdog | GUI status write failed: {}", e);
|
error!("XvB Watchdog | GUI status write failed: {}", e);
|
||||||
}
|
}
|
||||||
|
@ -151,12 +180,17 @@ impl Helper {
|
||||||
lock!(process).state = ProcessState::Syncing;
|
lock!(process).state = ProcessState::Syncing;
|
||||||
}
|
}
|
||||||
info!("XvB | print to console state");
|
info!("XvB | print to console state");
|
||||||
if lock!(process).state != ProcessState::Alive {
|
if lock!(process).state != ProcessState::Middle {
|
||||||
if let Err(e) = writeln!(lock!(gui_api).output, "{}\n", XVB_PUBLIC_ONLY,) {
|
if let Err(e) = writeln!(
|
||||||
|
lock!(gui_api).output,
|
||||||
|
"XvB partially started.\n{}\n",
|
||||||
|
XVB_PUBLIC_ONLY,
|
||||||
|
) {
|
||||||
error!("XvB Watchdog | GUI status write failed: {}", e);
|
error!("XvB Watchdog | GUI status write failed: {}", e);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
info!("XvB started");
|
info!("XvB Fully started");
|
||||||
|
lock!(process).state = ProcessState::Alive;
|
||||||
if let Err(e) = writeln!(lock!(gui_api).output, "XvB started\n") {
|
if let Err(e) = writeln!(lock!(gui_api).output, "XvB started\n") {
|
||||||
error!("XvB Watchdog | GUI status write failed: {}", e);
|
error!("XvB Watchdog | GUI status write failed: {}", e);
|
||||||
}
|
}
|
||||||
|
@ -177,19 +211,23 @@ impl Helper {
|
||||||
// let mut old_shares = 0;
|
// let mut old_shares = 0;
|
||||||
let mut time_last_share: Option<Instant> = None;
|
let mut time_last_share: Option<Instant> = None;
|
||||||
let start = lock!(process).start;
|
let start = lock!(process).start;
|
||||||
|
let mut start_algorithm = tokio::time::Instant::now();
|
||||||
info!("XvB | Entering watchdog mode... woof!");
|
info!("XvB | Entering watchdog mode... woof!");
|
||||||
loop {
|
loop {
|
||||||
debug!("XvB Watchdog | ----------- Start of loop -----------");
|
debug!("XvB Watchdog | ----------- Start of loop -----------");
|
||||||
// verify if p2pool node is running with correct token.
|
// if address and token valid, verify if p2pool and xmrig are running, else XvBmust be reloaded with another token/address to start verifying the other process.
|
||||||
if lock!(process).state != ProcessState::NotMining {
|
if lock!(process).state != ProcessState::NotMining {
|
||||||
if lock!(process_p2pool).is_alive() {
|
// verify if p2pool node and xmrig are running
|
||||||
|
if lock!(process_p2pool).is_alive() && lock!(process_xmrig).is_alive() {
|
||||||
// verify if state is to changed
|
// verify if state is to changed
|
||||||
if lock!(process).state == ProcessState::Syncing {
|
if lock!(process).state == ProcessState::Syncing {
|
||||||
info!("XvB | started this time with p2pool");
|
info!("XvB | started this time with p2pool and xmrig");
|
||||||
|
*lock!(pub_api) = PubXvbApi::new();
|
||||||
|
*lock!(gui_api) = PubXvbApi::new();
|
||||||
lock!(process).state = ProcessState::Alive;
|
lock!(process).state = ProcessState::Alive;
|
||||||
if let Err(e) = writeln!(
|
if let Err(e) = writeln!(
|
||||||
lock!(gui_api).output,
|
lock!(gui_api).output,
|
||||||
"XvB is now started because p2pool node came online.\n",
|
"XvB is now started because p2pool and xmrig came online.\n",
|
||||||
) {
|
) {
|
||||||
error!("XvB Watchdog | GUI status write failed: {}", e);
|
error!("XvB Watchdog | GUI status write failed: {}", e);
|
||||||
}
|
}
|
||||||
|
@ -198,14 +236,15 @@ impl Helper {
|
||||||
// verify if the state is changing because p2pool is not alive anymore.
|
// verify if the state is changing because p2pool is not alive anymore.
|
||||||
if lock!(process).state != ProcessState::Syncing {
|
if lock!(process).state != ProcessState::Syncing {
|
||||||
info!("XvB | stop partially because p2pool is not alive anymore.");
|
info!("XvB | stop partially because p2pool is not alive anymore.");
|
||||||
lock!(process).state = ProcessState::Alive;
|
*lock!(pub_api) = PubXvbApi::new();
|
||||||
|
*lock!(gui_api) = PubXvbApi::new();
|
||||||
|
lock!(process).state = ProcessState::Syncing;
|
||||||
if let Err(e) = writeln!(
|
if let Err(e) = writeln!(
|
||||||
lock!(gui_api).output,
|
lock!(gui_api).output,
|
||||||
"XvB is now partially stopped because p2pool node came offline.\n",
|
"XvB is now partially stopped because p2pool node or xmrig came offline.\nCheck P2pool and Xmrig Tabs",
|
||||||
) {
|
) {
|
||||||
error!("XvB Watchdog | GUI status write failed: {}", e);
|
error!("XvB Watchdog | GUI status write failed: {}", e);
|
||||||
}
|
}
|
||||||
lock!(process).state = ProcessState::Syncing
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -307,13 +346,27 @@ impl Helper {
|
||||||
let round = if share {
|
let round = if share {
|
||||||
let stats_priv = &lock!(pub_api).stats_priv;
|
let stats_priv = &lock!(pub_api).stats_priv;
|
||||||
match (
|
match (
|
||||||
stats_priv.donor_1hr_avg / 1000.0,
|
stats_priv.donor_1hr_avg as u32,
|
||||||
stats_priv.donor_24hr_avg / 1000.0,
|
stats_priv.donor_24hr_avg as u32,
|
||||||
) {
|
) {
|
||||||
x if x.0 > 1000.0 && x.1 > 1000.0 => Some(XvbRound::DonorMega),
|
x if x.0 > XVB_ROUND_DONOR_MEGA_MIN_HR
|
||||||
x if x.0 > 100.0 && x.1 > 100.0 => Some(XvbRound::DonorWhale),
|
&& x.1 > XVB_ROUND_DONOR_MEGA_MIN_HR =>
|
||||||
x if x.0 > 10.0 && x.1 > 10.0 => Some(XvbRound::DonorVip),
|
{
|
||||||
x if x.0 > 1.0 && x.1 > 1.0 => Some(XvbRound::Donor),
|
Some(XvbRound::DonorMega)
|
||||||
|
}
|
||||||
|
x if x.0 > XVB_ROUND_DONOR_WHALE_MIN_HR
|
||||||
|
&& x.1 > XVB_ROUND_DONOR_WHALE_MIN_HR =>
|
||||||
|
{
|
||||||
|
Some(XvbRound::DonorWhale)
|
||||||
|
}
|
||||||
|
x if x.0 > XVB_ROUND_DONOR_VIP_MIN_HR
|
||||||
|
&& x.1 > XVB_ROUND_DONOR_VIP_MIN_HR =>
|
||||||
|
{
|
||||||
|
Some(XvbRound::DonorVip)
|
||||||
|
}
|
||||||
|
x if x.0 > XVB_ROUND_DONOR_MIN_HR && x.1 > XVB_ROUND_DONOR_MIN_HR => {
|
||||||
|
Some(XvbRound::Donor)
|
||||||
|
}
|
||||||
(_, _) => Some(XvbRound::Vip),
|
(_, _) => Some(XvbRound::Vip),
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -328,20 +381,83 @@ impl Helper {
|
||||||
{
|
{
|
||||||
lock!(pub_api).stats_priv.win_current = true
|
lock!(pub_api).stats_priv.win_current = true
|
||||||
}
|
}
|
||||||
|
|
||||||
// if 10 minutes passed since last check
|
// if 10 minutes passed since last check
|
||||||
if lock!(gui_api).tick_distribute_hr > (60 * 10) {
|
// the first 15 minutes, the HR of xmrig will be 0.0, so xmrig will always mine on p2pool for 15m.
|
||||||
|
if start_algorithm.elapsed() >= Duration::from_secs(XVB_TIME_ALGO.into()) {
|
||||||
|
info!("Xvb Process | Algorithm is started");
|
||||||
|
// the time that takes the algorithm do decide the next ten minutes could means less p2pool mining. It is solved by the buffer.
|
||||||
|
start_algorithm = tokio::time::Instant::now();
|
||||||
// request XMrig to mine on P2pool
|
// request XMrig to mine on P2pool
|
||||||
|
info!("Xvb Process | request to mine on p2pool");
|
||||||
|
let client: hyper::Client<hyper::client::HttpConnector> =
|
||||||
|
hyper::Client::builder().build(hyper::client::HttpConnector::new());
|
||||||
|
let api_uri = ["http://127.0.0.1:18088/", XMRIG_CONFIG_URI].concat();
|
||||||
|
if let Err(err) = PrivXmrigApi::update_xmrig_config(
|
||||||
|
&client,
|
||||||
|
&api_uri,
|
||||||
|
&state_xmrig.token,
|
||||||
|
XvbNode::P2pool,
|
||||||
|
&state_p2pool.address,
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
// show to console error about updating xmrig config
|
||||||
|
if let Err(e) = writeln!(
|
||||||
|
lock!(gui_api).output,
|
||||||
|
"Failure to update xmrig config with HTTP API.\nError: {}",
|
||||||
|
err
|
||||||
|
) {
|
||||||
|
error!("XvB Watchdog | GUI status write failed: {}", e);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// if share is in PW,
|
// if share is in PW,
|
||||||
// check average HR of last 15 minutes from XMrig.
|
if share {
|
||||||
// if HR + buffer >= mHR,
|
info!("Xvb Process | Algorithm share is in current window");
|
||||||
|
// calcul minimum HR
|
||||||
|
|
||||||
|
let hr = lock!(gui_api_xmrig).hashrate_raw_15m;
|
||||||
|
let min_hr = Helper::minimum_hashrate_share(
|
||||||
|
lock!(gui_api_p2pool).p2pool_difficulty_u64,
|
||||||
|
state_p2pool.mini,
|
||||||
|
);
|
||||||
|
info!("Xvb Process | hr {}, min_hr: {} ", hr, min_hr);
|
||||||
|
|
||||||
// calculate how much time can be spared
|
// calculate how much time can be spared
|
||||||
|
let mut spared_time = Helper::time_that_could_be_spared(hr, min_hr);
|
||||||
|
if spared_time > 0 {
|
||||||
// if not hero option
|
// if not hero option
|
||||||
|
if !state_xvb.hero {
|
||||||
// calculate how much time needed to be spared to be in most round type minimum HR + buffer
|
// calculate how much time needed to be spared to be in most round type minimum HR + buffer
|
||||||
// fi
|
spared_time = Helper::minimum_time_for_highest_accessible_round(
|
||||||
|
spared_time,
|
||||||
|
hr,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
info!("Xvb Process | spared time {} ", spared_time);
|
||||||
// sleep 10m less spared time then request XMrig to mine on XvB
|
// sleep 10m less spared time then request XMrig to mine on XvB
|
||||||
// fi
|
let was_instant = start_algorithm.clone();
|
||||||
// fi
|
let node = state_xvb.node.clone();
|
||||||
|
let token = state_xmrig.token.clone();
|
||||||
|
let address = state_p2pool.address.clone();
|
||||||
|
let gui_api = gui_api.clone();
|
||||||
|
spawn(async move {
|
||||||
|
Helper::sleep_then_update_node_xmrig(
|
||||||
|
was_instant,
|
||||||
|
spared_time,
|
||||||
|
&client,
|
||||||
|
&api_uri,
|
||||||
|
&token,
|
||||||
|
node,
|
||||||
|
&address,
|
||||||
|
gui_api,
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
// instant saved for next check
|
// instant saved for next check
|
||||||
// fi
|
// fi
|
||||||
}
|
}
|
||||||
|
@ -364,6 +480,61 @@ impl Helper {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
fn minimum_hashrate_share(difficulty: u64, mini: bool) -> f32 {
|
||||||
|
let pws = if mini {
|
||||||
|
BLOCK_PPLNS_WINDOW_MINI
|
||||||
|
} else {
|
||||||
|
BLOCK_PPLNS_WINDOW_MAIN
|
||||||
|
};
|
||||||
|
(difficulty / (pws * SECOND_PER_BLOCK_P2POOL)) as f32 * XVB_BUFFER
|
||||||
|
}
|
||||||
|
fn time_that_could_be_spared(hr: f32, min_hr: f32) -> u32 {
|
||||||
|
// percent of time minimum
|
||||||
|
let minimum_time_required_on_p2pool = XVB_TIME_ALGO as f32 / (hr / min_hr);
|
||||||
|
let spared_time = XVB_TIME_ALGO as f32 - minimum_time_required_on_p2pool;
|
||||||
|
// if less than 10 seconds, XMRig could hardly have the time to mine anything.
|
||||||
|
if spared_time >= 10f32 {
|
||||||
|
return spared_time as u32;
|
||||||
|
}
|
||||||
|
0
|
||||||
|
}
|
||||||
|
fn minimum_time_for_highest_accessible_round(st: u32, hr: f32) -> u32 {
|
||||||
|
let hr_for_xvb = ((st as f32 / XVB_TIME_ALGO as f32) * hr) as u32;
|
||||||
|
match hr_for_xvb {
|
||||||
|
x if x > XVB_ROUND_DONOR_MEGA_MIN_HR => x - XVB_ROUND_DONOR_MEGA_MIN_HR,
|
||||||
|
x if x > XVB_ROUND_DONOR_WHALE_MIN_HR => x - XVB_ROUND_DONOR_WHALE_MIN_HR,
|
||||||
|
x if x > XVB_ROUND_DONOR_VIP_MIN_HR => x - XVB_ROUND_DONOR_VIP_MIN_HR,
|
||||||
|
x if x > XVB_ROUND_DONOR_MIN_HR => x - XVB_ROUND_DONOR_MIN_HR,
|
||||||
|
_ => 0,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
async fn sleep_then_update_node_xmrig(
|
||||||
|
was_instant: tokio::time::Instant,
|
||||||
|
spared_time: u32,
|
||||||
|
client: &Client<HttpConnector>,
|
||||||
|
api_uri: &str,
|
||||||
|
token_xmrig: &str,
|
||||||
|
node: XvbNode,
|
||||||
|
address: &str,
|
||||||
|
gui_api: Arc<Mutex<PubXvbApi>>,
|
||||||
|
) {
|
||||||
|
info!("Xvb Process | for now mine on p2pol ");
|
||||||
|
sleep_until(was_instant + Duration::from_secs((XVB_TIME_ALGO - spared_time) as u64)).await;
|
||||||
|
if let Err(err) =
|
||||||
|
PrivXmrigApi::update_xmrig_config(client, api_uri, token_xmrig, node, address).await
|
||||||
|
{
|
||||||
|
// show to console error about updating xmrig config
|
||||||
|
if let Err(e) = writeln!(
|
||||||
|
lock!(gui_api).output,
|
||||||
|
"Failure to update xmrig config with HTTP API.\nError: {}",
|
||||||
|
err
|
||||||
|
) {
|
||||||
|
error!("XvB Watchdog | GUI status write failed: {}", e);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
info!("Xvb Process | mining on XvB pool");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
//---------------------------------------------------------------------------------------------------- Public XvB API
|
//---------------------------------------------------------------------------------------------------- Public XvB API
|
||||||
use serde_this_or_that::as_u64;
|
use serde_this_or_that::as_u64;
|
||||||
|
@ -553,6 +724,15 @@ fn signal_interrupt(
|
||||||
//---------------------------------------------------------------------------------------------------- TEST
|
//---------------------------------------------------------------------------------------------------- TEST
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod test {
|
mod test {
|
||||||
|
// use std::{sync::Arc, thread, time::Instant};
|
||||||
|
|
||||||
|
// use crate::{
|
||||||
|
// app::App,
|
||||||
|
// disk::state::XvbNode,
|
||||||
|
// helper::{xmrig::PrivXmrigApi, Helper},
|
||||||
|
// utils::constants::XMRIG_CONFIG_URI,
|
||||||
|
// };
|
||||||
|
|
||||||
use std::thread;
|
use std::thread;
|
||||||
|
|
||||||
use super::XvbPubStats;
|
use super::XvbPubStats;
|
||||||
|
@ -571,4 +751,39 @@ mod test {
|
||||||
async fn corr(client: Client<HttpsConnector<hyper::client::HttpConnector>>) -> XvbPubStats {
|
async fn corr(client: Client<HttpsConnector<hyper::client::HttpConnector>>) -> XvbPubStats {
|
||||||
XvbPubStats::request_api(&client).await.unwrap()
|
XvbPubStats::request_api(&client).await.unwrap()
|
||||||
}
|
}
|
||||||
|
// #[test]
|
||||||
|
// fn update_xmrig_config() {
|
||||||
|
// let client: hyper::Client<hyper::client::HttpConnector> =
|
||||||
|
// hyper::Client::builder().build(hyper::client::HttpConnector::new());
|
||||||
|
// let node = XvbNode::Europe;
|
||||||
|
// let api_uri = ["http://127.0.0.1:18088/", XMRIG_CONFIG_URI].concat();
|
||||||
|
// // start app
|
||||||
|
// let app = App::new(Instant::now());
|
||||||
|
// // start xmrig
|
||||||
|
|
||||||
|
// Helper::start_xmrig(
|
||||||
|
// &app.helper,
|
||||||
|
// &app.state.xmrig,
|
||||||
|
// &app.state.gupax.absolute_xmrig_path,
|
||||||
|
// Arc::clone(&app.sudo),
|
||||||
|
// );
|
||||||
|
// let token = app.state.xmrig.token;
|
||||||
|
// let address = app.state.p2pool.address;
|
||||||
|
// // change config
|
||||||
|
// thread::spawn(move || req_update_config(client, &api_uri, &token, node, &address))
|
||||||
|
// .join()
|
||||||
|
// .unwrap();
|
||||||
|
// }
|
||||||
|
// #[tokio::main]
|
||||||
|
// async fn req_update_config(
|
||||||
|
// client: hyper::Client<hyper::client::HttpConnector>,
|
||||||
|
// api_uri: &str,
|
||||||
|
// token: &str,
|
||||||
|
// node: XvbNode,
|
||||||
|
// address: &str,
|
||||||
|
// ) {
|
||||||
|
// PrivXmrigApi::replace_xmrig_config(client, &api_uri, token, node, address)
|
||||||
|
// .await
|
||||||
|
// .unwrap()
|
||||||
|
// }
|
||||||
}
|
}
|
||||||
|
|
|
@ -208,7 +208,7 @@ pub fn init_auto(app: &mut App) {
|
||||||
}
|
}
|
||||||
// [Auto-XvB]
|
// [Auto-XvB]
|
||||||
if app.state.gupax.auto_xvb {
|
if app.state.gupax.auto_xvb {
|
||||||
Helper::start_xvb(&app.helper, &app.state.xvb, &app.state.p2pool);
|
Helper::start_xvb(&app.helper, &app.state.xvb, &app.state.p2pool, &app.state.xmrig);
|
||||||
} else {
|
} else {
|
||||||
info!("Skipping auto-xvb...");
|
info!("Skipping auto-xvb...");
|
||||||
|
|
||||||
|
|
|
@ -86,6 +86,7 @@ pub const P2POOL_API_PATH_NETWORK: &str = "network/stats";
|
||||||
#[cfg(target_family = "unix")]
|
#[cfg(target_family = "unix")]
|
||||||
pub const P2POOL_API_PATH_POOL: &str = "pool/stats";
|
pub const P2POOL_API_PATH_POOL: &str = "pool/stats";
|
||||||
pub const XMRIG_API_URI: &str = "1/summary"; // The default relative URI of XMRig's API
|
pub const XMRIG_API_URI: &str = "1/summary"; // The default relative URI of XMRig's API
|
||||||
|
pub const XMRIG_CONFIG_URI: &str = "1/config"; // The default relative URI of XMRig's API config
|
||||||
|
|
||||||
// Process state tooltips (online, offline, etc)
|
// Process state tooltips (online, offline, etc)
|
||||||
pub const P2POOL_ALIVE: &str = "P2Pool is online and fully synchronized";
|
pub const P2POOL_ALIVE: &str = "P2Pool is online and fully synchronized";
|
||||||
|
@ -410,9 +411,15 @@ pub const XMRIG_PATH_EMPTY: &str = "XMRig PATH is empty! To fix: goto the [G
|
||||||
// XvB
|
// XvB
|
||||||
pub const XVB_HELP: &str = "You need to register an account by clicking on the link above to get your token with the same p2pool XMR address you use for payment.";
|
pub const XVB_HELP: &str = "You need to register an account by clicking on the link above to get your token with the same p2pool XMR address you use for payment.";
|
||||||
pub const XVB_URL: &str = "https://xmrvsbeast.com";
|
pub const XVB_URL: &str = "https://xmrvsbeast.com";
|
||||||
pub const XVB_URL_PUBLIC_API: &str = "https://xmrvsbeast.com/p2pool/stats";
|
|
||||||
pub const XVB_URL_RULES: &str = "https://xmrvsbeast.com/p2pool/rules.html";
|
|
||||||
|
|
||||||
|
pub const XVB_URL_PUBLIC_API: &str = "https://xmrvsbeast.com/p2pool/stats";
|
||||||
|
pub const XVB_NODE_EU: &str = "eu.xmrvsbeast.com:4247";
|
||||||
|
pub const XVB_NODE_NA: &str = "na.xmrvsbeast.com:4247";
|
||||||
|
pub const XVB_URL_RULES: &str = "https://xmrvsbeast.com/p2pool/rules.html";
|
||||||
|
// buffer in percentage of HR to have plus the requirement.
|
||||||
|
pub const XVB_BUFFER: f32 = 1.05;
|
||||||
|
// time in second the algorithm will distribute the HR
|
||||||
|
pub const XVB_TIME_ALGO: u32 = 600;
|
||||||
pub const XVB_TOKEN_LEN: usize = 9;
|
pub const XVB_TOKEN_LEN: usize = 9;
|
||||||
pub const XVB_HERO_SELECT: &str =
|
pub const XVB_HERO_SELECT: &str =
|
||||||
"Donate all spared hashrate to the raffle, even if there is more than enough to be in the most highest round type possible";
|
"Donate all spared hashrate to the raffle, even if there is more than enough to be in the most highest round type possible";
|
||||||
|
@ -422,6 +429,10 @@ pub const XVB_DONATED_1H_FIELD: &str = "Donated last hour";
|
||||||
pub const XVB_DONATED_24H_FIELD: &str = "Donated last 24 hours";
|
pub const XVB_DONATED_24H_FIELD: &str = "Donated last 24 hours";
|
||||||
pub const XVB_ROUND_TYPE_FIELD: &str = "Round";
|
pub const XVB_ROUND_TYPE_FIELD: &str = "Round";
|
||||||
pub const XVB_WINNER_FIELD: &str = "Win";
|
pub const XVB_WINNER_FIELD: &str = "Win";
|
||||||
|
pub const XVB_ROUND_DONOR_MIN_HR: u32 = 1000;
|
||||||
|
pub const XVB_ROUND_DONOR_VIP_MIN_HR: u32 = 10000;
|
||||||
|
pub const XVB_ROUND_DONOR_WHALE_MIN_HR: u32 = 100000;
|
||||||
|
pub const XVB_ROUND_DONOR_MEGA_MIN_HR: u32 = 1000000;
|
||||||
|
|
||||||
// CLI argument messages
|
// CLI argument messages
|
||||||
pub const ARG_HELP: &str = r#"USAGE: ./gupax [--flag]
|
pub const ARG_HELP: &str = r#"USAGE: ./gupax [--flag]
|
||||||
|
@ -449,8 +460,13 @@ For more information, see link below:
|
||||||
pub const UNKNOWN_DATA: &str = "???";
|
pub const UNKNOWN_DATA: &str = "???";
|
||||||
// Time PPLNS WINDOW in seconds
|
// Time PPLNS WINDOW in seconds
|
||||||
// it is an estimation based on number of block in a pplns window and block time (10s). The difficulty of the network should adapt to get close to this value.
|
// it is an estimation based on number of block in a pplns window and block time (10s). The difficulty of the network should adapt to get close to this value.
|
||||||
pub const TIME_PPLNS_WINDOW_MINI: Duration = Duration::from_secs(2160 * 10);
|
pub const BLOCK_PPLNS_WINDOW_MINI: u64 = 2160;
|
||||||
pub const TIME_PPLNS_WINDOW_MAIN: Duration = Duration::from_secs(363 * 10);
|
pub const BLOCK_PPLNS_WINDOW_MAIN: u64 = 363;
|
||||||
|
pub const SECOND_PER_BLOCK_P2POOL: u64 = 10;
|
||||||
|
pub const TIME_PPLNS_WINDOW_MINI: Duration =
|
||||||
|
Duration::from_secs(BLOCK_PPLNS_WINDOW_MINI * SECOND_PER_BLOCK_P2POOL);
|
||||||
|
pub const TIME_PPLNS_WINDOW_MAIN: Duration =
|
||||||
|
Duration::from_secs(BLOCK_PPLNS_WINDOW_MAIN * SECOND_PER_BLOCK_P2POOL);
|
||||||
|
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue