Dashboard: - Add aggregate stats across all running miners (total hashrate, shares) - Add workers table with per-miner stats, efficiency, and controls - Show hashrate bars and efficiency badges for each worker - Support stopping individual workers or all at once TT-Miner: - Implement Install, Start, GetStats, CheckInstallation, Uninstall - Add TT-Miner to Manager's StartMiner and ListAvailableMiners - Support GPU-specific config options (devices, intensity, cliArgs) Chart: - Improve styling with WA-Pro theme variables - Add hashrate unit formatting (H/s, kH/s, MH/s) - Better tooltip and axis styling Also: - Fix XMRig download URLs (linux-static-x64, windows-x64) - Add Playwright E2E testing infrastructure - Add XMR pool research documentation 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
389 lines
9.8 KiB
Go
389 lines
9.8 KiB
Go
package mining
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"net"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
// ManagerInterface defines the contract for a miner manager.
|
|
type ManagerInterface interface {
|
|
StartMiner(minerType string, config *Config) (Miner, error)
|
|
StopMiner(name string) error
|
|
GetMiner(name string) (Miner, error)
|
|
ListMiners() []Miner
|
|
ListAvailableMiners() []AvailableMiner
|
|
GetMinerHashrateHistory(name string) ([]HashratePoint, error)
|
|
UninstallMiner(minerType string) error
|
|
Stop()
|
|
}
|
|
|
|
// Manager handles the lifecycle and operations of multiple miners.
|
|
type Manager struct {
|
|
miners map[string]Miner
|
|
mu sync.RWMutex
|
|
stopChan chan struct{}
|
|
waitGroup sync.WaitGroup
|
|
}
|
|
|
|
var _ ManagerInterface = (*Manager)(nil)
|
|
|
|
// NewManager creates a new miner manager and autostarts miners based on config.
|
|
func NewManager() *Manager {
|
|
m := &Manager{
|
|
miners: make(map[string]Miner),
|
|
stopChan: make(chan struct{}),
|
|
waitGroup: sync.WaitGroup{},
|
|
}
|
|
m.syncMinersConfig() // Ensure config file is populated
|
|
m.autostartMiners()
|
|
m.startStatsCollection()
|
|
return m
|
|
}
|
|
|
|
// syncMinersConfig ensures the miners.json config file has entries for all available miners.
|
|
func (m *Manager) syncMinersConfig() {
|
|
cfg, err := LoadMinersConfig()
|
|
if err != nil {
|
|
log.Printf("Warning: could not load miners config for sync: %v", err)
|
|
return
|
|
}
|
|
|
|
availableMiners := m.ListAvailableMiners()
|
|
configUpdated := false
|
|
|
|
for _, availableMiner := range availableMiners {
|
|
found := false
|
|
for _, configuredMiner := range cfg.Miners {
|
|
if strings.EqualFold(configuredMiner.MinerType, availableMiner.Name) {
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
if !found {
|
|
cfg.Miners = append(cfg.Miners, MinerAutostartConfig{
|
|
MinerType: availableMiner.Name,
|
|
Autostart: false,
|
|
Config: nil, // No default config
|
|
})
|
|
configUpdated = true
|
|
log.Printf("Added default config for missing miner: %s", availableMiner.Name)
|
|
}
|
|
}
|
|
|
|
if configUpdated {
|
|
if err := SaveMinersConfig(cfg); err != nil {
|
|
log.Printf("Warning: failed to save updated miners config: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
// autostartMiners loads the miners config and starts any miners marked for autostart.
|
|
func (m *Manager) autostartMiners() {
|
|
cfg, err := LoadMinersConfig()
|
|
if err != nil {
|
|
log.Printf("Warning: could not load miners config for autostart: %v", err)
|
|
return
|
|
}
|
|
|
|
for _, minerCfg := range cfg.Miners {
|
|
if minerCfg.Autostart && minerCfg.Config != nil {
|
|
log.Printf("Autostarting miner: %s", minerCfg.MinerType)
|
|
if _, err := m.StartMiner(minerCfg.MinerType, minerCfg.Config); err != nil {
|
|
log.Printf("Failed to autostart miner %s: %v", minerCfg.MinerType, err)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// findAvailablePort finds an available TCP port on the local machine.
|
|
func findAvailablePort() (int, error) {
|
|
addr, err := net.ResolveTCPAddr("tcp", "localhost:0")
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
l, err := net.ListenTCP("tcp", addr)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
defer l.Close()
|
|
return l.Addr().(*net.TCPAddr).Port, nil
|
|
}
|
|
|
|
// StartMiner starts a new miner and saves its configuration.
|
|
func (m *Manager) StartMiner(minerType string, config *Config) (Miner, error) {
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
|
|
if config == nil {
|
|
config = &Config{}
|
|
}
|
|
|
|
var miner Miner
|
|
switch strings.ToLower(minerType) {
|
|
case "xmrig":
|
|
miner = NewXMRigMiner()
|
|
case "tt-miner", "ttminer":
|
|
miner = NewTTMiner()
|
|
default:
|
|
return nil, fmt.Errorf("unsupported miner type: %s", minerType)
|
|
}
|
|
|
|
instanceName := miner.GetName()
|
|
if config.Algo != "" {
|
|
instanceName = fmt.Sprintf("%s-%s", instanceName, config.Algo)
|
|
} else {
|
|
instanceName = fmt.Sprintf("%s-%d", instanceName, time.Now().UnixNano()%1000)
|
|
}
|
|
|
|
if _, exists := m.miners[instanceName]; exists {
|
|
return nil, fmt.Errorf("a miner with a similar configuration is already running: %s", instanceName)
|
|
}
|
|
|
|
apiPort, err := findAvailablePort()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to find an available port for the miner API: %w", err)
|
|
}
|
|
if config.HTTPPort == 0 {
|
|
config.HTTPPort = apiPort
|
|
}
|
|
|
|
if xmrigMiner, ok := miner.(*XMRigMiner); ok {
|
|
xmrigMiner.Name = instanceName
|
|
if xmrigMiner.API != nil {
|
|
xmrigMiner.API.ListenPort = apiPort
|
|
}
|
|
}
|
|
if ttMiner, ok := miner.(*TTMiner); ok {
|
|
ttMiner.Name = instanceName
|
|
if ttMiner.API != nil {
|
|
ttMiner.API.ListenPort = apiPort
|
|
}
|
|
}
|
|
|
|
if err := miner.Start(config); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
m.miners[instanceName] = miner
|
|
|
|
if err := m.updateMinerConfig(minerType, true, config); err != nil {
|
|
log.Printf("Warning: failed to save miner config for autostart: %v", err)
|
|
}
|
|
|
|
logMessage := fmt.Sprintf("CryptoCurrency Miner started: %s (Binary: %s)", miner.GetName(), miner.GetBinaryPath())
|
|
logToSyslog(logMessage)
|
|
|
|
return miner, nil
|
|
}
|
|
|
|
// UninstallMiner stops, uninstalls, and removes a miner's configuration.
|
|
func (m *Manager) UninstallMiner(minerType string) error {
|
|
m.mu.Lock()
|
|
for name, runningMiner := range m.miners {
|
|
if rm, ok := runningMiner.(*XMRigMiner); ok && strings.EqualFold(rm.ExecutableName, minerType) {
|
|
if err := runningMiner.Stop(); err != nil {
|
|
log.Printf("Warning: failed to stop running miner %s during uninstall: %v", name, err)
|
|
}
|
|
delete(m.miners, name)
|
|
}
|
|
if rm, ok := runningMiner.(*TTMiner); ok && strings.EqualFold(rm.ExecutableName, minerType) {
|
|
if err := runningMiner.Stop(); err != nil {
|
|
log.Printf("Warning: failed to stop running miner %s during uninstall: %v", name, err)
|
|
}
|
|
delete(m.miners, name)
|
|
}
|
|
}
|
|
m.mu.Unlock()
|
|
|
|
var miner Miner
|
|
switch strings.ToLower(minerType) {
|
|
case "xmrig":
|
|
miner = NewXMRigMiner()
|
|
case "tt-miner", "ttminer":
|
|
miner = NewTTMiner()
|
|
default:
|
|
return fmt.Errorf("unsupported miner type: %s", minerType)
|
|
}
|
|
|
|
if err := miner.Uninstall(); err != nil {
|
|
return fmt.Errorf("failed to uninstall miner files: %w", err)
|
|
}
|
|
|
|
cfg, err := LoadMinersConfig()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to load miners config to update uninstall status: %w", err)
|
|
}
|
|
|
|
var updatedMiners []MinerAutostartConfig
|
|
for _, minerCfg := range cfg.Miners {
|
|
if !strings.EqualFold(minerCfg.MinerType, minerType) {
|
|
updatedMiners = append(updatedMiners, minerCfg)
|
|
}
|
|
}
|
|
cfg.Miners = updatedMiners
|
|
|
|
return SaveMinersConfig(cfg)
|
|
}
|
|
|
|
// updateMinerConfig saves the autostart and last-used config for a miner.
|
|
func (m *Manager) updateMinerConfig(minerType string, autostart bool, config *Config) error {
|
|
cfg, err := LoadMinersConfig()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
found := false
|
|
for i, minerCfg := range cfg.Miners {
|
|
if strings.EqualFold(minerCfg.MinerType, minerType) {
|
|
cfg.Miners[i].Autostart = autostart
|
|
cfg.Miners[i].Config = config
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
|
|
if !found {
|
|
cfg.Miners = append(cfg.Miners, MinerAutostartConfig{
|
|
MinerType: minerType,
|
|
Autostart: autostart,
|
|
Config: config,
|
|
})
|
|
}
|
|
|
|
return SaveMinersConfig(cfg)
|
|
}
|
|
|
|
// StopMiner stops a running miner.
|
|
func (m *Manager) StopMiner(name string) error {
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
|
|
miner, exists := m.miners[name]
|
|
if !exists {
|
|
for k := range m.miners {
|
|
if strings.HasPrefix(k, name) {
|
|
miner = m.miners[k]
|
|
name = k
|
|
exists = true
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
if !exists {
|
|
return fmt.Errorf("miner not found: %s", name)
|
|
}
|
|
|
|
if err := miner.Stop(); err != nil {
|
|
return err
|
|
}
|
|
|
|
delete(m.miners, name)
|
|
return nil
|
|
}
|
|
|
|
// GetMiner retrieves a running miner by its name.
|
|
func (m *Manager) GetMiner(name string) (Miner, error) {
|
|
m.mu.RLock()
|
|
defer m.mu.RUnlock()
|
|
miner, exists := m.miners[name]
|
|
if !exists {
|
|
return nil, fmt.Errorf("miner not found: %s", name)
|
|
}
|
|
return miner, nil
|
|
}
|
|
|
|
// ListMiners returns a slice of all running miners.
|
|
func (m *Manager) ListMiners() []Miner {
|
|
m.mu.RLock()
|
|
defer m.mu.RUnlock()
|
|
miners := make([]Miner, 0, len(m.miners))
|
|
for _, miner := range m.miners {
|
|
miners = append(miners, miner)
|
|
}
|
|
return miners
|
|
}
|
|
|
|
// ListAvailableMiners returns a list of available miners that can be started.
|
|
func (m *Manager) ListAvailableMiners() []AvailableMiner {
|
|
return []AvailableMiner{
|
|
{
|
|
Name: "xmrig",
|
|
Description: "XMRig is a high performance, open source, cross platform RandomX, KawPow, CryptoNight and AstroBWT CPU/GPU miner and RandomX benchmark.",
|
|
},
|
|
{
|
|
Name: "tt-miner",
|
|
Description: "TT-Miner is a high performance NVIDIA GPU miner for various algorithms including Ethash, KawPow, ProgPow, and more. Requires CUDA.",
|
|
},
|
|
}
|
|
}
|
|
|
|
// startStatsCollection starts a goroutine to periodically collect stats from active miners.
|
|
func (m *Manager) startStatsCollection() {
|
|
m.waitGroup.Add(1)
|
|
go func() {
|
|
defer m.waitGroup.Done()
|
|
ticker := time.NewTicker(HighResolutionInterval)
|
|
defer ticker.Stop()
|
|
|
|
for {
|
|
select {
|
|
case <-ticker.C:
|
|
m.collectMinerStats()
|
|
case <-m.stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
// collectMinerStats iterates through active miners and collects their stats.
|
|
func (m *Manager) collectMinerStats() {
|
|
m.mu.RLock()
|
|
minersToCollect := make([]Miner, 0, len(m.miners))
|
|
for _, miner := range m.miners {
|
|
minersToCollect = append(minersToCollect, miner)
|
|
}
|
|
m.mu.RUnlock()
|
|
|
|
now := time.Now()
|
|
for _, miner := range minersToCollect {
|
|
stats, err := miner.GetStats()
|
|
if err != nil {
|
|
log.Printf("Error getting stats for miner %s: %v\n", miner.GetName(), err)
|
|
continue
|
|
}
|
|
miner.AddHashratePoint(HashratePoint{
|
|
Timestamp: now,
|
|
Hashrate: stats.Hashrate,
|
|
})
|
|
miner.ReduceHashrateHistory(now)
|
|
}
|
|
}
|
|
|
|
// GetMinerHashrateHistory returns the hashrate history for a specific miner.
|
|
func (m *Manager) GetMinerHashrateHistory(name string) ([]HashratePoint, error) {
|
|
m.mu.RLock()
|
|
defer m.mu.RUnlock()
|
|
miner, exists := m.miners[name]
|
|
if !exists {
|
|
return nil, fmt.Errorf("miner not found: %s", name)
|
|
}
|
|
return miner.GetHashrateHistory(), nil
|
|
}
|
|
|
|
// Stop stops the manager and its background goroutines.
|
|
func (m *Manager) Stop() {
|
|
close(m.stopChan)
|
|
m.waitGroup.Wait()
|
|
}
|
|
|
|
// Helper to convert port to string for net.JoinHostPort
|
|
func portToString(port int) string {
|
|
return strconv.Itoa(port)
|
|
}
|