This commit is contained in:
2026-06-03 01:27:00 +02:00
parent 99cf5eaef2
commit 7783f65b26
3 changed files with 303 additions and 59 deletions

View File

@@ -805,8 +805,7 @@
runtime_months: 1,
mining_speed_value: '',
mining_speed_unit: 'MH/s',
bonus_speed_value: '',
bonus_speed_unit: 'MH/s',
bonus_percent: '',
auto_renew: true,
base_price_amount: '',
payment_type: 'fiat',
@@ -826,10 +825,7 @@
const [minerOfferForm, setMinerOfferForm] = useState({
label: '',
runtime_months: '',
mining_speed_value: '',
mining_speed_unit: 'MH/s',
bonus_speed_value: '',
bonus_speed_unit: 'MH/s',
bonus_percent: '',
base_price_amount: '',
base_price_currency: 'USD',
payment_type: 'fiat',
@@ -841,9 +837,14 @@
const [purchaseMinerModalOpen, setPurchaseMinerModalOpen] = useState(false);
const [purchaseMinerForm, setPurchaseMinerForm] = useState({
offer_id: '',
base_offer_id: '',
purchased_at: '',
label: '',
mining_speed_value: '',
mining_speed_unit: '',
bonus_percent: '',
total_cost_amount: '',
currency: 'USD',
currency: '',
reference_price_amount: '',
reference_price_currency: '',
auto_renew: false,
@@ -941,7 +942,7 @@
: cryptoCurrencies;
const selectableFiatCurrencies = preferredSelectableFiatCurrencies.length ? preferredSelectableFiatCurrencies : fiatCurrencies;
const selectableCryptoCurrencies = preferredSelectableCryptoCurrencies.length ? preferredSelectableCryptoCurrencies : cryptoCurrencies;
const selectedMinerScenario = availableMinerOffers.find((offer) => Number(offer.id) === Number(selectedMinerScenarioId)) || null;
const selectedMinerScenario = availableMinerOffers.find((offer) => String(offer.id) === String(selectedMinerScenarioId)) || null;
const activeMinerRows = currentPurchasedMiners.map((miner) => ({
id: `purchase-${miner.id}`,
source: 'miete',
@@ -962,7 +963,7 @@
: 'fiat',
is_active: miner.is_active !== false,
can_toggle_auto_renew: Number(miner.runtime_months) > 0 && renewableOfferIds.has(Number(miner.miner_offer_id)),
hashrate_text: `${fmtNumber(((Number(miner.mining_speed_value) || 0) + (Number(miner.bonus_speed_value) || 0)), 4)} ${miner.mining_speed_unit || miner.bonus_speed_unit || ''}`.trim(),
hashrate_text: formatHashrateWithBonus(miner.mining_speed_value, miner.mining_speed_unit, miner.bonus_speed_value, miner.bonus_speed_unit),
type_label: 'Aus Angebot gemietet',
})).concat(currentCostPlans.map((plan) => ({
id: `plan-${plan.id}`,
@@ -978,10 +979,7 @@
payment_type: plan.payment_type,
is_active: !!plan.is_active,
can_toggle_auto_renew: false,
hashrate_text: [
formatSpeed(plan.mining_speed_value, plan.mining_speed_unit, 'Basis'),
formatSpeed(plan.bonus_speed_value, plan.bonus_speed_unit, 'Bonus'),
].filter(Boolean).join(' · ') || 'n/a',
hashrate_text: formatHashrateWithBonus(plan.mining_speed_value, plan.mining_speed_unit, plan.bonus_speed_value, plan.bonus_speed_unit),
type_label: 'Manuell eingetragen',
}))).sort((left, right) => String(right.starts_at || '').localeCompare(String(left.starts_at || '')));
@@ -996,7 +994,7 @@
return;
}
const exists = availableMinerOffers.some((offer) => Number(offer.id) === Number(selectedMinerScenarioId));
const exists = availableMinerOffers.some((offer) => String(offer.id) === String(selectedMinerScenarioId));
if (!exists) {
setSelectedMinerScenarioId(null);
}
@@ -1017,6 +1015,11 @@
setPurchaseMinerForm((current) => ({
...current,
offer_id: current.offer_id || String(selectedOffer.id),
base_offer_id: current.base_offer_id || String(selectedOffer.base_offer_id || selectedOffer.id || ''),
label: current.label || String(selectedOffer.label || ''),
mining_speed_value: current.mining_speed_value || String(selectedOffer.mining_speed_value || ''),
mining_speed_unit: current.mining_speed_unit || String(selectedOffer.mining_speed_unit || ''),
bonus_percent: current.bonus_percent || (selectedOffer.bonus_percent !== null && selectedOffer.bonus_percent !== undefined ? String(selectedOffer.bonus_percent) : ''),
currency: current.currency || (!selectedOffer.auto_renew ? (currentSettings.crypto_currency || 'DOGE') : (selectedOffer.effective_price_currency || selectedOffer.base_price_currency || 'USD')),
total_cost_amount: current.total_cost_amount || (selectedOffer.effective_price_amount !== null && selectedOffer.effective_price_amount !== undefined ? String(selectedOffer.effective_price_amount) : ''),
reference_price_amount: current.reference_price_amount || (selectedOffer.reference_price_amount !== null && selectedOffer.reference_price_amount !== undefined ? String(selectedOffer.reference_price_amount) : ''),
@@ -1672,8 +1675,7 @@
runtime_months: 1,
mining_speed_value: '',
mining_speed_unit: 'MH/s',
bonus_speed_value: '',
bonus_speed_unit: 'MH/s',
bonus_percent: '',
auto_renew: true,
base_price_amount: '',
payment_type: 'fiat',
@@ -1726,10 +1728,7 @@
setMinerOfferForm({
label: '',
runtime_months: '',
mining_speed_value: '',
mining_speed_unit: 'MH/s',
bonus_speed_value: '',
bonus_speed_unit: 'MH/s',
bonus_percent: '',
base_price_amount: '',
base_price_currency: 'USD',
payment_type: 'fiat',
@@ -1756,6 +1755,21 @@
body: JSON.stringify(overrides || { purchased_at: nowDateTimeLocalValue() }),
});
setMessage('Miner als gemietet erfasst.');
setPurchaseMinerForm({
offer_id: '',
base_offer_id: '',
purchased_at: '',
label: '',
mining_speed_value: '',
mining_speed_unit: '',
bonus_percent: '',
total_cost_amount: '',
currency: '',
reference_price_amount: '',
reference_price_currency: '',
auto_renew: false,
note: '',
});
setPurchaseMinerModalOpen(false);
await loadBootstrap(projectKey);
} catch (err) {
@@ -1772,7 +1786,17 @@
return;
}
await purchaseMinerOffer(purchaseMinerForm.offer_id, {
const selectedOffer = availableMinerOffers.find((offer) => String(offer.id) === String(purchaseMinerForm.offer_id));
if (!selectedOffer) {
setError('Bitte ein gueltiges Miner-Angebot auswaehlen.');
return;
}
await purchaseMinerOffer(String(selectedOffer.base_offer_id || selectedOffer.id), {
label: purchaseMinerForm.label || null,
mining_speed_value: purchaseMinerForm.mining_speed_value || null,
mining_speed_unit: purchaseMinerForm.mining_speed_unit || null,
bonus_percent: purchaseMinerForm.bonus_percent || null,
purchased_at: purchaseMinerForm.purchased_at || nowDateTimeLocalValue(),
total_cost_amount: purchaseMinerForm.total_cost_amount || null,
currency: purchaseMinerForm.currency || null,
@@ -2413,7 +2437,24 @@
key: 'rent-miner',
type: 'button',
className: 'mc-button mc-button--ghost',
onClick: () => setPurchaseMinerModalOpen(true),
onClick: () => {
setPurchaseMinerForm({
offer_id: '',
base_offer_id: '',
purchased_at: '',
label: '',
mining_speed_value: '',
mining_speed_unit: '',
bonus_percent: '',
total_cost_amount: '',
currency: '',
reference_price_amount: '',
reference_price_currency: '',
auto_renew: false,
note: '',
});
setPurchaseMinerModalOpen(true);
},
disabled: !availableMinerOffers.length,
}, 'Neuen Miner mieten'),
]),
@@ -2493,14 +2534,14 @@
]),
]),
]),
panel('Miner-Angebote', 'Angebote fuer neue Miner und eine grobe Reinvestitionsbewertung auf Basis der aktuellen Leistung.', [
panel('Miner-Angebote', 'Hier werden nur Basis-Miner gepflegt. Alle vorgegebenen Geschwindigkeiten und Preise werden daraus automatisch abgeleitet.', [
h('div', { key: 'actions', className: 'mc-inline-row' }, [
h('button', {
key: 'add-offer',
type: 'button',
className: 'mc-button mc-button--secondary',
onClick: () => setMinerOfferModalOpen(true),
}, 'Miner-Angebot anlegen'),
}, 'Basis-Angebot anlegen'),
]),
h('div', { key: 'filters', className: 'mc-filter-grid' }, [
inputField(`Min. Geschwindigkeit (${minerOfferFilters.speed_unit === 'kh' ? 'kH/s' : 'MH/s'})`, 'number', minerOfferFilters.speed_min, (value) => setMinerOfferFilters({ ...minerOfferFilters, speed_min: value }), '0.0001'),
@@ -2548,7 +2589,7 @@
h('button', {
key: 'scenario',
type: 'button',
className: cx('mc-button', Number(selectedMinerScenario?.id) === Number(offer.id) ? 'mc-button--secondary' : 'mc-button--ghost'),
className: cx('mc-button', String(selectedMinerScenario?.id) === String(offer.id) ? 'mc-button--secondary' : 'mc-button--ghost'),
onClick: () => setSelectedMinerScenarioId(offer.id),
}, 'Szenario'),
h('button', {
@@ -2560,7 +2601,7 @@
label: offer.label,
target_amount_fiat: String(offer.base_price_amount ?? offer.effective_price_amount ?? ''),
currency: offer.base_price_currency || offer.effective_price_currency || 'EUR',
miner_offer_id: String(offer.id),
miner_offer_id: '',
is_active: true,
sort_order: 0,
});
@@ -2574,7 +2615,12 @@
onClick: () => {
setPurchaseMinerForm({
offer_id: String(offer.id),
base_offer_id: String(offer.base_offer_id || offer.id || ''),
purchased_at: nowDateTimeLocalValue(),
label: String(offer.label || ''),
mining_speed_value: String(offer.mining_speed_value || ''),
mining_speed_unit: String(offer.mining_speed_unit || ''),
bonus_percent: offer.bonus_percent !== null && offer.bonus_percent !== undefined ? String(offer.bonus_percent) : '',
total_cost_amount: offer.effective_price_amount !== null && offer.effective_price_amount !== undefined ? String(offer.effective_price_amount) : '',
currency: offer.effective_price_currency || offer.base_price_currency || 'USD',
reference_price_amount: offer.base_price_amount !== null && offer.base_price_amount !== undefined ? String(offer.base_price_amount) : '',
@@ -2702,8 +2748,7 @@
inputField('Laufzeit in Monaten', 'number', String(costPlanForm.runtime_months), (value) => setCostPlanForm({ ...costPlanForm, runtime_months: Number(value) || 0 })),
inputField('Mining-Geschwindigkeit', 'number', costPlanForm.mining_speed_value, (value) => setCostPlanForm({ ...costPlanForm, mining_speed_value: value }), '0.0001'),
selectField('Mining-Einheit', costPlanForm.mining_speed_unit, speedUnits, (value) => setCostPlanForm({ ...costPlanForm, mining_speed_unit: value })),
inputField('Bonus-Geschwindigkeit', 'number', costPlanForm.bonus_speed_value, (value) => setCostPlanForm({ ...costPlanForm, bonus_speed_value: value }), '0.0001'),
selectField('Bonus-Einheit', costPlanForm.bonus_speed_unit, speedUnits, (value) => setCostPlanForm({ ...costPlanForm, bonus_speed_unit: value })),
inputField('Bonus-Hashrate in %', 'number', costPlanForm.bonus_percent, (value) => setCostPlanForm({ ...costPlanForm, bonus_percent: value }), '0.01'),
inputField(`Basispreis in ${settingsForm.report_currency || 'EUR'}`, 'number', costPlanForm.base_price_amount, (value) => setCostPlanForm({ ...costPlanForm, base_price_amount: value }), '0.000001'),
selectField('Zahlungsart', costPlanForm.payment_type, [{ value: 'fiat', label: 'FIAT' }, { value: 'crypto', label: 'Krypto' }], (value) => setCostPlanForm({ ...costPlanForm, payment_type: value })),
textareaField('Notiz', costPlanForm.note, (value) => setCostPlanForm({ ...costPlanForm, note: value })),
@@ -2733,17 +2778,21 @@
]),
]),
], () => setPayoutModalOpen(false)) : null,
minerOfferModalOpen ? renderModal('Miner-Angebot anlegen', [
minerOfferModalOpen ? renderModal('Basis-Miner-Angebot anlegen', [
h('form', { key: 'form', className: 'mc-form', onSubmit: submitMinerOffer }, [
inputField('Label', 'text', minerOfferForm.label, (value) => setMinerOfferForm({ ...minerOfferForm, label: value })),
inputField('Laufzeit in Monaten', 'number', minerOfferForm.runtime_months, (value) => setMinerOfferForm({ ...minerOfferForm, runtime_months: value })),
inputField('Mining-Geschwindigkeit', 'number', minerOfferForm.mining_speed_value, (value) => setMinerOfferForm({ ...minerOfferForm, mining_speed_value: value }), '0.0001'),
selectField('Mining-Einheit', minerOfferForm.mining_speed_unit, speedUnits, (value) => setMinerOfferForm({ ...minerOfferForm, mining_speed_unit: value })),
inputField('Bonus-Geschwindigkeit', 'number', minerOfferForm.bonus_speed_value, (value) => setMinerOfferForm({ ...minerOfferForm, bonus_speed_value: value }), '0.0001'),
selectField('Bonus-Einheit', minerOfferForm.bonus_speed_unit, speedUnits, (value) => setMinerOfferForm({ ...minerOfferForm, bonus_speed_unit: value })),
displayField('Basis-Geschwindigkeit', baseOfferSpeedLabel(minerOfferForm.payment_type)),
inputField('Bonus-Hashrate in %', 'number', minerOfferForm.bonus_percent, (value) => setMinerOfferForm({ ...minerOfferForm, bonus_percent: value }), '0.01'),
inputField('Basispreis', 'number', minerOfferForm.base_price_amount, (value) => setMinerOfferForm({ ...minerOfferForm, base_price_amount: value }), '0.000001'),
selectField('Basiswährung', minerOfferForm.base_price_currency, selectableFiatCurrencies.map((currency) => currency.code), (value) => setMinerOfferForm({ ...minerOfferForm, base_price_currency: value })),
selectField('Zahlungsart', minerOfferForm.payment_type, [{ value: 'fiat', label: 'FIAT' }, { value: 'crypto', label: 'Krypto' }], (value) => setMinerOfferForm({ ...minerOfferForm, payment_type: value })),
selectField('Zahlungsart', minerOfferForm.payment_type, [{ value: 'fiat', label: 'FIAT' }, { value: 'crypto', label: 'Krypto' }], (value) => setMinerOfferForm({
...minerOfferForm,
payment_type: value,
base_price_currency: value === 'crypto'
? (currentSettings.crypto_currency || selectableCryptoCurrencies[0]?.code || 'DOGE')
: (selectableFiatCurrencies[0]?.code || 'USD'),
})),
selectField('Basiswährung', minerOfferForm.base_price_currency, (minerOfferForm.payment_type === 'crypto' ? selectableCryptoCurrencies : selectableFiatCurrencies).map((currency) => currency.code), (value) => setMinerOfferForm({ ...minerOfferForm, base_price_currency: value })),
h('label', { className: 'mc-checkbox' }, [
h('input', { type: 'checkbox', checked: !!minerOfferForm.auto_renew, onChange: (event) => setMinerOfferForm({ ...minerOfferForm, auto_renew: event.target.checked }) }),
'Automatische Verlängerung',
@@ -2768,7 +2817,12 @@
const offer = availableMinerOffers.find((item) => String(item.id) === String(value));
setPurchaseMinerForm({
offer_id: value,
base_offer_id: offer ? String(offer.base_offer_id || offer.id || '') : '',
purchased_at: purchaseMinerForm.purchased_at || nowDateTimeLocalValue(),
label: offer ? String(offer.label || '') : '',
mining_speed_value: offer && offer.mining_speed_value !== null && offer.mining_speed_value !== undefined ? String(offer.mining_speed_value) : '',
mining_speed_unit: offer?.mining_speed_unit || '',
bonus_percent: offer && offer.bonus_percent !== null && offer.bonus_percent !== undefined ? String(offer.bonus_percent) : '',
total_cost_amount: offer && offer.effective_price_amount !== null && offer.effective_price_amount !== undefined ? String(offer.effective_price_amount) : '',
currency: offer?.effective_price_currency || offer?.base_price_currency || 'USD',
reference_price_amount: offer && offer.reference_price_amount !== null && offer.reference_price_amount !== undefined ? String(offer.reference_price_amount) : '',
@@ -2778,6 +2832,8 @@
});
}),
inputField('Mietdatum/-zeit', 'datetime-local', purchaseMinerForm.purchased_at, (value) => setPurchaseMinerForm({ ...purchaseMinerForm, purchased_at: value })),
displayField('Gewaehlte Geschwindigkeit', purchaseMinerForm.mining_speed_value && purchaseMinerForm.mining_speed_unit ? `${purchaseMinerForm.mining_speed_value} ${purchaseMinerForm.mining_speed_unit}` : 'n/a'),
displayField('Bonus-Hashrate', purchaseMinerForm.bonus_percent ? `${purchaseMinerForm.bonus_percent}%` : 'kein Bonus'),
inputField('Exakter Mietpreis', 'number', purchaseMinerForm.total_cost_amount, (value) => setPurchaseMinerForm({ ...purchaseMinerForm, total_cost_amount: value }), '0.000001'),
selectField('Mietwährung', purchaseMinerForm.currency, selectableCurrencies.map((currency) => currency.code), (value) => setPurchaseMinerForm({ ...purchaseMinerForm, currency: value })),
inputField('Referenzpreis', 'number', purchaseMinerForm.reference_price_amount, (value) => setPurchaseMinerForm({ ...purchaseMinerForm, reference_price_amount: value }), '0.000001'),
@@ -2796,11 +2852,11 @@
targetModalOpen ? renderModal('Ziel anlegen', [
h('form', { key: 'form', className: 'mc-form', onSubmit: submitTarget }, [
inputField('Label', 'text', targetForm.label, (value) => setTargetForm({ ...targetForm, label: value })),
selectField('Angebots-Verknuepfung', targetForm.miner_offer_id || '', [{ value: '', label: 'Kein verknuepftes Angebot' }].concat(availableMinerOffers.map((offer) => ({
selectField('Angebots-Verknuepfung', targetForm.miner_offer_id || '', [{ value: '', label: 'Kein verknuepftes Angebot' }].concat(currentMinerOffers.map((offer) => ({
value: String(offer.id),
label: `${offer.label} · ${fmtNumber(offer.base_price_amount ?? offer.effective_price_amount, 6)} ${offer.base_price_currency || offer.effective_price_currency}`,
label: `${offer.label} · ${fmtNumber(offer.base_price_amount, 6)} ${offer.base_price_currency}`,
}))), (value) => {
const offer = availableMinerOffers.find((item) => String(item.id) === String(value));
const offer = currentMinerOffers.find((item) => String(item.id) === String(value));
setTargetForm({
...targetForm,
miner_offer_id: value,
@@ -3051,6 +3107,18 @@
return `${label ? label + ' ' : ''}${fmtNumber(value, 4)} ${unit}`;
}
function baseOfferSpeedLabel(paymentType) {
return String(paymentType || 'fiat') === 'crypto' ? '75 kH/s' : '50 kH/s';
}
function formatHashrateWithBonus(speedValue, speedUnit, bonusValue, bonusUnit) {
const parts = [
formatSpeed(speedValue, speedUnit, 'Basis'),
Number(bonusValue) > 0 ? formatSpeed(bonusValue, bonusUnit, 'Bonus') : '',
].filter(Boolean);
return parts.length ? parts.join(' · ') : 'n/a';
}
function formatAdaptiveSpeed(valueMh) {
const numericValue = Number(valueMh);
if (!Number.isFinite(numericValue)) {

View File

@@ -24,6 +24,10 @@ final class Router
private const LONG_REQUEST_BUDGET_SECONDS = 8.0;
private const OVERVIEW_WINDOW_DAYS = 15;
private const FX_FETCH_MAX_AGE_HOURS = 3.0;
private const BASE_OFFER_SPEEDS = [
'fiat' => ['value' => 50.0, 'unit' => 'kH/s'],
'crypto' => ['value' => 75.0, 'unit' => 'kH/s'],
];
private string $moduleBasePath;
private ModuleConfig $config;
@@ -1679,14 +1683,18 @@ final class Router
private function saveCostPlan(string $projectKey, array $input): array
{
$projectTimezone = $this->projectTimezone($projectKey);
$miningSpeedValue = $this->optionalDecimal($input['mining_speed_value'] ?? null);
$miningSpeedUnit = $this->optionalSpeedUnit($input['mining_speed_unit'] ?? null);
$bonusPercent = $this->optionalDecimal($input['bonus_percent'] ?? null);
$payload = [
'label' => $this->requiredString($input['label'] ?? null, 'label', 120),
'purchased_at' => $this->requiredDateTime($input['starts_at'] ?? null, 'starts_at', $projectTimezone),
'runtime_months' => $this->requiredPositiveInt($input['runtime_months'] ?? null, 'runtime_months'),
'mining_speed_value' => $this->optionalDecimal($input['mining_speed_value'] ?? null),
'mining_speed_unit' => $this->optionalSpeedUnit($input['mining_speed_unit'] ?? null),
'bonus_speed_value' => $this->optionalDecimal($input['bonus_speed_value'] ?? null),
'bonus_speed_unit' => $this->optionalSpeedUnit($input['bonus_speed_unit'] ?? null),
'mining_speed_value' => $miningSpeedValue,
'mining_speed_unit' => $miningSpeedUnit,
'bonus_speed_value' => $this->bonusSpeedFromPercent($miningSpeedValue, $miningSpeedUnit, $bonusPercent),
'bonus_speed_unit' => $bonusPercent !== null ? $miningSpeedUnit : null,
'bonus_percent' => $bonusPercent,
'auto_renew' => !empty($input['auto_renew']) ? 1 : 0,
'base_price_amount' => $this->requiredDecimal($input['base_price_amount'] ?? ($input['total_cost_amount'] ?? null), 'base_price_amount'),
'payment_type' => $this->enumValue($input['payment_type'] ?? 'fiat', ['fiat', 'crypto'], 'payment_type'),
@@ -1698,8 +1706,11 @@ final class Router
throw new ApiException('Mining-Geschwindigkeit und Einheit muessen gemeinsam gesetzt oder beide leer sein.', 422);
}
if (($payload['bonus_speed_value'] === null) xor ($payload['bonus_speed_unit'] === null)) {
throw new ApiException('Bonus-Geschwindigkeit und Einheit muessen gemeinsam gesetzt oder beide leer sein.', 422);
if ($bonusPercent !== null && ($miningSpeedValue === null || $miningSpeedUnit === null)) {
throw new ApiException('Bonus-Prozent setzt eine Mining-Geschwindigkeit voraus.', 422);
}
if ($bonusPercent !== null && $bonusPercent < 0) {
throw new ApiException('Bonus-Prozent darf nicht negativ sein.', 422);
}
$settings = $this->settings($projectKey);
@@ -1787,22 +1798,31 @@ final class Router
private function saveMinerOffer(string $projectKey, array $input): array
{
$paymentType = $this->enumValue($input['payment_type'] ?? 'fiat', ['fiat', 'crypto'], 'payment_type');
$baseSpeed = self::BASE_OFFER_SPEEDS[$paymentType] ?? self::BASE_OFFER_SPEEDS['fiat'];
$bonusPercent = $this->optionalDecimal($input['bonus_percent'] ?? null);
$basePriceCurrency = $this->requiredCurrency($input['base_price_currency'] ?? ($input['reference_price_currency'] ?? $input['price_currency'] ?? null), 'base_price_currency');
$payload = [
'label' => $this->requiredString($input['label'] ?? null, 'label', 120),
'runtime_months' => $this->optionalPositiveInt($input['runtime_months'] ?? null),
'mining_speed_value' => $this->optionalDecimal($input['mining_speed_value'] ?? null),
'mining_speed_unit' => $this->optionalSpeedUnit($input['mining_speed_unit'] ?? null),
'bonus_speed_value' => $this->optionalDecimal($input['bonus_speed_value'] ?? null),
'bonus_speed_unit' => $this->optionalSpeedUnit($input['bonus_speed_unit'] ?? null),
'mining_speed_value' => $baseSpeed['value'],
'mining_speed_unit' => $baseSpeed['unit'],
'bonus_speed_value' => $this->bonusSpeedFromPercent((float) $baseSpeed['value'], (string) $baseSpeed['unit'], $bonusPercent),
'bonus_speed_unit' => $bonusPercent !== null ? (string) $baseSpeed['unit'] : null,
'bonus_percent' => $bonusPercent,
'base_price_amount' => $this->requiredDecimal($input['base_price_amount'] ?? ($input['reference_price_amount'] ?? $input['price_amount'] ?? null), 'base_price_amount'),
'base_price_currency' => $this->requiredCurrency($input['base_price_currency'] ?? ($input['reference_price_currency'] ?? $input['price_currency'] ?? null), 'base_price_currency'),
'payment_type' => $this->enumValue($input['payment_type'] ?? 'fiat', ['fiat', 'crypto'], 'payment_type'),
'base_price_currency' => $basePriceCurrency,
'payment_type' => $paymentType,
'auto_renew' => !empty($input['auto_renew']) ? 1 : 0,
'note' => $this->optionalString($input['note'] ?? null, 1000),
'is_active' => !empty($input['is_active']) ? 1 : 0,
];
$this->assertCurrencyType($payload['base_price_currency'], false, 'base_price_currency');
if ($bonusPercent !== null && $bonusPercent < 0) {
throw new ApiException('Bonus-Prozent darf nicht negativ sein.', 422);
}
$this->assertCurrencyType($payload['base_price_currency'], $paymentType === 'crypto', 'base_price_currency');
return $this->repository()->saveMinerOffer($projectKey, $payload);
}
@@ -1817,14 +1837,31 @@ final class Router
$settings = $this->settings($projectKey);
$cryptoCurrency = $this->requiredCurrency($settings['crypto_currency'] ?? 'DOGE', 'crypto_currency');
$isAutoRenew = array_key_exists('auto_renew', $input) ? !empty($input['auto_renew']) : !empty($offer['auto_renew']);
$selectedSpeedValue = $this->optionalDecimal($input['mining_speed_value'] ?? null);
$selectedSpeedUnit = $this->optionalSpeedUnit($input['mining_speed_unit'] ?? null);
if (($selectedSpeedValue === null) xor ($selectedSpeedUnit === null)) {
throw new ApiException('Mining-Geschwindigkeit und Einheit muessen gemeinsam gesetzt oder beide leer sein.', 422);
}
$resolvedSpeedValue = $selectedSpeedValue ?? $this->optionalDecimal($offer['mining_speed_value'] ?? null);
$resolvedSpeedUnit = $selectedSpeedUnit ?? $this->optionalSpeedUnit($offer['mining_speed_unit'] ?? null);
$resolvedBonusPercent = $this->offerBonusPercent($offer);
$resolvedBonusValue = $this->bonusSpeedFromPercent($resolvedSpeedValue, $resolvedSpeedUnit, $resolvedBonusPercent);
$resolvedBonusUnit = $resolvedBonusValue !== null ? $resolvedSpeedUnit : null;
$purchaseCurrency = $this->optionalCurrency($input['currency'] ?? null) ?? (string) ($offer['effective_price_currency'] ?? $offer['price_currency'] ?? $offer['base_price_currency'] ?? '');
if (!$isAutoRenew) {
$purchaseCurrency = $cryptoCurrency;
}
$purchaseCost = $this->optionalDecimal($input['total_cost_amount'] ?? null);
if ($purchaseCost === null || !$isAutoRenew) {
if ($purchaseCost === null) {
$purchaseCost = $this->resolveOfferPurchaseCost(array_merge($offer, [
'mining_speed_value' => $resolvedSpeedValue,
'mining_speed_unit' => $resolvedSpeedUnit,
'bonus_speed_value' => $resolvedBonusValue,
'bonus_speed_unit' => $resolvedBonusUnit,
'base_price_amount' => $this->optionalDecimal($input['reference_price_amount'] ?? null) ?? ($offer['base_price_amount'] ?? null),
'base_price_currency' => $this->optionalCurrency($input['reference_price_currency'] ?? null) ?? ($offer['base_price_currency'] ?? null),
'price_currency' => $purchaseCurrency !== '' ? $purchaseCurrency : ($offer['price_currency'] ?? ''),
]));
}
@@ -1836,12 +1873,12 @@ final class Router
return $this->repository()->purchaseMiner($projectKey, $offerId, [
'purchased_at' => $purchasedAt,
'label' => $offer['label'],
'label' => $this->optionalString($input['label'] ?? ($offer['label'] ?? null), 120) ?? $offer['label'],
'runtime_months' => $offer['runtime_months'] ?? null,
'mining_speed_value' => $offer['mining_speed_value'] ?? null,
'mining_speed_unit' => $offer['mining_speed_unit'] ?? null,
'bonus_speed_value' => $offer['bonus_speed_value'] ?? null,
'bonus_speed_unit' => $offer['bonus_speed_unit'] ?? null,
'mining_speed_value' => $resolvedSpeedValue,
'mining_speed_unit' => $resolvedSpeedUnit,
'bonus_speed_value' => $resolvedBonusValue,
'bonus_speed_unit' => $resolvedBonusUnit,
'total_cost_amount' => $purchaseCost,
'currency' => $purchaseCurrency !== '' ? $purchaseCurrency : $offer['price_currency'],
'usd_reference_amount' => $offer['usd_reference_amount'] ?? null,
@@ -2440,6 +2477,39 @@ final class Router
return (float) ($baseAmount ?? $offer['price_amount'] ?? 0);
}
private function bonusSpeedFromPercent(?float $speedValue, ?string $speedUnit, ?float $bonusPercent): ?float
{
if ($speedValue === null || $speedUnit === null || $bonusPercent === null) {
return null;
}
return round($speedValue * ($bonusPercent / 100), 4);
}
private function offerBonusPercent(array $offer): ?float
{
$speedValue = $this->optionalDecimal($offer['mining_speed_value'] ?? null);
$speedUnit = $this->optionalSpeedUnit($offer['mining_speed_unit'] ?? null);
$bonusValue = $this->optionalDecimal($offer['bonus_speed_value'] ?? null);
$bonusUnit = $this->optionalSpeedUnit($offer['bonus_speed_unit'] ?? null);
if ($speedValue === null || $speedUnit === null || $bonusValue === null || $bonusUnit === null) {
return null;
}
$speedMh = $this->hashrateToMh($speedValue, $speedUnit);
$bonusMh = $this->hashrateToMh($bonusValue, $bonusUnit);
if ($speedMh <= 0 || $bonusMh <= 0) {
return null;
}
return ($bonusMh / $speedMh) * 100;
}
private function hashrateToMh(float $value, string $unit): float
{
return $unit === 'kH/s' ? $value / 1000 : $value;
}
private function pdo(): PDO
{
if ($this->pdo === null) {

View File

@@ -7,6 +7,16 @@ use Modules\MiningChecker\Support\ApiException;
final class AnalyticsService
{
private const BASE_OFFER_SPEEDS = [
'fiat' => ['value' => 50.0, 'unit' => 'kH/s'],
'crypto' => ['value' => 75.0, 'unit' => 'kH/s'],
];
private const OFFER_SPEED_MULTIPLIERS = [
1, 2, 3, 4, 5, 6, 7, 8, 9, 10,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
];
private ?FxService $fx;
public function __construct(?FxService $fx = null)
@@ -245,7 +255,7 @@ final class AnalyticsService
$purchasedMiners
));
$offerSummary = [];
foreach ($minerOffers as $offer) {
foreach ($this->expandOfferVariants($minerOffers, $settings) as $offer) {
$offerSummary[] = $this->evaluateMinerOffer($offer, $latest, $latestPriceByCurrency, $currentHashrateMh, $settings);
}
@@ -256,7 +266,7 @@ final class AnalyticsService
$linkedOffer = null;
if (is_numeric($target['miner_offer_id'] ?? null)) {
foreach ($offerSummary as $offer) {
if ((int) ($offer['id'] ?? 0) === (int) $target['miner_offer_id']) {
if ((int) ($offer['base_offer_id'] ?? $offer['id'] ?? 0) === (int) $target['miner_offer_id']) {
$linkedOffer = $offer;
break;
}
@@ -823,6 +833,102 @@ final class AnalyticsService
};
}
private function expandOfferVariants(array $offers, array $settings): array
{
$expanded = [];
foreach ($offers as $offer) {
$paymentType = $this->normalizeOfferPaymentType($offer);
$baseSpeed = self::BASE_OFFER_SPEEDS[$paymentType] ?? self::BASE_OFFER_SPEEDS['fiat'];
$baseSpeedValue = (float) $baseSpeed['value'];
$baseSpeedUnit = (string) $baseSpeed['unit'];
$baseBonusPercent = $this->deriveBonusPercent($offer, $baseSpeedValue, $baseSpeedUnit);
$basePriceAmount = is_numeric($offer['base_price_amount'] ?? null)
? (float) $offer['base_price_amount']
: (is_numeric($offer['reference_price_amount'] ?? null)
? (float) $offer['reference_price_amount']
: (is_numeric($offer['usd_reference_amount'] ?? null)
? (float) $offer['usd_reference_amount']
: (is_numeric($offer['price_amount'] ?? null) ? (float) $offer['price_amount'] : null)));
foreach (self::OFFER_SPEED_MULTIPLIERS as $multiplier) {
$derivedSpeedValue = $baseSpeedValue * $multiplier;
$derivedBonusValue = $baseBonusPercent !== null
? round($derivedSpeedValue * ($baseBonusPercent / 100), 4)
: null;
$derivedPriceAmount = $basePriceAmount !== null
? round($basePriceAmount * $multiplier, 8)
: null;
$expanded[] = array_merge($offer, [
'id' => sprintf('%s:%s:%s', (string) ($offer['id'] ?? 'offer'), $paymentType, rtrim(rtrim(number_format($derivedSpeedValue, 4, '.', ''), '0'), '.')),
'base_offer_id' => $offer['id'] ?? null,
'base_offer_label' => $offer['label'] ?? null,
'payment_type' => $paymentType,
'label' => trim(((string) ($offer['label'] ?? 'Miner-Angebot')) . ' · ' . $this->formatSpeedLabel($derivedSpeedValue, $baseSpeedUnit)),
'mining_speed_value' => $derivedSpeedValue,
'mining_speed_unit' => $baseSpeedUnit,
'bonus_speed_value' => $derivedBonusValue,
'bonus_speed_unit' => $derivedBonusValue !== null ? $baseSpeedUnit : null,
'bonus_percent' => $baseBonusPercent,
'speed_factor' => $multiplier,
'base_speed_value' => $baseSpeedValue,
'base_speed_unit' => $baseSpeedUnit,
'base_price_amount' => $derivedPriceAmount,
'reference_price_amount' => $derivedPriceAmount,
]);
}
}
usort($expanded, static function (array $left, array $right): int {
$paymentCompare = strcmp((string) ($left['payment_type'] ?? ''), (string) ($right['payment_type'] ?? ''));
if ($paymentCompare !== 0) {
return $paymentCompare;
}
$runtimeCompare = ((int) ($left['runtime_months'] ?? 0)) <=> ((int) ($right['runtime_months'] ?? 0));
if ($runtimeCompare !== 0) {
return $runtimeCompare;
}
return ((int) ($left['speed_factor'] ?? 0)) <=> ((int) ($right['speed_factor'] ?? 0));
});
return $expanded;
}
private function normalizeOfferPaymentType(array $offer): string
{
$paymentType = strtolower(trim((string) ($offer['payment_type'] ?? '')));
if (in_array($paymentType, ['fiat', 'crypto'], true)) {
return $paymentType;
}
return !empty($offer['price_currency']) && in_array(strtoupper((string) $offer['price_currency']), ['ADA','ARB','BNB','BTC','DAI','DOGE','DOT','ETH','LINK','LTC','SOL','USDC','USDT','XRP'], true)
? 'crypto'
: 'fiat';
}
private function deriveBonusPercent(array $offer, float $speedValue, string $speedUnit): ?float
{
if (is_numeric($offer['bonus_percent'] ?? null)) {
$numeric = (float) $offer['bonus_percent'];
return $numeric >= 0 ? $numeric : null;
}
$baseSpeedMh = $this->normalizeHashrateMh($speedValue, $speedUnit);
$bonusMh = $this->normalizeHashrateMh($offer['bonus_speed_value'] ?? null, $offer['bonus_speed_unit'] ?? null);
if ($baseSpeedMh <= 0 || $bonusMh <= 0) {
return null;
}
return ($bonusMh / $baseSpeedMh) * 100;
}
private function formatSpeedLabel(float $value, string $unit): string
{
return rtrim(rtrim(number_format($value, 4, '.', ''), '0'), '.') . ' ' . $unit;
}
private function evaluateMinerOffer(array $offer, array $latest, array $latestPriceByCurrency, float $currentHashrateMh, array $settings): array
{
$offerHashrateMh = $this->normalizeHashrateMh($offer['mining_speed_value'] ?? null, $offer['mining_speed_unit'] ?? null)