use serde::{Deserialize, Serialize};
use steel::*;
use crate::state::{miner_pda, Treasury};
use super::OreAccount;
#[repr(C)]
#[derive(Clone, Copy, Debug, PartialEq, Pod, Zeroable, Serialize, Deserialize)]
pub struct Miner {
pub authority: Pubkey,
pub deployed: [u64; 25],
pub cumulative: [u64; 25],
pub checkpoint_fee: u64,
pub checkpoint_id: u64,
pub last_claim_ore_at: i64,
pub last_claim_sol_at: i64,
pub rewards_factor: Numeric,
pub rewards_sol: u64,
pub rewards_ore: u64,
pub refined_ore: u64,
pub round_id: u64,
pub lifetime_rewards_sol: u64,
pub lifetime_rewards_ore: u64,
pub lifetime_deployed: u64,
}
impl Miner {
pub fn pda(&self) -> (Pubkey, u8) {
miner_pda(self.authority)
}
pub fn claim_ore(&mut self, clock: &Clock, treasury: &mut Treasury) -> u64 {
self.update_rewards(treasury);
let refined_ore = self.refined_ore;
let rewards_ore = self.rewards_ore;
let mut amount = refined_ore + rewards_ore;
self.refined_ore = 0;
self.rewards_ore = 0;
treasury.total_unclaimed -= rewards_ore;
treasury.total_refined -= refined_ore;
self.last_claim_ore_at = clock.unix_timestamp;
if treasury.total_unclaimed > 0 {
let fee = rewards_ore / 10;
amount -= fee;
treasury.miner_rewards_factor += Numeric::from_fraction(fee, treasury.total_unclaimed);
treasury.total_refined += fee;
self.lifetime_rewards_ore -= fee;
}
amount
}
pub fn claim_sol(&mut self, clock: &Clock) -> u64 {
let amount = self.rewards_sol;
self.rewards_sol = 0;
self.last_claim_sol_at = clock.unix_timestamp;
amount
}
pub fn update_rewards(&mut self, treasury: &Treasury) {
if treasury.miner_rewards_factor > self.rewards_factor {
let accumulated_rewards = treasury.miner_rewards_factor - self.rewards_factor;
if accumulated_rewards < Numeric::ZERO {
panic!("Accumulated rewards is negative");
}
let personal_rewards = accumulated_rewards * Numeric::from_u64(self.rewards_ore);
self.refined_ore += personal_rewards.to_u64();
self.lifetime_rewards_ore += personal_rewards.to_u64();
}
self.rewards_factor = treasury.miner_rewards_factor;
}
}
account!(OreAccount, Miner);