Files
salvium-rs/mine-testnet.js
Matt Hess 7f01cafc62 Crypto layer consolidation: eliminate JS scalar ops, add cn_scan + RCT batch verify
Crypto Backend Refactoring
   ──────────────────────────
   - Delete src/ed25519.js — all scalar/point operations now route through
     the crypto provider (WASM/FFI/JSI backends only)
   - Remove duplicate JS BigInt implementations of scReduce32, scReduce64,
     scalarAdd, scalarMul from scanning.js, carrot.js, and carrot-scanning.js;
     delegate to Rust backend via crypto/index.js
   - Remove debug/test exports from index.js (randomPoint, testDouble,
     getBasePoint, isOnCurve, etc.) that were only used during development
   - Rebuild WASM binary (476KB → 487KB) with updated Rust crate

   Consolidated CryptoNote Scanner (cn_scan)
   ─────────────────────────────────────────
   - New Rust module crates/salvium-crypto/src/cn_scan.rs replaces 5-12
     individual FFI round-trips per output with a single native call:
     view tag check → derive subaddress pubkey → subaddress map lookup →
     amount decryption → commitment mask → key image generation
   - FFI wrapper salvium_cn_scan_output in ffi.rs + C header declaration
   - FFI backend scanCnOutput() with full subaddress map marshaling
     (32-byte key + u32 major/minor LE per entry) and JSON result parsing
   - JSI backend delegation via this.native.cnScanOutput()
   - wallet-sync.js _scanCNOutput() tries native path first when available
     (FFI/JSI), falls through to existing JS pipeline for WASM/JS backends
   - Change pub(crate) visibility on subaddress.rs cn_subaddress_secret_key
   - 8 Rust unit tests covering view tag, amount, commitment mask,
     subaddress matching, and key image generation
   - Verified identical results: WASM (JS fallback) and FFI (native cn_scan)
     produce same 964 outputs at same chain height; FFI is 3x faster sync
     (0.8s vs 2.5s) with 12x less heap (12MB vs 150MB)

   RCT Batch Signature Verification
   ─────────────────────────────────
   - New Rust module crates/salvium-crypto/src/rct_verify.rs — single-call
     verification of all ring signatures in a transaction (CLSAG + TCLSAG),
     avoiding N individual JS↔Rust boundary crossings
   - Computes pre-MLSAG message hash matching C++ get_pre_mlsag_hash
   - FFI export salvium_verify_rct_signatures with flat byte array interface
   - FFI backend verifyRctSignatures() method
   - JS backend stub returns null (validation.js handles JS fallback)
   - validation.js: 200+ lines of RCT verification logic including
     flattenKeyImages, packTclsagSigsFlat, packClsagSigsFlat helpers

   Transaction Expansion
   ─────────────────────
   - transaction.js: add expandTransaction() matching C++ expand_transaction_2
     (copies key images from prefix inputs into TCLSAG/CLSAG signature structs)
   - New test/expand-transaction.test.js (634 lines)
   - New test/rct-verify-testnet.test.js (430 lines)

   Mining Resilience
   ─────────────────
   - salvium-miner main.rs: retry get_info and get_block_template up to 5
     times with 2s delay for transient daemon errors
   - full-testnet.js mineTo(): retry miner up to 3 times with 3s delay,
     check for partial progress between attempts

   Testnet Tooling
   ───────────────
   - sync-only.js: CRYPTO_BACKEND env var for A/B testing (wasm vs ffi)
   - full-testnet.js: daemon URL update (node12.whiskymine.io)
   - Debug scripts for cn_scan development (debug-cn-scan/marshal/match/wasm)
   - Android .gitignore and build-bundle.sh for mobile builds
2026-02-14 17:02:35 +00:00

188 lines
6.0 KiB
JavaScript

