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>
206 lines
6.6 KiB
TypeScript
206 lines
6.6 KiB
TypeScript
import { Component, ViewEncapsulation, CUSTOM_ELEMENTS_SCHEMA, inject, effect, signal, Input } from '@angular/core';
|
|
import { CommonModule } from '@angular/common';
|
|
import { HighchartsChartComponent, ChartConstructorType } from 'highcharts-angular';
|
|
import * as Highcharts from 'highcharts';
|
|
import { MinerService } from './miner.service';
|
|
|
|
// More specific type for series with data
|
|
type SeriesWithData = Highcharts.SeriesAreaOptions | Highcharts.SeriesSplineOptions;
|
|
|
|
@Component({
|
|
selector: 'snider-mining-chart',
|
|
standalone: true,
|
|
schemas: [CUSTOM_ELEMENTS_SCHEMA],
|
|
imports: [CommonModule, HighchartsChartComponent],
|
|
templateUrl: './chart.component.html',
|
|
styleUrls: ['./chart.component.css']
|
|
})
|
|
export class ChartComponent {
|
|
@Input() minerName?: string;
|
|
minerService = inject(MinerService);
|
|
|
|
Highcharts: typeof Highcharts = Highcharts;
|
|
chartConstructor: ChartConstructorType = 'chart';
|
|
chartOptions = signal<Highcharts.Options>({});
|
|
updateFlag = signal(false);
|
|
|
|
constructor() {
|
|
this.chartOptions.set(this.createBaseChartOptions());
|
|
|
|
effect(() => {
|
|
const historyMap = this.minerService.hashrateHistory();
|
|
let yAxisOptions: Highcharts.YAxisOptions = {};
|
|
|
|
if (this.minerName) {
|
|
// Single miner mode
|
|
const history = historyMap.get(this.minerName);
|
|
const chartData = history ? history.map(point => [new Date(point.timestamp).getTime(), point.hashrate]) : [];
|
|
|
|
yAxisOptions = this.calculateYAxisBoundsForSingle(chartData.map(d => d[1]));
|
|
|
|
this.chartOptions.update(options => ({
|
|
...options,
|
|
title: { text: `${this.minerName} Hashrate` },
|
|
chart: { type: 'spline' },
|
|
plotOptions: { area: undefined, spline: { marker: { enabled: false } } },
|
|
yAxis: { ...options.yAxis, ...yAxisOptions },
|
|
series: [{ type: 'spline', name: 'Hashrate', data: chartData }]
|
|
}));
|
|
|
|
} else {
|
|
// Overview mode
|
|
if (historyMap.size === 0) {
|
|
this.chartOptions.update(options => ({ ...options, series: [] }));
|
|
} else {
|
|
const newSeries: SeriesWithData[] = [];
|
|
historyMap.forEach((history, name) => {
|
|
const chartData = history.map(point => [new Date(point.timestamp).getTime(), point.hashrate]);
|
|
newSeries.push({ type: 'area', name: name, data: chartData });
|
|
});
|
|
|
|
yAxisOptions = this.calculateYAxisBoundsForStacked(newSeries);
|
|
|
|
this.chartOptions.update(options => ({
|
|
...options,
|
|
title: { text: 'Total Hashrate' },
|
|
chart: { type: 'area' },
|
|
plotOptions: { area: { stacking: 'normal', marker: { enabled: false } } },
|
|
yAxis: { ...options.yAxis, ...yAxisOptions },
|
|
series: newSeries
|
|
}));
|
|
}
|
|
}
|
|
|
|
this.updateFlag.update(flag => !flag);
|
|
});
|
|
}
|
|
|
|
private calculateYAxisBoundsForSingle(data: number[]): Highcharts.YAxisOptions {
|
|
if (data.length === 0) {
|
|
return { min: 0, max: undefined };
|
|
}
|
|
|
|
const min = Math.min(...data);
|
|
const max = Math.max(...data);
|
|
|
|
if (min === max) {
|
|
return { min: Math.max(0, min - 50), max: max + 50 };
|
|
}
|
|
|
|
const padding = (max - min) * 0.1; // 10% padding
|
|
|
|
return {
|
|
min: Math.max(0, min - padding),
|
|
max: max + padding
|
|
};
|
|
}
|
|
|
|
private calculateYAxisBoundsForStacked(series: SeriesWithData[]): Highcharts.YAxisOptions {
|
|
const totalsByTimestamp: { [key: number]: number } = {};
|
|
|
|
series.forEach(s => {
|
|
// Cast to any to avoid TS errors with union types where 'data' might be missing on some types
|
|
// even though we know SeriesWithData has it.
|
|
const data = (s as any).data;
|
|
if (data) {
|
|
(data as [number, number][]).forEach(([timestamp, value]) => {
|
|
totalsByTimestamp[timestamp] = (totalsByTimestamp[timestamp] || 0) + value;
|
|
});
|
|
}
|
|
});
|
|
|
|
const totalValues = Object.values(totalsByTimestamp);
|
|
if (totalValues.length === 0) {
|
|
return { min: 0, max: undefined };
|
|
}
|
|
|
|
const maxTotal = Math.max(...totalValues);
|
|
const padding = maxTotal * 0.1; // 10% padding on top
|
|
|
|
return {
|
|
min: 0, // Stacked chart should always start at 0
|
|
max: maxTotal + padding
|
|
};
|
|
}
|
|
|
|
createBaseChartOptions(): Highcharts.Options {
|
|
return {
|
|
chart: {
|
|
backgroundColor: 'transparent',
|
|
style: {
|
|
fontFamily: 'var(--wa-font-sans, system-ui, sans-serif)'
|
|
},
|
|
spacing: [10, 10, 10, 10]
|
|
},
|
|
title: { text: '' },
|
|
xAxis: {
|
|
type: 'datetime',
|
|
title: { text: '' },
|
|
lineColor: 'var(--wa-color-neutral-300)',
|
|
tickColor: 'var(--wa-color-neutral-300)',
|
|
labels: {
|
|
style: {
|
|
color: 'var(--wa-color-neutral-600)',
|
|
fontSize: '11px'
|
|
}
|
|
},
|
|
gridLineWidth: 0
|
|
},
|
|
yAxis: {
|
|
title: { text: '' },
|
|
labels: {
|
|
style: {
|
|
color: 'var(--wa-color-neutral-600)',
|
|
fontSize: '11px'
|
|
},
|
|
formatter: function() {
|
|
const val = this.value as number;
|
|
if (val >= 1000000) return (val / 1000000).toFixed(1) + ' MH/s';
|
|
if (val >= 1000) return (val / 1000).toFixed(1) + ' kH/s';
|
|
return val + ' H/s';
|
|
}
|
|
},
|
|
gridLineColor: 'var(--wa-color-neutral-200)',
|
|
gridLineDashStyle: 'Dash'
|
|
},
|
|
legend: {
|
|
enabled: false
|
|
},
|
|
tooltip: {
|
|
backgroundColor: 'var(--wa-color-neutral-900)',
|
|
borderColor: 'var(--wa-color-neutral-700)',
|
|
borderRadius: 8,
|
|
style: {
|
|
color: '#fff',
|
|
fontSize: '12px'
|
|
},
|
|
xDateFormat: '%H:%M:%S',
|
|
headerFormat: '<span style="font-size: 10px; opacity: 0.8">{point.key}</span><br/>',
|
|
pointFormatter: function() {
|
|
const val = this.y as number;
|
|
let formatted: string;
|
|
if (val >= 1000000) formatted = (val / 1000000).toFixed(2) + ' MH/s';
|
|
else if (val >= 1000) formatted = (val / 1000).toFixed(2) + ' kH/s';
|
|
else formatted = val.toFixed(0) + ' H/s';
|
|
return `<span style="color:${this.color}">●</span> ${this.series.name}: <b>${formatted}</b>`;
|
|
}
|
|
},
|
|
plotOptions: {
|
|
area: {
|
|
fillOpacity: 0.3,
|
|
lineWidth: 2,
|
|
marker: { enabled: false },
|
|
color: 'var(--wa-color-primary-600)'
|
|
},
|
|
spline: {
|
|
lineWidth: 2.5,
|
|
marker: { enabled: false },
|
|
color: 'var(--wa-color-primary-600)'
|
|
}
|
|
},
|
|
series: [],
|
|
credits: { enabled: false },
|
|
accessibility: { enabled: false }
|
|
};
|
|
}
|
|
}
|