#!/usr/bin/env bun
/**
* Local RandomX miner against remote Salvium daemon
*
* Uses our WASM RandomX (full mode, 2GB dataset) + daemon RPC.
* Single-threaded but with full dataset for maximum per-thread hashrate.
*
* Usage:
* bun mine-testnet.js [mode]
* mode: "full" (default, 2GB dataset, ~30-50 H/s) or "light" (256MB, ~10 H/s)
*/
import { DaemonRPC } from './src/rpc/daemon.js';
import { RandomXContext } from './src/randomx/index.js';
import { RandomXFullMode } from './src/randomx/full-mode.js';
import {
parseBlockTemplate,
findNonceOffset,
setNonce,
checkHash,
formatBlockForSubmission,
formatHashrate,
} from './src/mining.js';
import { hexToBytes, bytesToHex } from './src/address.js';
// ============================================================================
// Configuration
// ============================================================================
const DAEMON_URL = process.env.DAEMON_URL || 'http://node12.whiskymine.io:29081';
const WALLET_FILE = process.env.WALLET_FILE || `${process.env.HOME}/testnet-wallet/wallet.json`;
const MODE = process.argv[2] || 'full';
const TEMPLATE_REFRESH_MS = 5000;
// ============================================================================
// Main
// ============================================================================
const wallet = JSON.parse(await Bun.file(WALLET_FILE).text());
const daemon = new DaemonRPC({ url: DAEMON_URL });
console.log('Salvium Local RandomX Miner');
console.log('===========================');
console.log(`Daemon: ${DAEMON_URL}`);
console.log(`Wallet: ${wallet.address.slice(0, 20)}...`);
console.log(`Mode: ${MODE} (${MODE === 'full' ? '2GB dataset' : '256MB cache'})`);
console.log('');
// Verify daemon
const info = await daemon.getInfo();
if (!info.success) {
console.error('Cannot connect to daemon');
process.exit(1);
}
console.log(`Daemon height: ${info.result.height}, difficulty: ${info.result.difficulty}`);
console.log('');
// Get block template
let template = await daemon.getBlockTemplate(wallet.address, 8);
if (!template.success) {
console.error('Failed to get block template:', template.error);
process.exit(1);
}
let parsed = parseBlockTemplate(template.result);
console.log(`Template: height=${parsed.height}, difficulty=${parsed.difficulty}`);
// Initialize RandomX
const seedBytes = parsed.seedHash ? hexToBytes(parsed.seedHash) : new Uint8Array(32);
let rx;
if (MODE === 'full') {
console.log('Initializing RandomX full mode (256MB cache + 2GB dataset)...');
console.log('This will take 30-60 seconds...');
rx = new RandomXFullMode();
await rx.init(seedBytes, {
onProgress: (pct, phase) => {
process.stdout.write(`\r ${phase}: ${pct}% `);
}
});
console.log('');
} else {
console.log('Initializing RandomX light mode (256MB cache)...');
rx = new RandomXContext();
await rx.init(seedBytes);
}
console.log('RandomX ready.\n');
// ============================================================================
// Mining loop
// ============================================================================
let totalHashes = 0;
let blocksFound = 0;
const startTime = Date.now();
let lastTemplateFetch = Date.now();
let lastStatsPrint = Date.now();
let currentSeedHash = parsed.seedHash;
let running = true;
process.on('SIGINT', () => {
running = false;
console.log('\nShutting down...');
const elapsed = (Date.now() - startTime) / 1000;
console.log(`Total hashes: ${totalHashes.toLocaleString()}`);
console.log(`Blocks found: ${blocksFound}`);
console.log(`Avg hashrate: ${formatHashrate(totalHashes / elapsed)}`);
process.exit(0);
});
console.log('Mining started. Press Ctrl+C to stop.\n');
while (running) {
const hashingBlob = parsed.blockhashingBytes;
const nonceOffset = findNonceOffset(hashingBlob);
const difficulty = parsed.difficulty;
const templateBlob = parsed.blocktemplateBytes;
// Random starting nonce
const startNonce = Math.floor(Math.random() * 0xFFFFFFFF);
for (let i = 0; i < 256 && running; i++) {
const nonce = (startNonce + i) & 0xFFFFFFFF;
const blob = setNonce(hashingBlob, nonce, nonceOffset);
const hash = rx.hash(blob);
totalHashes++;
if (checkHash(hash, difficulty)) {
const blockHex = formatBlockForSubmission(templateBlob, nonce,
findNonceOffset(templateBlob));
console.log(`\n*** BLOCK FOUND at height ${parsed.height}! nonce=${nonce} ***`);
const submitResult = await daemon.submitBlock([blockHex]);
if (submitResult.success) {
blocksFound++;
console.log(`Block accepted! Total: ${blocksFound}`);
} else {
console.log(`Rejected: ${JSON.stringify(submitResult.error || submitResult.result)}`);
}
// New template immediately
lastTemplateFetch = 0;
break;
}
}
// Refresh template
const now = Date.now();
if (now - lastTemplateFetch > TEMPLATE_REFRESH_MS) {
try {
const newTemplate = await daemon.getBlockTemplate(wallet.address, 8);
if (newTemplate.success) {
const newParsed = parseBlockTemplate(newTemplate.result);
if (newParsed.seedHash !== currentSeedHash) {
console.log(`\nSeed changed, re-initializing RandomX...`);
if (MODE === 'full') {
await rx.init(hexToBytes(newParsed.seedHash), {});
} else {
await rx.init(hexToBytes(newParsed.seedHash));
}
currentSeedHash = newParsed.seedHash;
}
parsed = newParsed;
}
lastTemplateFetch = now;
} catch (e) {
// Keep mining
}
}
// Stats every 10s
if (now - lastStatsPrint > 10000) {
const elapsed = (now - startTime) / 1000;
const hr = totalHashes / elapsed;
const estBlockTime = Number(parsed.difficulty) / hr;
process.stdout.write(
`\r[H=${parsed.height}] ${formatHashrate(hr)} | ` +
`Hashes: ${totalHashes.toLocaleString()} | ` +
`Blocks: ${blocksFound} | ` +
`Diff: ${parsed.difficulty} | ` +
`Est: ${isFinite(estBlockTime) ? estBlockTime.toFixed(0) + 's' : '...'} `
);
lastStatsPrint = now;
}
}