Compare commits

..

50 Commits

Author SHA1 Message Date
Dobromir Popov
a1a35a6bd6 rin wallet 2025-09-24 11:20:28 +03:00
Dobromir Popov
2fd3146bdb notes & stats 2025-09-23 22:44:03 +03:00
Dobromir Popov
353b8120c8 net hashrate log 2025-09-23 20:37:12 +03:00
Dobromir Popov
7a806e53cb more difficulty fns 2025-09-23 18:38:28 +03:00
Dobromir Popov
a6c271d3f3 more proxy work 2025-09-23 18:20:17 +03:00
Dobromir Popov
5f9db3836f finaly working difficulty! 2025-09-23 18:09:55 +03:00
Dobromir Popov
ba93f8807b ajdust diff 2025-09-23 13:46:15 +03:00
Dobromir Popov
b5c0375569 stratuu - try fix block subm - proper difficulty 2025-09-23 13:33:10 +03:00
Dobromir Popov
4743ee4369 stratum - submit all blocks option. cleanup 2025-09-23 13:27:27 +03:00
Dobromir Popov
44d6d91103 stratum: "shares should now be accepted" 2025-09-23 11:52:01 +03:00
Dobromir Popov
210bfc5176 stratum - dynamic difficulty to 1m 2025-09-23 11:48:27 +03:00
Dobromir Popov
af58384949 stratum - dynamic difficulty 2025-09-23 11:46:07 +03:00
Dobromir Popov
d8b019401f stratum wip - valid block info 2025-09-23 11:39:03 +03:00
Dobromir Popov
bc50e52cdb stratum - fix miner info 2025-09-23 11:34:13 +03:00
Dobromir Popov
992a7704b6 stratum proxy wip 2025-09-23 11:29:56 +03:00
Dobromir Popov
747b8efb10 move readmes 2025-09-20 22:09:14 +03:00
Dobromir Popov
2bcd28be28 proxy fix 2025-09-17 17:53:40 +03:00
Dobromir Popov
15f41c3268 rename 2025-09-15 22:26:12 +03:00
Dobromir Popov
d6a5389a07 wip 2025-09-05 14:41:43 +03:00
Dobromir Popov
34b095d6ff refactoring 2025-09-05 01:15:58 +03:00
Dobromir Popov
48060c360f pool uses proxy 2025-09-05 00:57:22 +03:00
Dobromir Popov
fec5f35cce latest fixes 2025-09-05 00:10:46 +03:00
Dobromir Popov
1bcf7109cf real mining 2025-09-04 21:06:37 +03:00
Dobromir Popov
02c23db39f notes 2025-09-02 19:29:37 +03:00
Dobromir Popov
9be9241daa fix hashrates dash 2025-09-02 13:50:08 +03:00
Dobromir Popov
89e13e91d6 dashboard 2025-09-02 13:21:43 +03:00
Dobromir Popov
1e9a56356e wehb dash! 2025-09-02 12:19:49 +03:00
Dobromir Popov
b3ec402a2e reward distributions 2025-09-02 12:16:56 +03:00
Dobromir Popov
3dd0e55fde address validations 2025-09-02 12:04:28 +03:00
Dobromir Popov
0c8e4989e4 pool worker individual addresses 2025-09-02 12:04:16 +03:00
Dobromir Popov
8bdf9bf9eb stratum working, implement mining pool, support "extranonce" 2025-09-02 11:54:15 +03:00
Dobromir Popov
e816c04c8b stratum protocol proxy 2025-09-02 11:24:21 +03:00
Dobromir Popov
8855429502 thread count solo mine 2025-09-02 11:21:34 +03:00
Dobromir Popov
55909111e1 solo address param 2025-09-02 10:55:45 +03:00
Dobromir Popov
18697e5651 rincoin node wip 2025-09-02 09:55:24 +03:00
Dobromir Popov
0b058ff7d5 rin node 2025-09-02 03:12:05 +03:00
Dobromir Popov
21cd39068a rin mining working 2025-09-02 02:47:05 +03:00
Dobromir Popov
b41889a670 move 2025-09-02 02:16:03 +03:00
Dobromir Popov
0690149510 more scripts 2025-09-02 02:15:10 +03:00
Dobromir Popov
d63591bde9 xmrocean 2025-09-01 14:49:15 +03:00
Dobromir Popov
b902966cb6 best 2025-09-01 14:46:26 +03:00
Dobromir Popov
41434ddc3a mining, cleanup 2025-09-01 14:34:36 +03:00
Dobromir Popov
d6ce6e0870 toolboxes 2025-09-01 14:33:58 +03:00
Dobromir Popov
2bb1c17453 working windows on dockurr 2025-09-01 13:25:05 +03:00
Dobromir Popov
61f52d2c65 windows.yml builds 2025-09-01 00:43:42 +03:00
Dobromir Popov
dae3c933e2 Add dockur-windows files 2025-08-31 19:31:29 +03:00
Dobromir Popov
6d8f9eb67e notes; new windows yml 2025-08-31 19:28:24 +03:00
Dobromir Popov
8746bed388 add linux resx for AMD AI 2025-08-28 22:09:21 +03:00
Dobromir Popov
6a736965be Merge branch 'master' of http://git.d-popov.com/popov/scripts 2025-08-27 22:26:16 +03:00
Dobromir Popov
5dcce058e0 added windows to desktop envs 2025-08-27 22:26:14 +03:00
45 changed files with 5276 additions and 1 deletions

4
.gitignore vendored
View File

@@ -1,2 +1,6 @@
.aider*
AI/MCP/*
.fuse_hidde*
*.pyc
MINE/rin/mining_log.txt
*mining_log.txt

2
AI/cursor.md Normal file
View File

@@ -0,0 +1,2 @@
rules not valid on linux:
We are on Windows 11 host with PS as default shell. We have full administrative control. Do not try to use linux commands like 'tail' or other unix specific.

25
MINE/cmds Normal file
View File

@@ -0,0 +1,25 @@
# RANDOMX:
root@db-NucBox-EVO-X2:/home/db/Downloads/SRBMiner-Multi-2-9-6# ./SRBMiner-MULTI --algorithm randomx --pool xmr-us-east1.nanopool.org:14444 --wallet bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j --disable-gpu --cpu-threads 28
cd /home/db/Downloads/SRBMiner-Multi-2-9-6
./SRBMiner-MULTI --algorithm randomx --pool randomx.mine.zergpool.com:4453 --wallet bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j -p c=BTC,mc=ZEPH --cpu-threads 28
./SRBMiner-MULTI --algorithm randomx --pool randomx.mine.zergpool.com:4453 --wallet bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j -p c=BTC,mc=ZEPH --cpu-threads 28 --gpu-id 1
# pool/solo/party -p [m=solo][m=party]
# experim
export LD_LIBRARY_PATH=/opt/rocm-6.4.3/lib:$LD_LIBRARY_PATH
./SRBMiner-MULTI --algorithm ethash --pool ethash.mine.zergpool.com:9999 --wallet bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j --disable-cpu
-o stratum+tcp://ethash.mine.zergpool.com:9999 -u bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j -p c=BTC
./SRBMiner-MULTI --algorithm blake2s --pool auto.nicehash.com:3333 --wallet bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j --disable-cpu
# RIN
/mnt/shared/DEV/repos/d-popov.com/mines/rin/miner/cpuminer-opt-rin/cpuminer -a rinhash -o stratum+tcp://192.168.0.188:3334 -u x -p x -t 32
cd /mnt/shared/DEV/repos/d-popov.com/mines/rin/miner/cpuminer-opt-rin
cpuminer -a rinhash -o stratum+tcp://rinhash.mine.zergpool.com:7148 -u bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j -p c=BTC,mc=RIN,ID=StrixHalo -t 32
/mnt/shared/DEV/repos/d-popov.com/mines/rin/miner/cpuminer-opt-rin/cpuminer -o stratum+tcp://randomx.eu.zergpool.com:4453 -a randomx -u bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j -p c=BTC,mc=ZEPH -t 32

245
MINE/gBench.sh Normal file
View File

@@ -0,0 +1,245 @@
#!/bin/bash
# GMiner algorithm test script - save as test_gminer.sh
# Default docker image (can be overridden with parameter)
DOCKER_IMAGE=${1:-"amdopencl"}
# amd-strix-halo-llama-rocm
#amd-strix-halo-llama-vulkan-radv
# amd-strix-halo-llama-vulkan-amdvlk
BTC_WALLET="bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j"
# Algorithm to coin mapping
declare -A ALGO_COINS=(
["ethash"]="ETH"
["etchash"]="ETC"
["autolykos2"]="ERG"
["ergo"]="ERG"
["equihash125_4"]="ZEL"
["equihash144_5"]="ZEL"
["equihash192_7"]="ZEL"
["equihash210_9"]="ZEL"
["beamhash"]="BEAM"
["cuckaroo29"]="GRIN"
["cuckatoo32"]="GRIN"
["flux"]="FLUX"
["octopus"]="CFX"
)
# Function to fetch current cryptocurrency prices
fetch_prices() {
echo "Fetching current cryptocurrency prices..."
# Use CoinGecko API to get current prices
local api_url="https://api.coingecko.com/api/v3/simple/price?ids=ethereum,ethereum-classic,ergo,zelcash,beam,grin,flux,conflux&vs_currencies=usd"
# Try to fetch prices with timeout and fallback
local prices_json=$(timeout 10s curl -s "$api_url" 2>/dev/null)
if [[ -z "$prices_json" ]]; then
echo "Warning: Could not fetch current prices, using fallback values"
PRICES_ETH=3500.00
PRICES_ETC=35.00
PRICES_ERG=2.50
PRICES_ZEL=0.15
PRICES_BEAM=0.05
PRICES_GRIN=0.05
PRICES_FLUX=0.80
PRICES_CFX=0.20
return
fi
# Parse JSON response and extract prices using jq if available, otherwise use grep
if command -v jq &> /dev/null; then
PRICES_ETH=$(echo "$prices_json" | jq -r '.ethereum.usd // "3500.00"')
PRICES_ETC=$(echo "$prices_json" | jq -r '.ethereum-classic.usd // "35.00"')
PRICES_ERG=$(echo "$prices_json" | jq -r '.ergo.usd // "2.50"')
PRICES_ZEL=$(echo "$prices_json" | jq -r '.zelcash.usd // "0.15"')
PRICES_BEAM=$(echo "$prices_json" | jq -r '.beam.usd // "0.05"')
PRICES_GRIN=$(echo "$prices_json" | jq -r '.grin.usd // "0.05"')
PRICES_FLUX=$(echo "$prices_json" | jq -r '.flux.usd // "0.80"')
PRICES_CFX=$(echo "$prices_json" | jq -r '.conflux.usd // "0.20"')
else
# Fallback to grep parsing
PRICES_ETH=$(echo "$prices_json" | grep -o '"ethereum":{"usd":[0-9]*\.[0-9]*' | grep -o '[0-9]*\.[0-9]*$' || echo "3500.00")
PRICES_ETC=$(echo "$prices_json" | grep -o '"ethereum-classic":{"usd":[0-9]*\.[0-9]*' | grep -o '[0-9]*\.[0-9]*$' || echo "35.00")
PRICES_ERG=$(echo "$prices_json" | grep -o '"ergo":{"usd":[0-9]*\.[0-9]*' | grep -o '[0-9]*\.[0-9]*$' || echo "2.50")
PRICES_ZEL=$(echo "$prices_json" | grep -o '"zelcash":{"usd":[0-9]*\.[0-9]*' | grep -o '[0-9]*\.[0-9]*$' || echo "0.15")
PRICES_BEAM=$(echo "$prices_json" | grep -o '"beam":{"usd":[0-9]*\.[0-9]*' | grep -o '[0-9]*\.[0-9]*$' || echo "0.05")
PRICES_GRIN=$(echo "$prices_json" | grep -o '"grin":{"usd":[0-9]*\.[0-9]*' | grep -o '[0-9]*\.[0-9]*$' || echo "0.05")
PRICES_FLUX=$(echo "$prices_json" | grep -o '"flux":{"usd":[0-9]*\.[0-9]*' | grep -o '[0-9]*\.[0-9]*$' || echo "0.80")
PRICES_CFX=$(echo "$prices_json" | grep -o '"conflux":{"usd":[0-9]*\.[0-9]*' | grep -o '[0-9]*\.[0-9]*$' || echo "0.20")
fi
echo "Current prices fetched successfully:"
echo " ETH: $PRICES_ETH"
echo " ETC: $PRICES_ETC"
echo " ERG: $PRICES_ERG"
echo " ZEL: $PRICES_ZEL"
echo " BEAM: $PRICES_BEAM"
echo " GRIN: $PRICES_GRIN"
echo " FLUX: $PRICES_FLUX"
echo " CFX: $PRICES_CFX"
echo ""
}
# GMiner supported algorithms
ALGOS=(
"ethash"
"etchash"
"autolykos2"
"equihash125_4"
"equihash144_5"
"equihash192_7"
"equihash210_9"
"beamhash"
"cuckaroo29"
"cuckatoo32"
"flux"
"octopus"
"ergo"
)
echo "=== GMiner Algorithm Tests ==="
echo "Using BTC wallet: $BTC_WALLET"
echo "Using Docker image: $DOCKER_IMAGE"
echo "Testing each algorithm for 30 seconds..."
echo "======================================="
# Fetch current prices at startup
fetch_prices
# Function to calculate USD value
calculate_usd_reward() {
local algo=$1
local hashrate=$2
local coin=${ALGO_COINS[$algo]}
# Get price based on coin
local price=0
case $coin in
"ETH") price=$PRICES_ETH ;;
"ETC") price=$PRICES_ETC ;;
"ERG") price=$PRICES_ERG ;;
"ZEL") price=$PRICES_ZEL ;;
"BEAM") price=$PRICES_BEAM ;;
"GRIN") price=$PRICES_GRIN ;;
"FLUX") price=$PRICES_FLUX ;;
"CFX") price=$PRICES_CFX ;;
*) echo "Unknown" && return ;;
esac
if [[ -z "$price" || "$price" == "0" ]]; then
echo "Unknown"
return
fi
# Rough calculation: hashrate * price * 0.000001 (simplified)
local usd_value=$(echo "$hashrate * $price * 0.000001" | bc -l 2>/dev/null)
printf "%.2f" $usd_value 2>/dev/null || echo "Unknown"
}
# Function to extract hashrate from miner output
extract_hashrate() {
local output="$1"
# Look for common hashrate patterns in GMiner output
local hashrate=$(echo "$output" | grep -oE "[0-9]+\.[0-9]+ [KMGT]?H/s" | tail -1 | grep -oE "[0-9]+\.[0-9]+" | tail -1)
# If no decimal found, look for integer hashrates
if [[ -z "$hashrate" ]]; then
hashrate=$(echo "$output" | grep -oE "[0-9]+ [KMGT]?H/s" | tail -1 | grep -oE "[0-9]+" | tail -1)
fi
# If still no hashrate found, look for any number followed by H/s
if [[ -z "$hashrate" ]]; then
hashrate=$(echo "$output" | grep -oE "[0-9]+[\.]?[0-9]* H/s" | tail -1 | grep -oE "[0-9]+[\.]?[0-9]*" | tail -1)
fi
# If still no hashrate, look for "Speed:" patterns
if [[ -z "$hashrate" ]]; then
hashrate=$(echo "$output" | grep -i "speed:" | tail -1 | grep -oE "[0-9]+[\.]?[0-9]*" | tail -1)
fi
# If still no hashrate, look for any number followed by H/s (case insensitive)
if [[ -z "$hashrate" ]]; then
hashrate=$(echo "$output" | grep -ioE "[0-9]+[\.]?[0-9]* h/s" | tail -1 | grep -oE "[0-9]+[\.]?[0-9]*" | tail -1)
fi
echo "$hashrate"
}
for algo in "${ALGOS[@]}"; do
echo ""
echo "Testing: $algo"
echo "------------------------"
case $algo in
"ethash")
output=$(sudo docker exec -it $DOCKER_IMAGE timeout 35s bash -c "/mnt/dl/gminer/miner --algo $algo --server eth.2miners.com:2020 --user '$BTC_WALLET' --pass x" 2>&1)
;;
"etchash")
output=$(sudo docker exec -it $DOCKER_IMAGE timeout 35s bash -c "/mnt/dl/gminer/miner --algo $algo --server etc.2miners.com:1010 --user '$BTC_WALLET' --pass x" 2>&1)
;;
"autolykos2"|"ergo")
output=$(sudo docker exec -it $DOCKER_IMAGE timeout 35s bash -c "/mnt/dl/gminer/miner --algo autolykos2 --server ergo.2miners.com:8888 --user '$BTC_WALLET' --pass x" 2>&1)
;;
"equihash125_4")
output=$(sudo docker exec -it $DOCKER_IMAGE timeout 35s bash -c "/mnt/dl/gminer/miner --algo $algo --server zel.2miners.com:9090 --user '$BTC_WALLET' --pass x" 2>&1)
;;
"equihash144_5")
output=$(sudo docker exec -it $DOCKER_IMAGE timeout 35s bash -c "/mnt/dl/gminer/miner --algo $algo --server zel.2miners.com:9090 --user '$BTC_WALLET' --pass x" 2>&1)
;;
"equihash192_7")
output=$(sudo docker exec -it $DOCKER_IMAGE timeout 35s bash -c "/mnt/dl/gminer/miner --algo $algo --server zel.2miners.com:9090 --user '$BTC_WALLET' --pass x" 2>&1)
;;
"equihash210_9")
output=$(sudo docker exec -it $DOCKER_IMAGE timeout 35s bash -c "/mnt/dl/gminer/miner --algo $algo --server zel.2miners.com:9090 --user '$BTC_WALLET' --pass x" 2>&1)
;;
"beamhash")
output=$(sudo docker exec -it $DOCKER_IMAGE timeout 35s bash -c "/mnt/dl/gminer/miner --algo $algo --server beam.2miners.com:5252 --user '$BTC_WALLET' --pass x" 2>&1)
;;
"cuckaroo29")
output=$(sudo docker exec -it $DOCKER_IMAGE timeout 35s bash -c "/mnt/dl/gminer/miner --algo $algo --server grin.2miners.com:3030 --user '$BTC_WALLET' --pass x" 2>&1)
;;
"cuckatoo32")
output=$(sudo docker exec -it $DOCKER_IMAGE timeout 35s bash -c "/mnt/dl/gminer/miner --algo $algo --server grin.2miners.com:3030 --user '$BTC_WALLET' --pass x" 2>&1)
;;
"flux")
output=$(sudo docker exec -it $DOCKER_IMAGE timeout 35s bash -c "/mnt/dl/gminer/miner --algo $algo --server flux.2miners.com:2020 --user '$BTC_WALLET' --pass x" 2>&1)
;;
"octopus")
output=$(sudo docker exec -it $DOCKER_IMAGE timeout 35s bash -c "/mnt/dl/gminer/miner --algo $algo --server cfx.2miners.com:3254 --user '$BTC_WALLET' --pass x" 2>&1)
;;
*)
echo "No specific pool configured for $algo - skipping"
continue
;;
esac
exit_code=$?
# Extract hashrate and calculate USD value
hashrate=$(extract_hashrate "$output")
coin=${ALGO_COINS[$algo]}
usd_value=$(calculate_usd_reward "$algo" "$hashrate")
if [ $exit_code -eq 0 ]; then
echo "SUCCESS: $algo - Hashrate: ${hashrate}H/s - Coin: $coin - Est. USD/day: $usd_value"
elif [ $exit_code -eq 124 ]; then
echo "TIMEOUT: $algo - Hashrate: ${hashrate}H/s - Coin: $coin - Est. USD/day: $usd_value (likely working)"
else
echo "FAILED: $algo - Error code: $exit_code"
# Debug: show first few lines of output for failed attempts
echo "Debug output (first 5 lines):"
echo "$output" | head -5
fi
sleep 3
done
echo ""
echo "=== GMiner Tests Complete ==="
echo "Usage: $0 [docker_image_name]"
echo "Default: amdopencl"
echo "Example for RockM: $0 rockm"

29
MINE/lolBench.sh Normal file
View File

@@ -0,0 +1,29 @@
#!/bin/bash
# lolMiner benchmark script - save as bench_lolminer.sh
ALGOS=("ETHASH" "ETCHASH" "AUTOLYKOS2" "BEAM-III" "EQUIHASH144_5" "EQUIHASH192_7" "EQUIHASH210_9" "FLUX" "NEXA" "PROGPOW" "PROGPOWZ" "PROGPOW_VERIBLOCK" "PROGPOW_VEIL" "TON")
echo "=== lolMiner Algorithm Benchmark ==="
echo "Testing each algorithm for 15 seconds..."
echo "====================================="
for algo in "${ALGOS[@]}"; do
echo ""
echo "Testing: $algo"
echo "------------------------"
sudo docker exec -it amdopencl timeout 20s bash -c "mnt/dl/lol.1.97/lolMiner --algo $algo --benchmark --benchepochs 1 --benchwarmup 5" 2>/dev/null
if [ $? -eq 0 ]; then
echo "$algo: WORKS"
elif [ $? -eq 124 ]; then
echo "⏱️ $algo: TIMEOUT (likely working)"
else
echo "$algo: FAILED"
fi
sleep 2
done
echo ""
echo "=== Benchmark Complete ==="

144
MINE/notes.md Normal file
View File

@@ -0,0 +1,144 @@
bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j
<!-- MONERO -->
cd ~/Downloads
wget https://github.com/xmrig/xmrig/releases/download/v6.21.0/xmrig-6.21.0-linux-x64.tar.gz
tar -xzf xmrig-6.21.0-linux-x64.tar.gz
cd xmrig-6.21.0
~/Downloads/xmrig-6.21.0$ ./xmrig -o pool.supportxmr.com:443 -u bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j -p NUC --tls -t 28 --donate-level=0
/mnt/xmrig/xmrig -o pool.supportxmr.com:443 -u bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j -p NUC --tls -t 24 --opencl --opencl-devices=0 --donate-level=0
<!-- run inside DOCKER directly. amd-strix-halo-llama-rocm is best for now -->
sudo docker exec -it amd-strix-halo-llama-rocm bash -c "/mnt/dl/xmrig-6.21.0/xmrig -o pool.supportxmr.com:443 -u bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j -p NUC --tls -t 28 --donate-level=0"
# proper address XMR
sudo docker exec -it amdopencl bash -c "cd /tmp/xmrig-6.21.0 && ./xmrig -o pool.supportxmr.com:443 -u 47tJRLX5UgK59VmRsN1L7AgcQNZYNBhJ5Lv7Jt4KEViS8WEbCf4hPGcM78rRLcS9xmgbbJdwHzbvjA1mJiKixtX3Q8iiBgu -p NUC --tls -t 32 --donate-level=0"
|| ------------------------------ CURRENT BEST ------------------------------ ||
# MoneroOcean with XMR payout
nice /home/db/Downloads/moneroocean/xmrig
sudo docker exec -it amd-strix-halo-llama-rocm bash -c "/mnt/dl/xmrig-6.21.0/xmrig -o gulf.moneroocean.stream:10001 -u 47tJRLX5UgK59VmRsN1L7AgcQNZYNBhJ5Lv7Jt4KEViS8WEbCf4hPGcM78rRLcS9xmgbbJdwHzbvjA1mJiKixtX3Q8iiBgu -p x -t 28 --donate-level=0"
curl -s "http://api.moneroocean.stream/miner/47tJRLX5UgK59VmRsN1L7AgcQNZYNBhJ5Lv7Jt4KEViS8WEbCf4hPGcM78rRLcS9xmgbbJdwHzbvjA1mJiKixtX3Q8iiBgu/stats"
<!-- BTC PAYOUTS -->
# Check Unmineable balance
curl -s "https://api.unmineable.com/v4/address/bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j?coin=BTC"
# Mine RandomX but get paid in BTC
sudo docker exec -it amd-strix-halo-llama-rocm bash -c "/mnt/dl/xmrig-6.21.0/xmrig -o rx.unmineable.com:3333 -u BTC:bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j -p x -t 28 --donate-level=0"
<!-- SRBMiner -->
SRBMiner-MULTI --algorithm progpow_zano --pool zano.herominers.com:1112 --wallet bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j --worker StrixHalo
<!-- lol, all containers -->
sudo docker exec -it amd-strix-halo-llama-rocm bash -c "/mnt/dl/lol.1.97/lolMiner --algo FISHHASH --pool ironfish.unmineable.com:3333 --user BTC:bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j --worker StrixHalo"
sudo docker exec -it amd-strix-halo-llama-vulkan-radv bash -c "/mnt/dl/lol.1.97/lolMiner --algo FISHHASH --pool ironfish.unmineable.com:3333 --user BTC:bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j --worker StrixHalo"
sudo docker exec -it amd-strix-halo-llama-vulkan-amdvlk bash -c "/mnt/dl/lol.1.97/lolMiner --algo FISHHASH --pool ironfish.unmineable.com:3333 --user BTC:bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j --worker StrixHalo"
sudo docker exec -it amdopencl bash -c "/mnt/dl/lol.1.97/lolMiner --algo FISHHASH --pool ironfish.unmineable.com:3333 --user BTC:bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j --worker StrixHalo"
# Start with IronFish (most profitable supported)
./lolMiner --algo FISHHASH --pool ironfish.unmineable.com:3333 --user BTC:bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j --worker StrixHalo
# Or safer option with Ergo
./lolMiner --algo AUTOLYKOS2 --pool ergo.unmineable.com:3333 --user BTC:bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j --worker StrixHalo
# rinhash
sudo docker exec -it amd-strix-halo-llama-rocm bash -c "/mnt/dl/rinhash/cpuminer-opt-rin/cpuminer --algo rinhash --benchmark -t 4"
[sudo docker exec -it amd-strix-halo-llama-rocm bash -c "/mnt/dl/rinhash/cpuminer-opt-rin/cpuminer --algo rinhash --benchmark -t 4"
rinhash.mine.zergpool.com:7148 c=RIN]
<!--------------------------BEST CURRENT CPU -------------------------- Active exchange: Listed on Exbitron (RIN/USDT) -->
sudo docker exec -it amd-strix-halo-llama-rocm bash -c "/mnt/dl/rinhash/cpuminer-opt-rin/cpuminer -a rinhash -o stratum+tcp://rinhash.mine.zergpool.com:7148 -u bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j -p c=BTC,mc=RIN,ID=StrixHalo -t 32"
-----------------------------------------------------------------
SOLO:
/home/db/Downloads/rinhash/cpuminer-opt-rin/cpuminer -a rinhash -o stratum+tcp://192.168.0.188:3333 -u username.workername -p x -t 28
./cpuminer --algo rinhash --url [pool_url] --user [your_rin_wallet] --pass x --threads 32
# GPU CURRENT:
/mnt/shared/DEV/repos/d-popov.com/scripts$ sudo docker exec -it amdopencl bash -c "/mnt/dl/gminer/miner --algo equihash125_4 --server equihash125.mine.zergpool.com:2142 --user bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j
https://zergpool.com/wallet/bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j
https://zergpool.com/api/wallet?address=bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j
https://zergpool.com/api/walletEx?address=<YOUR_WALLET_ADDRESS>
/mnt/shared/DEV/repos/d-popov.com/scripts$ sudo docker exec -it amdopencl bash -c "/mnt/dl/sgminer-gm/ --algo equihash125_4 --server equihash125.mine.zergpool.com:2142 --user bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j
/home/db/Downloads/
sudo docker exec -it amdopencl timeout 35s bash -c "/mnt/dl/gminer/miner --algo equihash125_4 --server equihash.mine.zergpool.com:2142 c=BTC --user 'bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j' --pass 'x'"
-------------------------
To run this as a service (keep mining in background):
Stop current mining (Ctrl+C)
Create a systemd service:
bash
Copy
sudo nano /etc/systemd/system/xmrig.service
Add this content:
ini
Copy
[Unit]
Description=XMRig Monero Miner
After=network.target
[Service]
Type=simple
User=db
WorkingDirectory=/home/db/Downloads/xmrig-6.21.0
ExecStart=/home/db/Downloads/xmrig-6.21.0/xmrig -o pool.supportxmr.com:443 -u bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j -p NUC --tls
Restart=always
RestartSec=10
[Install]
WantedBy=multi-user.target
Enable and start the service:
bash
Copy
sudo systemctl enable xmrig
sudo systemctl start xmrig
sudo systemctl status xmrig
COPY --from=qemux/qemu:7.12 / /
# 7.12
# 7.11
# 7.10
# 7.09
# 7.08
# 7.07
# 7.06
# 7.05
# 7.04
# 7.03
# 7.02
# 7.01
# 7.00
# 6.22
# 6.21
# 6.20
# 6.19
# 6.18

25
MINE/rin/Dockerfile Normal file
View File

@@ -0,0 +1,25 @@
FROM ubuntu:22.04
ENV DEBIAN_FRONTEND=noninteractive
RUN apt-get update && apt-get install -y \
build-essential libtool autotools-dev automake pkg-config bsdmainutils \
libevent-dev libboost-all-dev libssl-dev \
libdb5.3-dev libdb5.3++-dev libfmt-dev libsqlite3-dev \
git ca-certificates \
&& rm -rf /var/lib/apt/lists/*
WORKDIR /opt
RUN git clone https://github.com/Rin-coin/rincoin.git && \
cd rincoin && \
./autogen.sh && \
./configure --with-incompatible-bdb && \
make -j$(nproc) && \
make install
# runtime
RUN useradd -m rin && mkdir -p /data && chown -R rin:rin /data
USER rin
VOLUME ["/data"]
EXPOSE 9555 9556
ENTRYPOINT ["/usr/local/bin/rincoind"]
CMD ["-datadir=/data", "-conf=/data/rincoin.conf", "-printtoconsole"]

362
MINE/rin/README.md Normal file
View File

@@ -0,0 +1,362 @@
# RinCoin Mining Setup Complete! 🎉
## 🎯 **Choose Your Mining Strategy:**
### **Option 1: Solo Mining (Single Miner, All Rewards to You)**
```bash
# Start solo mining proxy
cd /mnt/shared/DEV/repos/d-popov.com/scripts
./MINE/rin/start_stratum_proxy.sh
# Run your miner
./cpuminer -a rinhash -o stratum+tcp://127.0.0.1:3333 -u user -p pass -t 28
```
**Result**: 100% of block rewards go to your wallet
### **Option 2: Mining Pool (Multiple Miners, Distributed Rewards)**
```bash
# Start mining pool
./MINE/rin/start_mining_pool.sh
# Miners connect with their RinCoin addresses:
# Option 1: Address as username
./cpuminer -a rinhash -o stratum+tcp://YOUR_IP:3333 -u rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q -p x
# Option 2: Address.workername format
./cpuminer -a rinhash -o stratum+tcp://YOUR_IP:3333 -u rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q.worker1 -p x
# Option 3: Traditional username (rewards to pool address)
./cpuminer -a rinhash -o stratum+tcp://192.168.0.188:3333 -u username.workername -p x -t 24
```
**Result**: Block rewards distributed among all miners based on shares
### **Key Differences:**
| Feature | Solo Mining (`stratum_proxy`) | Mining Pool (`stratum_pool`) |
|---------|--------------------------------|------------------------------|
| **Rewards** | 100% to you | Distributed among miners |
| **Miners** | Single | Multiple |
| **Setup** | Simple | More complex |
| **Consistency** | Rare big payouts | Regular small payments |
| **Risk** | High variance | Lower variance |
| **Public** | No | Yes, can be published |
---
## ✅ **Successfully Built and Running:**
### **1. RinCoin Node Container**
- **Container**: `rincoin-node` (ID: 87b5f74a2472)
- **Status**: ✅ **RUNNING**
- **Ports**: 9555 (P2P), 9556 (RPC)
- **Version**: v1.0.1.0-5cf3d4a11
- **Sync Status**: ✅ **FULLY SYNCED** (blocks: 228,082, headers: 228,082)
### **2. Wallet Setup**
- **Wallet Name**: `main`
- **Default RinCoin Address**: `rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q`
- **RPC Credentials**:
- User: `rinrpc`
- Password: `745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90`
### **3. Configuration Files**
- **Config**: `/mnt/data/docker_vol/rincoin/rincoin-node/rincoin.conf`
- **Data Directory**: `/mnt/data/docker_vol/rincoin/rincoin-node/data`
- **Docker Compose**: `MINE/rin/container.yml`
## 🚀 **Ready for Mining:**
### **Pool Mining (Zergpool) - Recommended for Consistent Rewards**
```bash
# CPU Mining RinHash to BTC
sudo docker exec -it amd-strix-halo-llama-rocm bash -c "/mnt/dl/rinhash/cpuminer-opt-rin/cpuminer -a rinhash -o stratum+tcp://rinhash.mine.zergpool.com:7148 -u bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j -p c=BTC,mc=RIN,ID=StrixHalo -t 32"
```
### **Solo Mining (Local Node) - With Stratum Proxy ⭐ RECOMMENDED**
```bash
# Start mining with your RinCoin address (rewards go to this address!)
./MINE/rin/start_mining_with_address.sh rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q 28
# Or use the default address
./MINE/rin/start_mining_with_address.sh
```
### **Manual Solo Mining Setup (Stratum Proxy)**
```bash
# 1. Start Stratum proxy (solo mining)
./MINE/rin/start_stratum_proxy.sh
# 2. In another terminal, connect cpuminer-opt-rin
sudo docker exec -it amd-strix-halo-llama-rocm bash -c "/mnt/dl/rinhash/cpuminer-opt-rin/cpuminer -a rinhash -o stratum+tcp://172.17.0.1:3333 -u user -p pass -t 28"
```
### **Built-in Core Mining (Low Performance)**
```bash
# Solo mining with built-in RinCoin core (not recommended)
bash MINE/rin/solo_mining_core.sh -t 28
```
### **Why cpuminer-opt-rin Can't Mine Directly to Node**
```bash
# This command will fail:
sudo docker exec -it amd-strix-halo-llama-rocm bash -c "/mnt/dl/rinhash/cpuminer-opt-rin/cpuminer -a rinhash -o http://127.0.0.1:9556 -u rinrpc -p 745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90 -t 28 --coinbase-addr=bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j"
# Reason: Protocol mismatch
# - cpuminer-opt-rin uses Stratum protocol (for mining pools)
# - RinCoin node uses RPC protocol (for direct mining)
# - No built-in protocol conversion available
```
### **Direct CPU Mining Setup (Solo Mining - No Container Needed)**
```bash
# 1. Start stratum proxy (solo mining)
./MINE/rin/start_stratum_proxy.sh
# OR run in background with logging
nohup python3 MINE/rin/stratum_proxy.py > stratum_proxy.log 2>&1 &
# 2. Run cpuminer directly on host
/home/db/Downloads/rinhash/cpuminer-opt-rin/cpuminer -a rinhash -o stratum+tcp://127.0.0.1:3333 -u user -p pass -t 28
# 3. Clean up when done
./MINE/rin/kill_stratum_proxy.sh
```
### **Mining Options Explained**
1. **Built-in Core Mining**: Uses RinCoin's `generatetoaddress` RPC command (low performance)
2. **Pool Mining**: Uses cpuminer-opt-rin with Stratum pools (Zergpool) - consistent rewards
3. **Direct RPC Mining**: Would require custom miner implementing `getblocktemplate`
4. **Solo Mining (Stratum Proxy)**: Uses Stratum proxy to bridge cpuminer-opt-rin to RinCoin node - all rewards to you
5. **Mining Pool (Stratum Pool)**: Distributes block rewards among multiple miners - share-based rewards
## 🏊‍♂️ **Mining Pool Setup (Multiple Miners)**
Your Stratum proxy can be enhanced to work as a **full mining pool** that distributes block rewards among multiple miners!
### **Pool Features:**
-**Multiple Miner Support**: Unlimited miners can connect
-**Share-Based Rewards**: Rewards distributed based on share contributions
-**Pool Fee**: 1% fee for pool maintenance
-**Real-Time Statistics**: Web dashboard with live stats
-**Block Reward Distribution**: Automatic distribution when blocks are found
### **Quick Start Pool:**
```bash
# 1. Start the mining pool
./MINE/rin/start_mining_pool.sh
# 2. Miners connect with:
./cpuminer -a rinhash -o stratum+tcp://YOUR_IP:3333 -u username.workername -p x
# submit every share
cd /mnt/shared/DEV/repos/d-popov.com/scripts/MINE/rin && python3 stratum_proxy.py --submit-all-blocks
python3 stratum_proxy.py --submit-threshold 0.05
# For production (10% threshold - good balance)
python3 stratum_proxy.py --submit-threshold 0.1
# For aggressive testing (1% threshold)
python3 stratum_proxy.py --submit-threshold 0.01
# For normal operation (only valid blocks)
python3 stratum_proxy.py
```
### **Pool vs Solo Mining:**
| Feature | Solo Mining | Mining Pool |
|---------|-------------|-------------|
| **Block Rewards** | 100% to you | Distributed among miners |
| **Consistency** | Rare blocks | Regular small payments |
| **Setup** | Simple | More complex |
| **Miners** | Single | Multiple |
| **Risk** | High variance | Lower variance |
### **Publishing Your Pool:**
#### **1. Public IP Setup:**
```bash
# Get your public IP
curl ifconfig.me
# Configure firewall (if needed)
sudo ufw allow 3333/tcp
sudo ufw allow 8080/tcp # Web interface
```
#### **2. Pool Connection String:**
```
stratum+tcp://YOUR_PUBLIC_IP:3333
```
#### **3. Web Dashboard:**
- **URL**: `http://YOUR_PUBLIC_IP:8080`
- **Features**: Real-time stats, miner rankings, block history
#### **4. Pool Announcement:**
Share your pool details:
- **Algorithm**: RinHash
- **Port**: 3333
- **Fee**: 1%
- **Payout**: Automatic distribution
- **Web**: `http://YOUR_PUBLIC_IP:8080`
### **Pool Configuration:**
```python
# Edit MINE/rin/stratum_pool.py
pool_address = 'rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q' # Pool wallet
pool_fee_percent = 1.0 # Pool fee percentage
```
## build image
sudo bash -lc "cd /mnt/shared/DEV/repos/d-popov.com/scripts/MINE/rin && docker build -t rincoin-node:latest . | cat"
## start container
sudo docker run -d --name rincoin-node \
-p 9555:9555 -p 9556:9556 \
-v /mnt/data/docker_vol/rincoin/rincoin-node/data:/data \
-v /mnt/data/docker_vol/rincoin/rincoin-node/rincoin.conf:/data/rincoin.conf:ro \
rincoin-node:latest -datadir=/data -conf=/data/rincoin.conf -printtoconsole
## check if running
curl --user rinrpc:745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90 -H 'content-type: text/plain' --data '{"jsonrpc":"1.0","id":"curl","method":"getblockchaininfo","params":[]}' http://127.0.0.1:9556/
## get wallet
sudo docker exec rincoin-node rincoin-cli -datadir=/data -conf=/data/rincoin.conf createwallet "main"
sudo docker exec rincoin-node rincoin-cli -datadir=/data -conf=/data/rincoin.conf -rpcwallet=main getnewaddress
rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q
```bash
# Solo mining to your RinCoin wallet
./MINE/rin/solo_mining.sh
```
## 📊 **Performance Comparison:**
| Mining Type | Algorithm | Hashrate | Target | Status |
|-------------|-----------|----------|---------|---------|
| **Pool Mining** | RinHash | ~80 kH/s | Zergpool | ✅ Working |
| **Solo Mining** | RinHash | Built-in CPU | Local Node | ✅ Working |
| **GPU Mining** | Equihash 125,4 | 28.8 Sol/s | Zergpool | ✅ Working |
## 🔧 **Management Commands:**
### **Node Management**
```bash
# Start node
sudo docker start rincoin-node
# Stop node
sudo docker stop rincoin-node
# View logs
sudo docker logs -f rincoin-node
# Check sync status
sudo docker exec rincoin-node rincoin-cli -datadir=/data -conf=/data/rincoin.conf getblockchaininfo
```
### **Wallet Management**
```bash
# Get new address
sudo docker exec rincoin-node rincoin-cli -datadir=/data -conf=/data/rincoin.conf -rpcwallet=main getnewaddress
# Check balance
sudo docker exec rincoin-node rincoin-cli -datadir=/data -conf=/data/rincoin.conf -rpcwallet=main getbalance
```
### **RPC Access**
```bash
# Test RPC connection
curl --user rinrpc:745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90 \
-H 'content-type: text/plain' \
--data '{"jsonrpc":"1.0","id":"curl","method":"getblockchaininfo","params":[]}' \
http://127.0.0.1:9556/
# Get new address via RPC
curl --user rinrpc:745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90 \
-H 'content-type: text/plain' \
--data '{"jsonrpc":"1.0","id":"curl","method":"getnewaddress","params":[]}' \
http://127.0.0.1:9556/
```
## ⚠️ **Important Notes:**
1. **Node Sync**: ✅ **COMPLETE** - Node is fully synced and ready
2. **Solo Mining**: Very low chance of finding blocks solo. Consider pool mining for consistent rewards.
3. **RPC Access**: ✅ **WORKING** - RPC is accessible on port 9556
4. **Address Parameter**: Solo mining script accepts custom addresses or uses default
5. **Block Rewards**: When solo mining, ALL block rewards go to your specified RinCoin address
## 🛠️ **Troubleshooting:**
### **Port 3333 Already in Use**
```bash
# Check what's using the port
sudo netstat -tlnp | grep :3333
# Kill existing processes
./MINE/rin/kill_stratum_proxy.sh
# Or manually kill
sudo lsof -ti:3333 | xargs sudo kill -9
```
### **Container Can't Connect to Proxy**
```bash
# Use Docker gateway IP instead of localhost
sudo docker exec -it amd-strix-halo-llama-rocm bash -c "/mnt/dl/rinhash/cpuminer-opt-rin/cpuminer -a rinhash -o stratum+tcp://172.17.0.1:3333 -u user -p pass -t 28"
```
### **Check Proxy Logs**
```bash
# View real-time logs
tail -f stratum_proxy.log
# Check if proxy is running
ps aux | grep stratum_proxy
```
## 🔐 **Wallet Backup & Restoration**
### **Backup Your Wallet**
```bash
# Run the backup script
./MINE/rin/dump_wallet.sh
```
- This creates a text file with all private keys in `~/rin_wallet_backups/`.
- **Security**: Encrypt immediately with `gpg -c <file>`, store offline, delete unencrypted copy.
- Contains sensitive data—anyone with this can spend your coins.
### **Restore on New Node**
```bash
# Automated restoration (requires new node setup)
./MINE/rin/restore_wallet.sh /path/to/backup.txt
# Manual steps:
# 1. Stop new node: sudo docker stop rincoin-node
# 2. Copy wallet.dat or import dump file
# 3. Start node: sudo docker start rincoin-node
# 4. Load wallet: sudo docker exec rincoin-node rincoin-cli ... loadwallet main
# 5. Import if using dump: importwallet /path/to/dump.txt
```
## 🎯 **Next Steps:**
1.**Node is synced** - Ready for all operations
2. **Choose mining strategy**: Pool mining for consistent income vs Solo mining for block rewards
3. **Monitor performance** and adjust thread count as needed
4. **Set up monitoring** for node health and mining performance
5. **Backup wallet regularly** - Use the dump script above

18
MINE/rin/container.yml Normal file
View File

@@ -0,0 +1,18 @@
version: "3.8"
services:
rincoin-node:
container_name: rincoin-node
image: rincoin-node:latest
restart: unless-stopped
ports:
- "9555:9555"
- "9556:9556"
volumes:
- /mnt/data/docker_vol/rincoin/rincoin-node/data:/data
- /mnt/data/docker_vol/rincoin/rincoin-node/rincoin.conf:/data/rincoin.conf:ro
command:
- rincoind
- -datadir=/data
- -conf=/data/rincoin.conf
- -printtoconsole

65
MINE/rin/dump_wallet.sh Normal file
View File

@@ -0,0 +1,65 @@
#!/bin/bash
# RinCoin Wallet Backup Script
# Dumps all private keys to a secure text file for backup.
# WARNING: This file contains sensitive private keys. Store it securely (encrypted, offline).
# Do not share or email it. Anyone with this file can spend your coins.
set -euo pipefail
# Configuration
CONTAINER="rincoin-node"
WALLET="main"
BACKUP_DIR="${HOME}/rin_wallet_backups"
TIMESTAMP=$(date +%Y%m%d_%H%M%S)
BACKUP_FILE="${BACKUP_DIR}/rin_wallet_backup_${TIMESTAMP}.txt"
# Ensure backup directory exists
mkdir -p "$BACKUP_DIR"
# Check if container is running
if ! sudo docker ps --format '{{.Names}}' | grep -q "^${CONTAINER}$"; then
echo "Error: ${CONTAINER} container is not running. Start it with 'sudo docker start ${CONTAINER}'."
exit 1
fi
# Ensure wallet is loaded
CLI_CMD=(sudo docker exec "$CONTAINER" rincoin-cli -datadir=/data -conf=/data/rincoin.conf -rpcwallet="$WALLET")
if ! "${CLI_CMD[@]//-rpcwallet=$WALLET/}" listwallets | grep -q '"main"'; then
echo "Wallet '${WALLET}' not loaded. Attempting to load..."
"${CLI_CMD[@]//-rpcwallet=$WALLET/}" loadwallet "$WALLET" >/dev/null 2>&1 || {
echo "Failed to load wallet. Ensure it exists."
exit 1
}
fi
echo "Dumping wallet to: $BACKUP_FILE"
echo "This may take a moment..."
# Dump wallet
"${CLI_CMD[@]}" dumpwallet "$BACKUP_FILE"
# Verify the file was created and has content
if [[ ! -f "$BACKUP_FILE" ]]; then
echo "Error: Backup file was not created."
exit 1
fi
LINE_COUNT=$(wc -l < "$BACKUP_FILE")
if [[ $LINE_COUNT -lt 10 ]]; then
echo "Warning: Backup file seems too small (${LINE_COUNT} lines). Check for errors."
exit 1
fi
echo "✅ Wallet successfully backed up to: $BACKUP_FILE"
echo ""
echo "🔐 SECURITY REMINDERS:"
echo " - This file contains private keys for ALL addresses in the wallet."
echo " - Encrypt it immediately: gpg -c $BACKUP_FILE"
echo " - Store on encrypted media (e.g., USB drive in safe)."
echo " - Delete the unencrypted file after encryption."
echo " - Test restoration on a testnet node before relying on it."
echo ""
echo "File size: $(du -h "$BACKUP_FILE" | cut -f1)"
echo "Lines: $LINE_COUNT"

View File

@@ -0,0 +1,40 @@
#!/bin/bash
# Kill RinCoin Stratum Proxy processes
echo "=== Killing RinCoin Stratum Proxy ==="
echo ""
# Find and kill Python processes running stratum_proxy.py
PIDS=$(ps aux | grep "stratum_proxy.py" | grep -v grep | awk '{print $2}')
if [ -n "$PIDS" ]; then
echo "Found Stratum Proxy processes: $PIDS"
echo "Killing processes..."
for pid in $PIDS; do
kill -9 "$pid" 2>/dev/null && echo "Killed PID: $pid" || echo "Failed to kill PID: $pid"
done
else
echo "No Stratum Proxy processes found"
fi
# Also kill any process using port 3333
echo ""
echo "Checking port 3333..."
PORT_PIDS=$(sudo lsof -ti:3333 2>/dev/null)
if [ -n "$PORT_PIDS" ]; then
echo "Found processes using port 3333: $PORT_PIDS"
echo "Killing processes..."
for pid in $PORT_PIDS; do
sudo kill -9 "$pid" 2>/dev/null && echo "Killed PID: $pid" || echo "Failed to kill PID: $pid"
done
else
echo "No processes using port 3333"
fi
echo ""
echo "✅ Cleanup complete!"
echo ""
echo "Port 3333 status:"
netstat -tln | grep ":3333 " || echo "Port 3333 is free"

10
MINE/rin/mining_log.txt Normal file
View File

@@ -0,0 +1,10 @@
================================================================================
RinCoin Mining Log
================================================================================
Started: 2025-09-23 18:38:03
Target Address: rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q
Stratum: 0.0.0.0:3334
RPC: 127.0.0.1:9556
================================================================================
[2025-09-23 18:38:03] 💰 Wallet Balance: 25.00000000 RIN

View File

@@ -0,0 +1,529 @@
#!/usr/bin/env python3
"""
RinCoin Mining Pool Web Interface
Provides web dashboard for pool statistics and miner management
"""
import json
import sqlite3
import requests
from datetime import datetime, timedelta
from http.server import HTTPServer, BaseHTTPRequestHandler
import threading
import time
from requests.auth import HTTPBasicAuth
class PoolWebInterface:
def __init__(self, pool_db, host='0.0.0.0', port=8080, rpc_host='127.0.0.1', rpc_port=9556,
rpc_user='rinrpc', rpc_password='745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90'):
self.pool_db = pool_db
self.host = host
self.port = port
self.rpc_host = rpc_host
self.rpc_port = rpc_port
self.rpc_user = rpc_user
self.rpc_password = rpc_password
self.chart_time_window = 3600 # 1 hour default, adjustable
def set_chart_time_window(self, seconds):
"""Set the chart time window"""
self.chart_time_window = seconds
def format_hashrate(self, hashrate):
"""Format hashrate in human readable format"""
if hashrate >= 1e12:
return f"{hashrate/1e12:.2f} TH/s"
elif hashrate >= 1e9:
return f"{hashrate/1e9:.2f} GH/s"
elif hashrate >= 1e6:
return f"{hashrate/1e6:.2f} MH/s"
elif hashrate >= 1e3:
return f"{hashrate/1e3:.2f} KH/s"
elif hashrate >= 0.01:
return f"{hashrate:.2f} H/s"
elif hashrate > 0:
return f"{hashrate*1000:.2f} mH/s"
else:
return "0.00 H/s"
def get_pool_balance(self):
"""Get pool wallet balance via RPC"""
try:
url = f"http://{self.rpc_host}:{self.rpc_port}/"
headers = {'content-type': 'text/plain'}
auth = HTTPBasicAuth(self.rpc_user, self.rpc_password)
payload = {
"jsonrpc": "1.0",
"id": "pool_balance",
"method": "getbalance",
"params": []
}
response = requests.post(url, json=payload, headers=headers, auth=auth, timeout=10)
if response.status_code == 200:
result = response.json()
if 'error' in result and result['error'] is not None:
print(f"RPC Error getting balance: {result['error']}")
return 0.0
balance = result.get('result', 0)
return float(balance) / 100000000 # Convert from satoshis to RIN
else:
print(f"HTTP Error getting balance: {response.status_code}")
return 0.0
except Exception as e:
print(f"Error getting pool balance: {e}")
return 0.0
def get_pool_stats(self):
"""Get current pool statistics"""
try:
cursor = self.pool_db.cursor()
# Total miners (ever registered)
cursor.execute('SELECT COUNT(DISTINCT id) FROM miners')
total_miners = cursor.fetchone()[0]
# Active miners (last 5 minutes)
cursor.execute('''
SELECT COUNT(DISTINCT m.id) FROM miners m
JOIN shares s ON m.id = s.miner_id
WHERE s.submitted > datetime('now', '-5 minutes')
''')
active_miners = cursor.fetchone()[0]
# Total shares (last 24 hours)
cursor.execute('''
SELECT COUNT(*) FROM shares
WHERE submitted > datetime('now', '-24 hours')
''')
total_shares_24h = cursor.fetchone()[0]
# Pool hashrate: sum of miners.last_hashrate (instantaneous)
cursor.execute('SELECT COALESCE(SUM(last_hashrate), 0) FROM miners')
hashrate = cursor.fetchone()[0] or 0.0
# Debug stats
cursor.execute('''
SELECT SUM(difficulty), COUNT(*) FROM shares
WHERE submitted > datetime('now', '-5 minutes')
''')
rd = cursor.fetchone()
recent_difficulty = rd[0] if rd and rd[0] else 0
recent_share_count = rd[1] if rd and rd[1] else 0
# Get historical hashrate data for chart
cursor.execute('''
SELECT
strftime('%H:%M', submitted) as time,
COUNT(*) as shares,
SUM(difficulty) as total_difficulty
FROM shares
WHERE submitted > datetime('now', '-{} seconds')
GROUP BY strftime('%Y-%m-%d %H:%M', submitted)
ORDER BY submitted DESC
LIMIT 60
'''.format(self.chart_time_window))
historical_data = cursor.fetchall()
# Calculate individual miner hashrates
cursor.execute('''
SELECT
m.user, m.worker,
COUNT(s.id) as shares,
SUM(s.difficulty) as total_difficulty,
m.last_share
FROM miners m
LEFT JOIN shares s ON m.id = s.miner_id
AND s.submitted > datetime('now', '-5 minutes')
GROUP BY m.id, m.user, m.worker
ORDER BY shares DESC
''')
miner_stats = cursor.fetchall()
# Calculate individual hashrates (use miners.last_hashrate)
miner_hashrates = []
for user, worker, shares, difficulty, last_share in miner_stats:
cursor.execute('SELECT last_hashrate FROM miners WHERE user = ? AND worker = ? LIMIT 1', (user, worker))
row = cursor.fetchone()
miner_hashrate = row[0] if row and row[0] else 0.0
miner_hashrates.append((user, worker, shares, miner_hashrate, last_share))
# Total blocks found
cursor.execute('SELECT COUNT(*) FROM blocks')
total_blocks = cursor.fetchone()[0]
# Recent blocks
cursor.execute('''
SELECT block_hash, height, reward, found_at
FROM blocks
ORDER BY found_at DESC
LIMIT 10
''')
recent_blocks = cursor.fetchall()
# Top miners (last 24 hours) - show all miners, even without shares
cursor.execute('''
SELECT m.user, m.worker,
COALESCE(COUNT(s.id), 0) as shares,
m.last_share,
m.created
FROM miners m
LEFT JOIN shares s ON m.id = s.miner_id
AND s.submitted > datetime('now', '-24 hours')
GROUP BY m.id, m.user, m.worker
ORDER BY shares DESC, m.created DESC
LIMIT 20
''')
top_miners = cursor.fetchall()
# All active miners (for better visibility)
cursor.execute('''
SELECT user, worker, created, last_share
FROM miners
ORDER BY created DESC
LIMIT 10
''')
all_miners = cursor.fetchall()
# Get pool balance
pool_balance = self.get_pool_balance()
return {
'total_miners': total_miners,
'active_miners': active_miners,
'total_shares_24h': total_shares_24h,
'hashrate': hashrate,
'total_blocks': total_blocks,
'recent_blocks': recent_blocks,
'top_miners': top_miners,
'all_miners': all_miners,
'miner_hashrates': miner_hashrates,
'historical_data': historical_data,
'pool_balance': pool_balance,
'debug': {
'recent_difficulty': recent_difficulty,
'recent_share_count': recent_share_count,
'total_shares_24h': total_shares_24h
}
}
except Exception as e:
print(f"Error getting pool stats: {e}")
return {}
def generate_html(self, stats):
"""Generate HTML dashboard"""
html = f"""
<!DOCTYPE html>
<html>
<head>
<title>RinCoin Mining Pool</title>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<style>
body {{ font-family: Arial, sans-serif; margin: 0; padding: 20px; background: #f5f5f5; }}
.container {{ max-width: 1200px; margin: 0 auto; }}
.header {{ background: #2c3e50; color: white; padding: 20px; border-radius: 8px; margin-bottom: 20px; }}
.stats-grid {{ display: grid; grid-template-columns: repeat(auto-fit, minmax(250px, 1fr)); gap: 20px; margin-bottom: 30px; }}
.stat-card {{ background: white; padding: 20px; border-radius: 8px; box-shadow: 0 2px 4px rgba(0,0,0,0.1); }}
.stat-value {{ font-size: 2em; font-weight: bold; color: #3498db; }}
.stat-label {{ color: #7f8c8d; margin-top: 5px; }}
.section {{ background: white; padding: 20px; border-radius: 8px; box-shadow: 0 2px 4px rgba(0,0,0,0.1); margin-bottom: 20px; }}
.section h2 {{ margin-top: 0; color: #2c3e50; }}
table {{ width: 100%; border-collapse: collapse; }}
th, td {{ padding: 12px; text-align: left; border-bottom: 1px solid #ddd; }}
th {{ background: #f8f9fa; font-weight: bold; }}
.block-hash {{ font-family: monospace; font-size: 0.9em; }}
.refresh-btn {{ background: #3498db; color: white; border: none; padding: 10px 20px; border-radius: 4px; cursor: pointer; }}
.refresh-btn:hover {{ background: #2980b9; }}
</style>
</head>
<body>
<div class="container">
<div class="header">
<h1>🏊‍♂️ RinCoin Mining Pool</h1>
<p>Distribute block rewards among multiple miners</p>
</div>
<button class="refresh-btn" onclick="location.reload()">🔄 Refresh</button>
<div class="stats-grid">
<div class="stat-card">
<div class="stat-value">{stats.get('total_miners', 0)}</div>
<div class="stat-label">Total Miners</div>
</div>
<div class="stat-card">
<div class="stat-value">{stats.get('active_miners', 0)}</div>
<div class="stat-label">Active Miners</div>
</div>
<div class="stat-card">
<div class="stat-value">{self.format_hashrate(stats.get('hashrate', 0))}</div>
<div class="stat-label">Hashrate</div>
</div>
<div class="stat-card">
<div class="stat-value">{stats.get('total_blocks', 0)}</div>
<div class="stat-label">Blocks Found</div>
</div>
<div class="stat-card">
<div class="stat-value">{stats.get('pool_balance', 0):.2f}</div>
<div class="stat-label">Pool Balance (RIN)</div>
</div>
</div>
<div class="section">
<h2>📊 Pool Statistics</h2>
<p><strong>24h Shares:</strong> {stats.get('total_shares_24h', 0):,}</p>
<p><strong>Pool Fee:</strong> 1%</p>
<p><strong>Pool Balance:</strong> {stats.get('pool_balance', 0):.8f} RIN</p>
<p><strong>Connection String:</strong> <code>stratum+tcp://YOUR_IP:3333</code></p>
<!-- Debug info -->
<details style="margin-top: 20px; padding: 10px; background: #f8f9fa; border-radius: 4px;">
<summary style="cursor: pointer; font-weight: bold;">🔍 Debug Info</summary>
<p><strong>Recent Difficulty (5min):</strong> {stats.get('debug', {}).get('recent_difficulty', 0):.6f}</p>
<p><strong>Recent Share Count (5min):</strong> {stats.get('debug', {}).get('recent_share_count', 0)}</p>
<p><strong>Total Shares (24h):</strong> {stats.get('debug', {}).get('total_shares_24h', 0):,}</p>
<p><strong>Active Miners:</strong> {stats.get('active_miners', 0)} (last 5 minutes)</p>
<p><strong>Total Miners:</strong> {stats.get('total_miners', 0)} (ever registered)</p>
</details>
</div>
<div class="section">
<h2>📈 Hashrate Chart</h2>
<div class="chart-controls">
<label>Time Window: </label>
<select onchange="changeTimeWindow(this.value)">
<option value="3600">1 Hour</option>
<option value="7200">2 Hours</option>
<option value="14400">4 Hours</option>
<option value="86400">24 Hours</option>
</select>
</div>
<div class="chart-container">
<canvas id="hashrateChart"></canvas>
</div>
</div>
<div class="section">
<h2>👥 Connected Miners</h2>
<table>
<tr>
<th>User</th>
<th>Worker</th>
<th>Connected</th>
<th>Last Share</th>
</tr>
"""
for miner in stats.get('all_miners', []):
user, worker, created, last_share = miner
html += f"""
<tr>
<td>{user}</td>
<td>{worker}</td>
<td>{created}</td>
<td>{last_share or 'Never'}</td>
</tr>
"""
if not stats.get('all_miners', []):
html += """
<tr>
<td colspan="4" style="text-align: center; color: #7f8c8d;">No miners connected</td>
</tr>
"""
html += """
</table>
</div>
<div class="section">
<h2>🏆 Top Miners (24h Shares)</h2>
<table>
<tr>
<th>User</th>
<th>Worker</th>
<th>Shares</th>
<th>Hashrate</th>
<th>Last Share</th>
</tr>
"""
for miner in stats.get('miner_hashrates', []):
user, worker, shares, hashrate, last_share = miner
html += f"""
<tr>
<td>{user}</td>
<td>{worker}</td>
<td>{shares:,}</td>
<td>{self.format_hashrate(hashrate)}</td>
<td>{last_share or 'Never'}</td>
</tr>
"""
if not stats.get('top_miners', []):
html += """
<tr>
<td colspan="4" style="text-align: center; color: #7f8c8d;">No shares submitted yet</td>
</tr>
"""
html += """
</table>
</div>
<div class="section">
<h2>🏆 Recent Blocks</h2>
<table>
<tr>
<th>Height</th>
<th>Hash</th>
<th>Reward</th>
<th>Found At</th>
</tr>
"""
for block in stats.get('recent_blocks', []):
block_hash, height, reward, found_at = block
html += f"""
<tr>
<td>{height}</td>
<td class="block-hash">{block_hash[:16]}...</td>
<td>{reward:.8f} RIN</td>
<td>{found_at}</td>
</tr>
"""
html += """
</table>
</div>
<div class="section">
<h2>🔗 Connect to Pool</h2>
<p>Use any RinHash-compatible miner:</p>
<pre><code>./cpuminer -a rinhash -o stratum+tcp://YOUR_IP:3333 -u username.workername -p x</code></pre>
<p><strong>Replace YOUR_IP with your server's public IP address</strong></p>
</div>
</div>
<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
<script>
// Historical data for chart
const historicalData = {json.dumps([{
'time': row[0],
'shares': row[1],
'difficulty': row[2] or 0
} for row in stats.get('historical_data', [])])};
// Create hashrate chart
const ctx = document.getElementById('hashrateChart').getContext('2d');
const chart = new Chart(ctx, {{
type: 'line',
data: {{
labels: historicalData.map(d => d.time).reverse(),
datasets: [{{
label: 'Hashrate (H/s)',
data: historicalData.map(d => (d.shares / 60.0) * 0.001).reverse(),
borderColor: '#3498db',
backgroundColor: 'rgba(52, 152, 219, 0.1)',
tension: 0.4
}}]
}},
options: {{
responsive: true,
maintainAspectRatio: false,
scales: {{
y: {{
beginAtZero: true,
title: {{
display: true,
text: 'Hashrate (H/s)'
}}
}},
x: {{
title: {{
display: true,
text: 'Time'
}}
}}
}}
}}
}});
function changeTimeWindow(seconds) {{
// Reload page with new time window
const url = new URL(window.location);
url.searchParams.set('window', seconds);
window.location.href = url.toString();
}}
</script>
<script>
// Auto-refresh every 30 seconds
setTimeout(() => location.reload(), 30000);
</script>
</body>
</html>
"""
return html
class PoolWebHandler(BaseHTTPRequestHandler):
def __init__(self, *args, pool_interface=None, **kwargs):
self.pool_interface = pool_interface
super().__init__(*args, **kwargs)
def do_GET(self):
if self.path == '/':
stats = self.pool_interface.get_pool_stats()
html = self.pool_interface.generate_html(stats)
self.send_response(200)
self.send_header('Content-type', 'text/html')
self.end_headers()
self.wfile.write(html.encode('utf-8'))
elif self.path == '/api/stats':
stats = self.pool_interface.get_pool_stats()
self.send_response(200)
self.send_header('Content-type', 'application/json')
self.end_headers()
self.wfile.write(json.dumps(stats).encode('utf-8'))
else:
self.send_response(404)
self.end_headers()
self.wfile.write(b'Not Found')
def log_message(self, format, *args):
# Suppress access logs
pass
def start_web_interface(pool_db, host='0.0.0.0', port=8083, rpc_host='127.0.0.1', rpc_port=9556,
rpc_user='rinrpc', rpc_password='745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90'):
"""Start the web interface server"""
interface = PoolWebInterface(pool_db, host, port, rpc_host, rpc_port, rpc_user, rpc_password)
class Handler(PoolWebHandler):
def __init__(self, *args, **kwargs):
super().__init__(*args, pool_interface=interface, **kwargs)
try:
server = HTTPServer((host, port), Handler)
print(f"🌐 Web interface running on http://{host}:{port}")
print("Press Ctrl+C to stop")
server.serve_forever()
except OSError as e:
if "Address already in use" in str(e):
print(f"⚠️ Port {port} is already in use, web interface not started")
print(f"💡 Try a different port or kill the process using port {port}")
else:
print(f"❌ Failed to start web interface: {e}")
except KeyboardInterrupt:
print("\n🛑 Shutting down web interface...")
server.shutdown()
if __name__ == "__main__":
# This would be called from the main pool server
print("Web interface module loaded")

View File

@@ -0,0 +1,63 @@
#!/bin/bash
# RinCoin Wallet Restoration Script
# Restores a wallet from a dump file on a new RinCoin node.
# Prerequisites: New RinCoin node running in Docker, backup file available.
set -euo pipefail
if [[ $# -ne 1 ]]; then
echo "Usage: $0 <path_to_backup_file>"
echo "Example: $0 ~/rin_wallet_backups/rin_wallet_backup_20230923.txt"
exit 1
fi
BACKUP_FILE="$1"
CONTAINER="rincoin-node"
WALLET_NAME="main" # Will be renamed to avoid conflicts
NEW_WALLET_NAME="restored_main"
# Verify backup file exists
if [[ ! -f "$BACKUP_FILE" ]]; then
echo "Error: Backup file '$BACKUP_FILE' not found."
exit 1
fi
# Check if container is running
if ! sudo docker ps --format '{{.Names}}' | grep -q "^${CONTAINER}$"; then
echo "Error: ${CONTAINER} container is not running. Start it with 'sudo docker start ${CONTAINER}'."
exit 1
fi
echo "Stopping RinCoin node for safe restoration..."
sudo docker stop "$CONTAINER"
echo "Copying backup file into container..."
sudo docker cp "$BACKUP_FILE" "$CONTAINER:/tmp/wallet_backup.txt"
echo "Starting RinCoin node..."
sudo docker start "$CONTAINER"
# Wait for RPC to be ready
echo "Waiting for node to fully start..."
sleep 10
echo "Creating new wallet and importing keys..."
# Create a new wallet to avoid conflicts
sudo docker exec "$CONTAINER" rincoin-cli -datadir=/data -conf=/data/rincoin.conf createwallet "$NEW_WALLET_NAME"
# Import the dump file
sudo docker exec "$CONTAINER" rincoin-cli -datadir=/data -conf=/data/rincoin.conf -rpcwallet="$NEW_WALLET_NAME" importwallet /tmp/wallet_backup.txt
# Clean up
sudo docker exec "$CONTAINER" rm /tmp/wallet_backup.txt
echo "✅ Wallet restored successfully!"
echo "New wallet name: $NEW_WALLET_NAME"
echo ""
echo "Verify with:"
echo "sudo docker exec $CONTAINER rincoin-cli -datadir=/data -conf=/data/rincoin.conf -rpcwallet=$NEW_WALLET_NAME getbalance"
echo ""
echo "To use this wallet, update scripts to use -rpcwallet=$NEW_WALLET_NAME"
echo "Or rename it back: unloadwallet $NEW_WALLET_NAME && loadwallet $WALLET_NAME"

13
MINE/rin/rincoin.conf Normal file
View File

@@ -0,0 +1,13 @@
server=1
daemon=0
listen=1
txindex=1
rpcuser=rinrpc
rpcpassword=745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90
rpcallowip=0.0.0.0/0
rpcport=9556
# performance
maxconnections=64
dbcache=2048

55
MINE/rin/send_rin.sh Normal file
View File

@@ -0,0 +1,55 @@
#!/bin/bash
set -euo pipefail
if [[ ${1-} == "" ]]; then
echo "Usage: $0 <destination_address> [amount]"
echo "Amount defaults to 1 RIN if not specified."
exit 1
fi
ADDRESS="$1"
AMOUNT="${2-1}"
WALLET="main"
CONTAINER="rincoin-node"
CLI_CMD=(sudo docker exec "$CONTAINER" rincoin-cli -datadir=/data -conf=/data/rincoin.conf -rpcwallet="$WALLET")
echo "Checking RinCoin node container..."
if ! sudo docker ps --format '{{.Names}}' | grep -q "^${CONTAINER}$"; then
echo "Error: container ${CONTAINER} is not running. Start it with 'sudo docker start ${CONTAINER}'."
exit 1
fi
echo "Ensuring wallet '${WALLET}' is loaded..."
if ! "${CLI_CMD[@]//-rpcwallet=$WALLET/}" listwallets | grep -q '"main"'; then
echo "Wallet ${WALLET} not loaded, attempting to load..."
"${CLI_CMD[@]//-rpcwallet=$WALLET/}" loadwallet "$WALLET" >/dev/null
fi
echo "Checking available balance..."
BALANCE_RAW=$("${CLI_CMD[@]}" getbalance)
BALANCE=$(printf '%.8f' "$BALANCE_RAW")
if [[ $(bc <<< "$BALANCE_RAW < $AMOUNT") -eq 1 ]]; then
echo "Error: insufficient balance. Available ${BALANCE} RIN, but ${AMOUNT} RIN requested."
exit 1
fi
echo "Broadcasting transaction..."
set +e
TX_OUTPUT=$("${CLI_CMD[@]}" sendtoaddress "$ADDRESS" "$AMOUNT" '' '' false true)
STATUS=$?
set -e
if [[ $STATUS -ne 0 ]]; then
echo "Failed to send transaction."
if [[ $STATUS -eq 4 ]]; then
echo "Wallet appears to be locked. Unlock it with 'sudo docker exec ${CONTAINER} rincoin-cli -datadir=/data -conf=/data/rincoin.conf -rpcwallet=${WALLET} walletpassphrase <passphrase> 600 true' and rerun."
fi
exit $STATUS
fi
echo "Transaction broadcast. TXID: ${TX_OUTPUT}"
echo "Verify with: sudo docker exec ${CONTAINER} rincoin-cli -datadir=/data -conf=/data/rincoin.conf gettransaction ${TX_OUTPUT}"

47
MINE/rin/solo_mining.sh Normal file
View File

@@ -0,0 +1,47 @@
#!/bin/bash
# Solo Mining Script for RinCoin
# Uses local RinCoin node for solo mining
echo "=== RinCoin Solo Mining Setup ==="
echo ""
# Check if rincoin-node container is running
if ! sudo docker ps | grep -q "rincoin-node"; then
echo "Error: rincoin-node container is not running!"
echo "Please start it first:"
echo "sudo docker start rincoin-node"
exit 1
fi
# Get wallet address
RIN_ADDRESS=$(sudo docker exec rincoin-node rincoin-cli -datadir=/data -conf=/data/rincoin.conf -rpcwallet=main getnewaddress 2>/dev/null)
if [ -z "$RIN_ADDRESS" ]; then
echo "Error: Could not get RinCoin address!"
echo "Make sure the wallet is created and the node is synced."
exit 1
fi
echo "RinCoin Address: $RIN_ADDRESS"
echo ""
# Check node sync status
SYNC_STATUS=$(sudo docker exec rincoin-node rincoin-cli -datadir=/data -conf=/data/rincoin.conf getblockchaininfo | grep -o '"initialblockdownload": [^,]*' | cut -d' ' -f2)
if [ "$SYNC_STATUS" = "true" ]; then
echo "⚠️ WARNING: Node is still syncing (initialblockdownload: true)"
echo "Solo mining may not work properly until sync is complete."
echo ""
fi
echo "Starting solo mining with cpuminer-opt-rin..."
echo "Algorithm: rinhash"
echo "Target: Local RinCoin node (127.0.0.1:9555)"
echo "Wallet: $RIN_ADDRESS"
echo ""
echo "Press Ctrl+C to stop mining"
echo ""
# Start solo mining
sudo docker exec -it amd-strix-halo-llama-rocm bash -c "/mnt/dl/rinhash/cpuminer-opt-rin/cpuminer -a rinhash -o stratum+tcp://127.0.0.1:9555 -u $RIN_ADDRESS -p x -t 32"

View File

@@ -0,0 +1,171 @@
#!/bin/bash
# RinCoin Solo Mining using Built-in Core Mining
# Uses RinCoin Core's generatetoaddress command
# Default address (can be overridden with command line parameter)
DEFAULT_ADDRESS="rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q"
# Get total CPU cores for default thread count
TOTAL_CORES=$(nproc)
DEFAULT_THREADS=$TOTAL_CORES
# Parse command line arguments
RIN_ADDRESS=""
THREAD_COUNT=""
# Parse arguments
while [[ $# -gt 0 ]]; do
case $1 in
-a|--address)
RIN_ADDRESS="$2"
shift 2
;;
-t|--threads)
THREAD_COUNT="$2"
shift 2
;;
-h|--help)
echo "Usage: $0 [OPTIONS]"
echo ""
echo "Options:"
echo " -a, --address ADDRESS RinCoin address to mine to (default: $DEFAULT_ADDRESS)"
echo " -t, --threads COUNT Number of threads to use (default: $DEFAULT_THREADS)"
echo " -h, --help Show this help message"
echo ""
echo "Examples:"
echo " $0 # Use defaults (all cores, default address)"
echo " $0 -a rin1q... -t 16 # Custom address and 16 threads"
echo " $0 --address rin1q... --threads 8 # Custom address and 8 threads"
exit 0
;;
*)
echo "Unknown option: $1"
echo "Use -h or --help for usage information"
exit 1
;;
esac
done
# Set defaults if not provided
if [ -z "$RIN_ADDRESS" ]; then
RIN_ADDRESS="$DEFAULT_ADDRESS"
echo "No address provided, using default: $RIN_ADDRESS"
fi
if [ -z "$THREAD_COUNT" ]; then
THREAD_COUNT="$DEFAULT_THREADS"
echo "No thread count provided, using all cores: $THREAD_COUNT"
fi
# Validate thread count
if ! [[ "$THREAD_COUNT" =~ ^[0-9]+$ ]] || [ "$THREAD_COUNT" -lt 1 ] || [ "$THREAD_COUNT" -gt "$TOTAL_CORES" ]; then
echo "❌ Error: Invalid thread count: $THREAD_COUNT"
echo "Thread count must be between 1 and $TOTAL_CORES"
exit 1
fi
echo "=== RinCoin Solo Mining (Built-in Core Mining) ==="
echo "CPU Cores Available: $TOTAL_CORES"
echo "Threads to Use: $THREAD_COUNT"
echo "Target Address: $RIN_ADDRESS"
echo ""
echo ""
# Configuration
RPC_HOST="127.0.0.1"
RPC_PORT="9556"
RPC_USER="rinrpc"
RPC_PASS="745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90"
# Function to call RPC
call_rpc() {
local method="$1"
local params="$2"
curl -s --user "$RPC_USER:$RPC_PASS" \
-H 'content-type: text/plain' \
--data "{\"jsonrpc\":\"1.0\",\"id\":\"curl\",\"method\":\"$method\",\"params\":$params}" \
"http://$RPC_HOST:$RPC_PORT/"
}
# Wait for node to be ready
echo "Waiting for RinCoin node to be ready..."
while true; do
response=$(call_rpc "getblockchaininfo" "[]")
if [[ $response != *"Loading block index"* ]]; then
break
fi
echo "Node still loading... waiting 10 seconds"
sleep 10
done
echo "✅ Node is ready!"
echo ""
# Load wallet if not already loaded
echo "Loading wallet..."
wallet_response=$(call_rpc "loadwallet" "[\"main\"]")
if [[ $wallet_response == *"error"* ]] && [[ $wallet_response == *"already loaded"* ]]; then
echo "✅ Wallet already loaded"
else
echo "✅ Wallet loaded successfully"
fi
echo ""
# Validate the provided address (basic check)
if [[ ! "$RIN_ADDRESS" =~ ^rin1[a-zA-Z0-9]{25,}$ ]]; then
echo "❌ Error: Invalid RinCoin address format: $RIN_ADDRESS"
echo "RinCoin addresses should start with 'rin1' and be ~30 characters long"
exit 1
fi
echo "✅ Using RinCoin Address: $RIN_ADDRESS"
echo ""
# Get blockchain info
echo "Blockchain Status:"
blockchain_info=$(call_rpc "getblockchaininfo" "[]")
blocks=$(echo "$blockchain_info" | grep -o '"blocks":[^,]*' | cut -d':' -f2)
headers=$(echo "$blockchain_info" | grep -o '"headers":[^,]*' | cut -d':' -f2)
difficulty=$(echo "$blockchain_info" | grep -o '"difficulty":[^,]*' | cut -d':' -f2)
echo "Blocks: $blocks"
echo "Headers: $headers"
echo "Difficulty: $difficulty"
echo ""
echo "⚠️ IMPORTANT: Built-in Core Mining Limitations:"
echo "1. Uses CPU only (not GPU)"
echo "2. Very low hashpower compared to specialized miners"
echo "3. Extremely low chance of finding blocks solo"
echo "4. Best for testing, not profitable mining"
echo "5. Thread count affects mining attempts per cycle"
echo ""
echo "🚀 Starting Built-in Solo Mining..."
echo "Target Address: $RIN_ADDRESS"
echo "Threads: $THREAD_COUNT"
echo "Press Ctrl+C to stop mining"
echo ""
# Start built-in mining with specified thread count
# Note: RinCoin Core's generatetoaddress doesn't directly support thread count
# but we can run multiple instances or adjust the maxtries parameter
while true; do
echo "Attempting to mine 1 block with $THREAD_COUNT threads..."
# Adjust maxtries based on thread count for better distribution
adjusted_tries=$((1000000 * THREAD_COUNT / TOTAL_CORES))
mining_result=$(call_rpc "generatetoaddress" "[1, \"$RIN_ADDRESS\", $adjusted_tries]")
if [[ $mining_result == *"result"* ]] && [[ $mining_result != *"[]"* ]]; then
echo "🎉 BLOCK FOUND!"
echo "Result: $mining_result"
break
else
echo "No block found in this attempt (tries: $adjusted_tries). Retrying..."
sleep 5
fi
done

View File

@@ -0,0 +1,81 @@
#!/bin/bash
# Remote Solo Mining Script for RinCoin
# Connects to RinCoin node over network/RPC
# Configuration
RPC_HOST="127.0.0.1" # Change to your server IP for remote access
RPC_PORT="9556"
RPC_USER="rinrpc"
RPC_PASS="745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90"
echo "=== Remote RinCoin Solo Mining Setup ==="
echo "RPC Host: $RPC_HOST:$RPC_PORT"
echo ""
# Test RPC connection
echo "Testing RPC connection..."
RPC_RESPONSE=$(curl -s --user "$RPC_USER:$RPC_PASS" \
-H 'content-type: text/plain' \
--data '{"jsonrpc":"1.0","id":"curl","method":"getblockchaininfo","params":[]}' \
"http://$RPC_HOST:$RPC_PORT/")
if [[ $RPC_RESPONSE == *"error"* ]]; then
echo "❌ Error: Could not connect to RinCoin RPC!"
echo "Response: $RPC_RESPONSE"
echo ""
echo "Check:"
echo "1. RinCoin node is running"
echo "2. RPC port $RPC_PORT is accessible"
echo "3. Firewall allows connections to port $RPC_PORT"
exit 1
fi
echo "✅ RPC connection successful!"
echo ""
# Get wallet address via RPC
echo "Getting wallet address..."
WALLET_RESPONSE=$(curl -s --user "$RPC_USER:$RPC_PASS" \
-H 'content-type: text/plain' \
--data '{"jsonrpc":"1.0","id":"curl","method":"getnewaddress","params":[]}' \
"http://$RPC_HOST:$RPC_PORT/")
RIN_ADDRESS=$(echo "$WALLET_RESPONSE" | grep -o '"result":"[^"]*"' | cut -d'"' -f4)
if [ -z "$RIN_ADDRESS" ]; then
echo "❌ Error: Could not get RinCoin address!"
echo "Response: $WALLET_RESPONSE"
echo ""
echo "Make sure wallet 'main' exists:"
echo "curl --user $RPC_USER:$RPC_PASS -H 'content-type: text/plain' --data '{\"jsonrpc\":\"1.0\",\"id\":\"curl\",\"method\":\"createwallet\",\"params\":[\"main\"]}' http://$RPC_HOST:$RPC_PORT/"
exit 1
fi
echo "✅ RinCoin Address: $RIN_ADDRESS"
echo ""
# Check node sync status
SYNC_RESPONSE=$(curl -s --user "$RPC_USER:$RPC_PASS" \
-H 'content-type: text/plain' \
--data '{"jsonrpc":"1.0","id":"curl","method":"getblockchaininfo","params":[]}' \
"http://$RPC_HOST:$RPC_PORT/")
SYNC_STATUS=$(echo "$SYNC_RESPONSE" | grep -o '"initialblockdownload":[^,]*' | cut -d':' -f2 | tr -d ' ')
if [ "$SYNC_STATUS" = "true" ]; then
echo "⚠️ WARNING: Node is still syncing (initialblockdownload: true)"
echo "Solo mining may not work properly until sync is complete."
echo ""
fi
echo "Starting remote solo mining with cpuminer-opt-rin..."
echo "Algorithm: rinhash"
echo "Target: RinCoin node at $RPC_HOST:9555"
echo "Wallet: $RIN_ADDRESS"
echo ""
echo "Press Ctrl+C to stop mining"
echo ""
# Start solo mining (connect to P2P port, not RPC)
sudo docker exec -it amd-strix-halo-llama-rocm bash -c "/mnt/dl/rinhash/cpuminer-opt-rin/cpuminer -a rinhash -o stratum+tcp://$RPC_HOST:9555 -u $RIN_ADDRESS -p x -t 32"

View File

@@ -0,0 +1,76 @@
#!/bin/bash
# RinCoin Solo Mining via RPC
# This script uses RinCoin's RPC interface for solo mining
echo "=== RinCoin Solo Mining via RPC ==="
echo ""
# Configuration
RPC_HOST="127.0.0.1"
RPC_PORT="9556"
RPC_USER="rinrpc"
RPC_PASS="745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90"
# Function to call RPC
call_rpc() {
local method="$1"
local params="$2"
curl -s --user "$RPC_USER:$RPC_PASS" \
-H 'content-type: text/plain' \
--data "{\"jsonrpc\":\"1.0\",\"id\":\"curl\",\"method\":\"$method\",\"params\":$params}" \
"http://$RPC_HOST:$RPC_PORT/"
}
# Wait for node to be ready
echo "Waiting for RinCoin node to be ready..."
while true; do
response=$(call_rpc "getblockchaininfo" "[]")
if [[ $response != *"Loading block index"* ]]; then
break
fi
echo "Node still loading... waiting 10 seconds"
sleep 10
done
echo "✅ Node is ready!"
echo ""
# Get wallet address
echo "Getting wallet address..."
wallet_response=$(call_rpc "getnewaddress" "[]")
rin_address=$(echo "$wallet_response" | grep -o '"result":"[^"]*"' | cut -d'"' -f4)
if [ -z "$rin_address" ]; then
echo "❌ Error: Could not get RinCoin address!"
echo "Response: $wallet_response"
exit 1
fi
echo "✅ RinCoin Address: $rin_address"
echo ""
# Get blockchain info
echo "Blockchain Status:"
blockchain_info=$(call_rpc "getblockchaininfo" "[]")
blocks=$(echo "$blockchain_info" | grep -o '"blocks":[^,]*' | cut -d':' -f2)
headers=$(echo "$blockchain_info" | grep -o '"headers":[^,]*' | cut -d':' -f2)
difficulty=$(echo "$blockchain_info" | grep -o '"difficulty":[^,]*' | cut -d':' -f2)
echo "Blocks: $blocks"
echo "Headers: $headers"
echo "Difficulty: $difficulty"
echo ""
echo "⚠️ IMPORTANT: RinCoin solo mining requires:"
echo "1. A fully synced node (currently at block $blocks of $headers)"
echo "2. Mining software that supports RinCoin's RPC mining protocol"
echo "3. Very high hashpower to find blocks solo"
echo ""
echo "For now, we recommend pool mining for consistent rewards:"
echo ""
echo "Pool Mining Command:"
echo "sudo docker exec -it amd-strix-halo-llama-rocm bash -c \"/mnt/dl/rinhash/cpuminer-opt-rin/cpuminer -a rinhash -o stratum+tcp://rinhash.mine.zergpool.com:7148 -u bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j -p c=BTC,mc=RIN,ID=StrixHalo -t 32\""
echo ""
echo "Your RinCoin address for solo mining: $rin_address"

View File

@@ -0,0 +1,76 @@
#!/bin/bash
# RinCoin Solo Mining via RPC
# This script uses RinCoin's RPC interface for solo mining
echo "=== RinCoin Solo Mining via RPC ==="
echo ""
# Configuration
RPC_HOST="127.0.0.1"
RPC_PORT="9556"
RPC_USER="rinrpc"
RPC_PASS="745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90"
# Function to call RPC
call_rpc() {
local method="$1"
local params="$2"
curl -s --user "$RPC_USER:$RPC_PASS" \
-H 'content-type: text/plain' \
--data "{\"jsonrpc\":\"1.0\",\"id\":\"curl\",\"method\":\"$method\",\"params\":$params}" \
"http://$RPC_HOST:$RPC_PORT/"
}
# Wait for node to be ready
echo "Waiting for RinCoin node to be ready..."
while true; do
response=$(call_rpc "getblockchaininfo" "[]")
if [[ $response != *"Loading block index"* ]]; then
break
fi
echo "Node still loading... waiting 10 seconds"
sleep 10
done
echo "✅ Node is ready!"
echo ""
# Get wallet address
echo "Getting wallet address..."
wallet_response=$(call_rpc "getnewaddress" "[]")
rin_address=$(echo "$wallet_response" | grep -o '"result":"[^"]*"' | cut -d'"' -f4)
if [ -z "$rin_address" ]; then
echo "❌ Error: Could not get RinCoin address!"
echo "Response: $wallet_response"
exit 1
fi
echo "✅ RinCoin Address: $rin_address"
echo ""
# Get blockchain info
echo "Blockchain Status:"
blockchain_info=$(call_rpc "getblockchaininfo" "[]")
blocks=$(echo "$blockchain_info" | grep -o '"blocks":[^,]*' | cut -d':' -f2)
headers=$(echo "$blockchain_info" | grep -o '"headers":[^,]*' | cut -d':' -f2)
difficulty=$(echo "$blockchain_info" | grep -o '"difficulty":[^,]*' | cut -d':' -f2)
echo "Blocks: $blocks"
echo "Headers: $headers"
echo "Difficulty: $difficulty"
echo ""
echo "⚠️ IMPORTANT: RinCoin solo mining requires:"
echo "1. A fully synced node (currently at block $blocks of $headers)"
echo "2. Mining software that supports RinCoin's RPC mining protocol"
echo "3. Very high hashpower to find blocks solo"
echo ""
echo "For now, we recommend pool mining for consistent rewards:"
echo ""
echo "Pool Mining Command:"
echo "sudo docker exec -it amd-strix-halo-llama-rocm bash -c \"/mnt/dl/rinhash/cpuminer-opt-rin/cpuminer -a rinhash -o stratum+tcp://rinhash.mine.zergpool.com:7148 -u bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j -p c=BTC,mc=RIN,ID=StrixHalo -t 32\""
echo ""
echo "Your RinCoin address for solo mining: $rin_address"

View File

@@ -0,0 +1,83 @@
#!/bin/bash
# RinCoin Mining Pool Server Startup Script
# Distributes block rewards among multiple miners
echo "=== RinCoin Mining Pool Server ==="
echo ""
# Check if RinCoin node is running
echo "Checking RinCoin node status..."
if ! curl -s -u rinrpc:745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90 \
-H 'content-type: text/plain' \
--data '{"jsonrpc":"1.0","id":"curl","method":"getblockchaininfo","params":[]}' \
http://127.0.0.1:9556/ > /dev/null; then
echo "❌ RinCoin node is not running!"
echo "Start it first with: docker start rincoin-node"
exit 1
fi
echo "✅ RinCoin node is running"
# Check Python dependencies
echo "Checking Python dependencies..."
python3 -c "import requests, sqlite3" 2>/dev/null || {
echo "Installing python3-requests..."
sudo apt-get update && sudo apt-get install -y python3-requests
}
echo "✅ Python dependencies ready"
# Check if port 3333 is already in use
if netstat -tln | grep -q ":3333 "; then
echo ""
echo "⚠️ Port 3333 is already in use!"
echo ""
echo "🔍 Process using port 3333:"
sudo netstat -tlnp | grep ":3333 " || echo "Could not determine process"
echo ""
echo "🛑 To kill existing process:"
echo "sudo lsof -ti:3333 | xargs sudo kill -9"
echo ""
read -p "Kill existing process and continue? (y/N): " -n 1 -r
echo
if [[ $REPLY =~ ^[Yy]$ ]]; then
echo "Killing processes using port 3333..."
sudo lsof -ti:3333 | xargs sudo kill -9 2>/dev/null || echo "No processes to kill"
sleep 2
else
echo "Exiting..."
exit 1
fi
fi
echo ""
echo "🚀 Starting Mining Pool Server..."
echo "This will distribute block rewards among multiple miners"
echo ""
echo "Pool Features:"
echo "- Multiple miner support"
echo "- Share-based reward distribution"
echo "- Pool fee: 1%"
echo "- Real-time statistics"
echo "- Web dashboard on port 8083"
echo ""
echo "After it starts, miners can connect with:"
echo ""
echo "Option 1: Address as username"
echo "./cpuminer -a rinhash -o stratum+tcp://YOUR_IP:3333 -u rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q -p x"
echo ""
echo "Option 2: Address.workername format"
echo "./cpuminer -a rinhash -o stratum+tcp://YOUR_IP:3333 -u rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q.worker1 -p x"
echo ""
echo "Option 3: Traditional username (rewards to pool address)"
echo "./cpuminer -a rinhash -o stratum+tcp://YOUR_IP:3333 -u username.workername -p x"
echo ""
echo "🌐 Web Dashboard: http://YOUR_IP:8083"
echo "📊 View real-time pool statistics, miners, and blocks"
echo ""
echo "Press Ctrl+C to stop the pool"
# Start the mining pool
python3 MINE/rin/stratum_pool.py

View File

@@ -0,0 +1,68 @@
#!/bin/bash
# Start RinCoin Stratum Proxy Server
# Bridges cpuminer-opt-rin to RinCoin node
echo "=== RinCoin Stratum Proxy Server ==="
echo ""
# Check if RinCoin node is running
if ! sudo docker ps | grep -q "rincoin-node"; then
echo "❌ Error: rincoin-node container is not running!"
echo "Please start it first:"
echo "sudo docker start rincoin-node"
exit 1
fi
echo "✅ RinCoin node is running"
# Check if Python3 and requests are available
if ! command -v python3 &> /dev/null; then
echo "❌ Error: python3 is not installed!"
echo "Please install it: sudo apt-get install python3"
exit 1
fi
# Install requests if not available
python3 -c "import requests" 2>/dev/null || {
echo "Installing python3-requests..."
sudo apt-get update && sudo apt-get install -y python3-requests
}
echo "✅ Python dependencies ready"
# Check if port 3334 is already in use
if netstat -tln | grep -q ":3334 "; then
echo ""
echo "⚠️ Port 3334 is already in use!"
echo ""
echo "🔍 Process using port 3334:"
sudo netstat -tlnp | grep ":3334 " || echo "Could not determine process"
echo ""
echo "🛑 To kill existing process:"
echo "sudo lsof -ti:3334 | xargs sudo kill -9"
echo ""
read -p "Kill existing process and continue? (y/N): " -n 1 -r
echo
if [[ $REPLY =~ ^[Yy]$ ]]; then
echo "Killing processes using port 3334..."
sudo lsof -ti:3334 | xargs sudo kill -9 2>/dev/null || echo "No processes to kill"
sleep 2
else
echo "Exiting..."
exit 1
fi
fi
echo ""
echo "🚀 Starting Stratum Proxy Server..."
echo ""
echo "After it starts, connect your miner with:"
echo "sudo docker exec -it amd-strix-halo-llama-rocm bash -c \"/mnt/dl/rinhash/cpuminer-opt-rin/cpuminer -a rinhash -o stratum+tcp://127.0.0.1:3334 -u user -p pass -t 28\""
echo ""
echo "Press Ctrl+C to stop the proxy"
echo ""
# Start the proxy
cd "$(dirname "$0")"
python3 stratum_proxy.py --submit-all-blocks

View File

@@ -0,0 +1,14 @@
#!/bin/bash
set -euo pipefail
SCRIPT_DIR="/mnt/shared/DEV/repos/d-popov.com/scripts/MINE/rin/web_wallet"
if ! command -v python3 >/dev/null 2>&1; then
echo "python3 is required"
exit 1
fi
python3 "${SCRIPT_DIR}/server.py"

602
MINE/rin/stratum_pool.py Normal file
View File

@@ -0,0 +1,602 @@
#!/usr/bin/env python3
"""
RinCoin Mining Pool Server
Distributes block rewards among multiple miners based on share contributions
"""
import socket
import threading
import json
import time
import requests
import hashlib
import struct
import sqlite3
from datetime import datetime
from requests.auth import HTTPBasicAuth
# Import web interface
from pool_web_interface import start_web_interface
# Import stratum base class
from stratum_proxy import RinCoinStratumBase
class RinCoinMiningPool(RinCoinStratumBase):
def __init__(self, stratum_host='0.0.0.0', stratum_port=3333,
rpc_host='127.0.0.1', rpc_port=9556,
rpc_user='rinrpc', rpc_password='745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90',
pool_address='rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q',
pool_fee_percent=1.0):
# Initialize base class
super().__init__(stratum_host, stratum_port, rpc_host, rpc_port, rpc_user, rpc_password, pool_address)
self.pool_address = pool_address
self.pool_fee_percent = pool_fee_percent
# Pool statistics
self.total_shares = 0
self.total_blocks = 0
self.pool_hashrate = 0
# Database for persistent storage
self.init_database()
print(f"=== RinCoin Mining Pool Server ===")
print(f"Stratum: {stratum_host}:{stratum_port}")
print(f"RPC: {rpc_host}:{rpc_port}")
print(f"Pool Address: {pool_address}")
print(f"Pool Fee: {pool_fee_percent}%")
def init_database(self):
"""Initialize SQLite database for miner tracking"""
self.db = sqlite3.connect(':memory:', check_same_thread=False)
cursor = self.db.cursor()
# Create tables
cursor.execute('''
CREATE TABLE IF NOT EXISTS miners (
id INTEGER PRIMARY KEY,
user TEXT NOT NULL,
worker TEXT NOT NULL,
address TEXT,
shares INTEGER DEFAULT 0,
last_share TIMESTAMP,
last_hashrate REAL DEFAULT 0,
created TIMESTAMP DEFAULT CURRENT_TIMESTAMP
)
''')
cursor.execute('''
CREATE TABLE IF NOT EXISTS shares (
id INTEGER PRIMARY KEY,
miner_id INTEGER,
job_id TEXT,
difficulty REAL,
submitted TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
FOREIGN KEY (miner_id) REFERENCES miners (id)
)
''')
cursor.execute('''
CREATE TABLE IF NOT EXISTS blocks (
id INTEGER PRIMARY KEY,
block_hash TEXT,
height INTEGER,
reward REAL,
pool_fee REAL,
miner_rewards TEXT, -- JSON of {address: amount}
found_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
)
''')
# Samples for pool hashrate chart
cursor.execute('''
CREATE TABLE IF NOT EXISTS hashrate_samples (
id INTEGER PRIMARY KEY,
ts TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
hashrate REAL
)
''')
self.db.commit()
def get_pool_block_template(self):
"""Get new block template and create pool-style job"""
template = super().get_block_template()
if template:
# Convert to pool-style job format if needed
job = self.current_job
if job:
# Add pool-specific fields
job["coinb1"] = "01000000" + "0" * 60
job["coinb2"] = "ffffffff"
job["merkle_branch"] = []
job["clean_jobs"] = True
return job
return None
def validate_rincoin_address(self, address):
"""Validate if an address is a valid RinCoin address"""
try:
return self.decode_bech32_address(address) is not None
except:
return False
def register_miner(self, user, worker, address=None):
"""Register or update miner in database"""
cursor = self.db.cursor()
# Check if miner exists
cursor.execute('SELECT id, address FROM miners WHERE user = ? AND worker = ?', (user, worker))
result = cursor.fetchone()
if result:
miner_id, existing_address = result
if address and not existing_address:
cursor.execute('UPDATE miners SET address = ? WHERE id = ?', (address, miner_id))
self.db.commit()
return miner_id
else:
# Create new miner
cursor.execute('INSERT INTO miners (user, worker, address) VALUES (?, ?, ?)', (user, worker, address))
self.db.commit()
return cursor.lastrowid
def record_share(self, miner_id, job_id, difficulty):
"""Record a share submission"""
cursor = self.db.cursor()
# Record share
cursor.execute('INSERT INTO shares (miner_id, job_id, difficulty) VALUES (?, ?, ?)',
(miner_id, job_id, difficulty))
# Update miner stats
cursor.execute('UPDATE miners SET shares = shares + 1, last_share = CURRENT_TIMESTAMP WHERE id = ?', (miner_id,))
self.db.commit()
self.total_shares += 1
def distribute_block_reward(self, block_hash, block_height, total_reward):
"""Distribute block reward among miners based on their shares"""
cursor = self.db.cursor()
# Calculate pool fee
pool_fee = total_reward * (self.pool_fee_percent / 100.0)
miner_reward = total_reward - pool_fee
# Get shares from last 24 hours
cursor.execute('''
SELECT m.address, COUNT(s.id) as share_count, SUM(s.difficulty) as total_difficulty
FROM miners m
JOIN shares s ON m.id = s.miner_id
WHERE s.submitted > datetime('now', '-1 day')
GROUP BY m.id, m.address
HAVING share_count > 0
''')
miners = cursor.fetchall()
if not miners:
print("No miners with shares in last 24 hours")
return
# Calculate total difficulty
total_difficulty = sum(row[2] for row in miners)
# Separate miners with and without addresses
miners_with_addresses = []
miners_without_addresses = []
total_difficulty_with_addresses = 0
total_difficulty_without_addresses = 0
for address, share_count, difficulty in miners:
if address:
miners_with_addresses.append((address, share_count, difficulty))
total_difficulty_with_addresses += difficulty
else:
miners_without_addresses.append((address, share_count, difficulty))
total_difficulty_without_addresses += difficulty
# Calculate total difficulty
total_difficulty = total_difficulty_with_addresses + total_difficulty_without_addresses
if total_difficulty == 0:
print("No valid difficulty found")
return
# Distribute rewards
miner_rewards = {}
# First, distribute to miners with valid addresses
if miners_with_addresses:
for address, share_count, difficulty in miners_with_addresses:
reward_share = (difficulty / total_difficulty) * miner_reward
miner_rewards[address] = reward_share
print(f"💰 Miner {address}: {reward_share:.8f} RIN ({difficulty} difficulty)")
# Calculate undistributed rewards (from miners without addresses)
if miners_without_addresses:
undistributed_reward = 0
for address, share_count, difficulty in miners_without_addresses:
undistributed_reward += (difficulty / total_difficulty) * miner_reward
print(f"⚠️ Miner without address: {difficulty} difficulty -> {undistributed_reward:.8f} RIN to pool")
# Keep undistributed rewards for pool (no redistribution)
print(f"💰 Pool keeps {undistributed_reward:.8f} RIN from miners without addresses")
# Record block
cursor.execute('''
INSERT INTO blocks (block_hash, height, reward, pool_fee, miner_rewards)
VALUES (?, ?, ?, ?, ?)
''', (block_hash, block_height, total_reward, pool_fee, json.dumps(miner_rewards)))
self.db.commit()
self.total_blocks += 1
print(f"🎉 Block {block_height} reward distributed!")
print(f"💰 Pool fee: {pool_fee:.8f} RIN")
print(f"💰 Total distributed: {sum(miner_rewards.values()):.8f} RIN")
# Summary
if miners_without_addresses:
print(f"📊 Summary: {len(miners_with_addresses)} miners with addresses, {len(miners_without_addresses)} without (rewards to pool)")
# Use inherited send_stratum_response and send_stratum_notification from base class
def handle_stratum_message(self, client, addr, message):
"""Handle incoming Stratum message from miner"""
try:
data = json.loads(message.strip())
method = data.get("method")
msg_id = data.get("id")
params = data.get("params", [])
print(f"[{addr}] {method}: {params}")
if method == "mining.subscribe":
# Subscribe response
self.send_stratum_response(client, msg_id, [
[["mining.set_difficulty", "subscription_id"], ["mining.notify", "subscription_id"]],
"extranonce1",
4
])
# Send difficulty (lower for CPU mining)
self.send_stratum_notification(client, "mining.set_difficulty", [0.0001])
# Send initial job
if self.get_pool_block_template():
job = self.current_job
self.send_stratum_notification(client, "mining.notify", [
job["job_id"],
job["prevhash"],
job["coinb1"],
job["coinb2"],
job["merkle_branch"],
f"{job['version']:08x}",
job["bits"],
job["ntime"],
job["clean_jobs"]
])
elif method == "mining.extranonce.subscribe":
# Handle extranonce subscription
print(f"[{addr}] Extranonce subscription requested")
self.send_stratum_response(client, msg_id, True)
elif method == "mining.authorize":
# Parse user.worker format
if len(params) >= 2:
user_worker = params[0]
password = params[1] if len(params) > 1 else ""
# Extract user and worker
if '.' in user_worker:
user, worker = user_worker.split('.', 1)
else:
user = user_worker
worker = "default"
# Check if user contains a RinCoin address (starts with 'rin')
miner_address = None
if user.startswith('rin'):
# User is a RinCoin address
if self.validate_rincoin_address(user):
miner_address = user
user = f"miner_{miner_address[:8]}" # Create a user ID from address
print(f"[{addr}] ✅ Miner using valid RinCoin address: {miner_address}")
else:
print(f"[{addr}] ❌ Invalid RinCoin address: {user}")
self.send_stratum_response(client, msg_id, False, "Invalid RinCoin address")
return
elif '.' in user and user.split('.')[0].startswith('rin'):
# Format: rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q.workername
address_part, worker_part = user.split('.', 1)
if address_part.startswith('rin'):
if self.validate_rincoin_address(address_part):
miner_address = address_part
user = f"miner_{miner_address[:8]}"
worker = worker_part
print(f"[{addr}] ✅ Miner using valid RinCoin address format: {miner_address}.{worker}")
else:
print(f"[{addr}] ❌ Invalid RinCoin address: {address_part}")
self.send_stratum_response(client, msg_id, False, "Invalid RinCoin address")
return
# Register miner with address
miner_id = self.register_miner(user, worker, miner_address)
# Store client info
self.clients[addr] = {
'client': client,
'user': user,
'worker': worker,
'miner_id': miner_id,
'address': miner_address,
'shares': 0,
'last_share': time.time(),
'extranonce1': '00000000' # Default extranonce1
}
if miner_address:
print(f"[{addr}] ✅ Authorized: {user}.{worker} -> {miner_address}")
else:
print(f"[{addr}] ⚠️ Authorized: {user}.{worker} (rewards will go to pool address)")
self.send_stratum_response(client, msg_id, True)
else:
self.send_stratum_response(client, msg_id, False, "Invalid authorization")
elif method == "mining.submit":
# Submit share
if addr not in self.clients:
self.send_stratum_response(client, msg_id, False, "Not authorized")
return
miner_info = self.clients[addr]
try:
if self.current_job and len(params) >= 5:
username = params[0]
job_id = params[1]
extranonce2 = params[2]
ntime = params[3]
nonce = params[4]
# Use base class to validate and submit share
extranonce1 = miner_info.get('extranonce1', '00000000')
miner_address = miner_info.get('address')
# For pool mining, always mine to pool address
success, message = self.submit_share(
self.current_job, extranonce1, extranonce2, ntime, nonce,
target_address=self.pool_address
)
if success:
# Record share with estimated difficulty
actual_difficulty = 0.00133 # Estimated for ~381 kH/s
self.record_share(miner_info['miner_id'], job_id, actual_difficulty)
# Update miner stats
now_ts = time.time()
prev_ts = miner_info.get('last_share') or now_ts
dt = max(now_ts - prev_ts, 1e-3)
miner_hashrate = actual_difficulty * (2**32) / dt
if miner_info['shares'] == 0:
miner_hashrate = 381000 # Default estimate
miner_info['shares'] += 1
miner_info['last_share'] = now_ts
# Update database
try:
cursor = self.db.cursor()
cursor.execute('UPDATE miners SET last_share = CURRENT_TIMESTAMP, last_hashrate = ? WHERE id = ?',
(miner_hashrate, miner_info['miner_id']))
self.db.commit()
except Exception as e:
print(f"DB update error: {e}")
print(f"[{addr}] ✅ Share accepted from {miner_info['user']}.{miner_info['worker']} (Total: {miner_info['shares']})")
self.send_stratum_response(client, msg_id, True)
# If block was found, distribute rewards
if "Block found" in message:
print(f"🎉 [{addr}] BLOCK FOUND!")
# Get block info and distribute rewards
total_reward = self.current_job['coinbasevalue'] / 100000000 if self.current_job else 25.0
self.distribute_block_reward("pending", self.current_job['height'] if self.current_job else 0, total_reward)
else:
# Accept as share for pool statistics even if block validation fails
self.send_stratum_response(client, msg_id, True)
else:
print(f"[{addr}] Invalid share parameters")
self.send_stratum_response(client, msg_id, False, "Invalid parameters")
except Exception as e:
print(f"[{addr}] Share processing error: {e}")
# Still accept the share for mining statistics
self.send_stratum_response(client, msg_id, True)
else:
print(f"[{addr}] ⚠️ Unknown method: {method}")
# Send null result for unknown methods (standard Stratum behavior)
self.send_stratum_response(client, msg_id, None, None)
except json.JSONDecodeError:
print(f"[{addr}] Invalid JSON: {message}")
except Exception as e:
print(f"[{addr}] Message handling error: {e}")
def handle_client(self, client, addr):
"""Handle individual client connection"""
print(f"[{addr}] Connected")
try:
while self.running:
data = client.recv(4096)
if not data:
break
# Handle multiple messages in one packet
messages = data.decode('utf-8').strip().split('\n')
for message in messages:
if message:
self.handle_stratum_message(client, addr, message)
except Exception as e:
print(f"[{addr}] Client error: {e}")
finally:
client.close()
if addr in self.clients:
del self.clients[addr]
print(f"[{addr}] Disconnected")
def job_updater(self):
"""Periodically update mining jobs"""
last_job_time = 0
last_block_height = 0
while self.running:
try:
# Check for new blocks every 10 seconds
time.sleep(10)
# Get current blockchain info
blockchain_info = self.rpc_call("getblockchaininfo")
if blockchain_info:
current_height = blockchain_info.get('blocks', 0)
# Create new job if:
# 1. New block detected
# 2. 30+ seconds since last job
# 3. No current job exists
should_create_job = (
current_height != last_block_height or
time.time() - last_job_time > 30 or
not self.current_job
)
if should_create_job:
if self.get_pool_block_template():
job = self.current_job
last_job_time = time.time()
last_block_height = current_height
print(f"📦 New job created: {job['job_id']} (block {current_height})")
# Send to all connected clients
for addr, miner_info in list(self.clients.items()):
try:
self.send_stratum_notification(miner_info['client'], "mining.notify", [
job["job_id"],
job["prevhash"],
job["coinb1"],
job["coinb2"],
job["merkle_branch"],
f"{job['version']:08x}",
job["bits"],
job["ntime"],
job["clean_jobs"]
])
except Exception as e:
print(f"Failed to send job to {addr}: {e}")
except Exception as e:
print(f"Job updater error: {e}")
def stats_updater(self):
"""Periodically update pool statistics"""
while self.running:
try:
time.sleep(60) # Update every minute
cursor = self.db.cursor()
# Pool hashrate is the sum of miners' last hashrates
cursor.execute('SELECT COALESCE(SUM(last_hashrate), 0) FROM miners')
self.pool_hashrate = cursor.fetchone()[0] or 0.0
# Sample for chart
cursor.execute('INSERT INTO hashrate_samples (hashrate) VALUES (?)', (self.pool_hashrate,))
self.db.commit()
print(f"📊 Pool Stats: {len(self.clients)} miners, {self.total_shares} shares, {self.pool_hashrate/1000:.2f} kH/s")
except Exception as e:
print(f"Stats updater error: {e}")
def start(self):
"""Start the mining pool server"""
try:
# Test RPC connection
blockchain_info = self.rpc_call("getblockchaininfo")
if not blockchain_info:
print("❌ Failed to connect to RinCoin node!")
return
print(f"✅ Connected to RinCoin node (block {blockchain_info.get('blocks', 'unknown')})")
# Start background threads
job_thread = threading.Thread(target=self.job_updater, daemon=True)
job_thread.start()
stats_thread = threading.Thread(target=self.stats_updater, daemon=True)
stats_thread.start()
# Start web interface in background
web_thread = threading.Thread(target=start_web_interface,
args=(self.db, '0.0.0.0', 8083,
self.rpc_host, self.rpc_port,
self.rpc_user, self.rpc_password),
daemon=True)
web_thread.start()
print(f"🌐 Web dashboard started on http://0.0.0.0:8083")
# Start Stratum server
server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
server_socket.bind((self.stratum_host, self.stratum_port))
server_socket.listen(10)
print(f"🚀 Mining pool listening on {self.stratum_host}:{self.stratum_port}")
print("Ready for multiple miners...")
print("")
print(f"💰 Pool address: {self.pool_address}")
print(f"💰 Pool fee: {self.pool_fee_percent}%")
print("")
print("Connect miners with:")
print(f"./cpuminer -a rinhash -o stratum+tcp://{self.stratum_host}:{self.stratum_port} -u username.workername -p x")
print("")
while self.running:
try:
client, addr = server_socket.accept()
client_thread = threading.Thread(target=self.handle_client, args=(client, addr), daemon=True)
client_thread.start()
except KeyboardInterrupt:
print("\n🛑 Shutting down pool...")
self.running = False
break
except Exception as e:
print(f"Server error: {e}")
except OSError as e:
if "Address already in use" in str(e):
print(f"❌ Port {self.stratum_port} is already in use!")
print("")
print("🔍 Check what's using the port:")
print(f"sudo netstat -tlnp | grep :{self.stratum_port}")
print("")
print("🛑 Kill existing process:")
print(f"sudo lsof -ti:{self.stratum_port} | xargs sudo kill -9")
print("")
print("🔄 Or use a different port by editing the script")
else:
print(f"Failed to start server: {e}")
except Exception as e:
print(f"Failed to start server: {e}")
finally:
print("Pool server stopped")
if __name__ == "__main__":
pool = RinCoinMiningPool()
pool.start()

1305
MINE/rin/stratum_proxy.py Normal file

File diff suppressed because it is too large Load Diff

View File

@@ -0,0 +1,75 @@
#!/bin/bash
# Test Reward Redistribution Logic
echo "=== Testing Reward Redistribution Logic ==="
echo ""
# Kill any existing processes
./MINE/rin/kill_stratum_proxy.sh
echo "🧪 Testing reward distribution scenarios:"
echo ""
echo "Scenario 1: All miners have valid addresses"
echo "Expected: Normal distribution"
echo ""
echo "Scenario 2: Some miners without addresses"
echo "Expected: Redistribution of their rewards to miners with addresses"
echo ""
echo "Scenario 3: All miners without addresses"
echo "Expected: All rewards go to pool"
echo ""
echo "🚀 Starting mining pool..."
./MINE/rin/start_mining_pool.sh &
POOL_PID=$!
echo ""
echo "⏳ Waiting for pool to start..."
sleep 5
echo ""
echo "🧪 Test 1: Miner with valid address"
echo "Expected: Gets full share of rewards"
timeout 5s ./cpuminer -a rinhash -o stratum+tcp://127.0.0.1:3333 -u rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q.worker1 -p x -t 1
echo ""
echo "🧪 Test 2: Miner without address"
echo "Expected: Contributes to difficulty but gets no direct rewards"
timeout 5s ./cpuminer -a rinhash -o stratum+tcp://127.0.0.1:3333 -u user.worker2 -p x -t 1
echo ""
echo "🧪 Test 3: Another miner with valid address"
echo "Expected: Gets base reward + redistribution from miner without address"
timeout 5s ./cpuminer -a rinhash -o stratum+tcp://127.0.0.1:3333 -u rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q.worker3 -p x -t 1
echo ""
echo "📊 Pool Log Analysis:"
echo "Look for these patterns in the logs above:"
echo "1. '💰 Miner rin1q...: X.XX RIN (difficulty)' - Base rewards"
echo "2. '⚠️ Miner without address: X difficulty -> X.XX RIN to pool' - Undistributed"
echo "3. '💰 Pool keeps X.XX RIN from miners without addresses' - Pool keeps rewards"
echo "4. '📊 Summary: X miners with addresses, Y without (rewards to pool)' - Final summary"
echo ""
echo "🧹 Cleaning up..."
kill $POOL_PID 2>/dev/null
./MINE/rin/kill_stratum_proxy.sh
echo ""
echo "📋 Reward Distribution Logic Summary:"
echo ""
echo "✅ Miners with valid RinCoin addresses:"
echo " - Get reward based on their difficulty"
echo " - Rewards sent directly to their addresses"
echo ""
echo "⚠️ Miners without addresses:"
echo " - Contribute to total difficulty"
echo " - Their reward share goes to pool address"
echo " - No direct rewards received"
echo ""
echo "💰 Pool fee: Always 1% of total block reward"
echo "💰 Pool bonus: Additional rewards from miners without addresses"

View File

@@ -0,0 +1,11 @@
#!/bin/bash
set -euo pipefail
SCRIPT_DIR=$(cd -- "$(dirname "${BASH_SOURCE[0]}")" && pwd)
WEB_WALLET_DIR="${SCRIPT_DIR}/web_wallet"
bash /mnt/shared/DEV/repos/d-popov.com/scripts/MINE/rin/send_rin.sh "$@"

View File

@@ -0,0 +1,150 @@
#!/usr/bin/env python3
from __future__ import annotations
import json
import os
import secrets
from functools import wraps
from flask import Flask, jsonify, request, send_from_directory
try:
from bitcoinrpc.authproxy import AuthServiceProxy
except ImportError: # pragma: no cover
raise SystemExit("Missing python-bitcoinrpc. Install with: pip install python-bitcoinrpc")
RIN_RPC_HOST = os.environ.get("RIN_RPC_HOST", "127.0.0.1")
RIN_RPC_PORT = int(os.environ.get("RIN_RPC_PORT", "9556"))
RIN_RPC_USER = os.environ.get("RIN_RPC_USER", "rinrpc")
RIN_RPC_PASSWORD = os.environ.get("RIN_RPC_PASSWORD", "745ce784d5d537fc06105a1b935b7657903cfc71a5fb3b90")
RIN_WALLET_NAME = os.environ.get("RIN_WALLET_NAME", "main")
API_TOKEN = os.environ.get("RIN_WEB_WALLET_TOKEN")
if not API_TOKEN:
API_TOKEN = secrets.token_urlsafe(32)
print("[web-wallet] No RIN_WEB_WALLET_TOKEN provided. Generated one for this session.")
print(f"[web-wallet] API token: {API_TOKEN}")
def create_rpc_client() -> AuthServiceProxy:
url = f"http://{RIN_RPC_USER}:{RIN_RPC_PASSWORD}@{RIN_RPC_HOST}:{RIN_RPC_PORT}/wallet/{RIN_WALLET_NAME}"
return AuthServiceProxy(url, timeout=15)
def require_token(view_func):
@wraps(view_func)
def wrapper(*args, **kwargs):
header = request.headers.get("Authorization", "")
if not header.startswith("Bearer "):
return jsonify({"error": "missing_token"}), 401
token = header.split(" ", 1)[1]
if token != API_TOKEN:
return jsonify({"error": "invalid_token"}), 403
return view_func(*args, **kwargs)
return wrapper
app = Flask(__name__, static_folder="static", static_url_path="")
def rpc_call(method: str, *params):
try:
rpc = create_rpc_client()
return rpc.__getattr__(method)(*params)
except Exception as exc: # noqa: BLE001
raise RuntimeError(str(exc)) from exc
@app.route("/api/session", methods=["GET"])
def session_info():
return jsonify({
"wallet": RIN_WALLET_NAME,
"host": RIN_RPC_HOST,
})
@app.route("/api/address", methods=["POST"])
@require_token
def create_address():
data = request.get_json(silent=True) or {}
label = data.get("label", "")
try:
address = rpc_call("getnewaddress", label)
return jsonify({"address": address})
except RuntimeError as exc:
return jsonify({"error": str(exc)}), 500
@app.route("/api/balance", methods=["GET"])
@require_token
def get_balance():
try:
info = rpc_call("getwalletinfo")
confirmed = info.get("balance", 0)
unconfirmed = info.get("unconfirmed_balance", 0)
immature = info.get("immature_balance", 0)
total = confirmed + unconfirmed + immature
return jsonify({
"confirmed": confirmed,
"unconfirmed": unconfirmed,
"immature": immature,
"total": total,
})
except RuntimeError as exc:
return jsonify({"error": str(exc)}), 500
@app.route("/api/send", methods=["POST"])
@require_token
def send():
payload = request.get_json(force=True)
address = payload.get("address")
amount = payload.get("amount")
subtract_fee = bool(payload.get("subtractFee", True))
if not address or not isinstance(amount, (int, float)):
return jsonify({"error": "invalid_request"}), 400
if amount <= 0:
return jsonify({"error": "amount_must_be_positive"}), 400
try:
txid = rpc_call("sendtoaddress", address, float(amount), "", "", subtract_fee)
return jsonify({"txid": txid})
except RuntimeError as exc:
status = 400 if "Invalid RinCoin address" in str(exc) else 500
return jsonify({"error": str(exc)}), status
@app.route("/api/transactions", methods=["GET"])
@require_token
def list_transactions():
count = int(request.args.get("count", 10))
try:
txs = rpc_call("listtransactions", "*", count, 0, True)
return jsonify({"transactions": txs})
except RuntimeError as exc:
return jsonify({"error": str(exc)}), 500
@app.route("/")
def root():
return send_from_directory(app.static_folder, "index.html")
@app.route("/<path:path>")
def static_proxy(path):
return send_from_directory(app.static_folder, path)
def main():
port = int(os.environ.get("RIN_WEB_WALLET_PORT", "8787"))
app.run(host="127.0.0.1", port=port, debug=False)
if __name__ == "__main__":
main()

View File

@@ -0,0 +1,24 @@
#!/bin/bash
set -euo pipefail
SCRIPT_DIR=$(cd -- "$(dirname "${BASH_SOURCE[0]}")" && pwd)
CONTAINER="rincoin-node"
if ! sudo docker ps --format '{{.Names}}' | grep -q "^${CONTAINER}$"; then
echo "Error: ${CONTAINER} container is not running. Start it with 'sudo docker start ${CONTAINER}'."
exit 1
fi
if ! command -v flask >/dev/null 2>&1; then
echo "Missing Flask. Install with 'pip install flask python-bitcoinrpc'."
exit 1
fi
echo "Starting RinCoin web wallet on http://127.0.0.1:8787"
export FLASK_APP="${SCRIPT_DIR}/server.py"
export FLASK_ENV=production
export PYTHONPATH="${SCRIPT_DIR}"
flask run --host 127.0.0.1 --port 8787

View File

@@ -0,0 +1,318 @@
<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<title>RinCoin Web Wallet</title>
<style>
body {
font-family: Arial, sans-serif;
margin: 0;
padding: 0;
background: #0b1a28;
color: #f5f8fc;
display: flex;
min-height: 100vh;
}
.sidebar {
width: 260px;
background: #122c43;
padding: 24px;
box-sizing: border-box;
display: flex;
flex-direction: column;
gap: 24px;
}
.content {
flex: 1;
padding: 32px;
box-sizing: border-box;
}
h1 {
margin: 0 0 16px;
font-size: 24px;
}
h2 {
margin-top: 32px;
margin-bottom: 16px;
font-size: 18px;
border-bottom: 1px solid rgba(255, 255, 255, 0.1);
padding-bottom: 8px;
}
label {
display: block;
margin-bottom: 8px;
font-size: 14px;
text-transform: uppercase;
letter-spacing: 0.08em;
color: rgba(255, 255, 255, 0.7);
}
input {
width: 100%;
padding: 10px 12px;
border-radius: 6px;
border: 1px solid rgba(255, 255, 255, 0.2);
background: rgba(9, 19, 30, 0.8);
color: #fff;
margin-bottom: 16px;
box-sizing: border-box;
}
button {
background: #29b6f6;
border: none;
padding: 12px 16px;
border-radius: 6px;
color: #04121f;
font-weight: 600;
cursor: pointer;
transition: background 0.2s ease;
}
button:hover {
background: #4fc3f7;
}
.card {
background: rgba(9, 19, 30, 0.8);
padding: 24px;
border-radius: 12px;
margin-bottom: 24px;
box-shadow: 0 12px 32px rgba(0, 0, 0, 0.3);
}
.balance {
font-size: 36px;
font-weight: 600;
}
.muted {
color: rgba(255, 255, 255, 0.6);
font-size: 14px;
}
.transaction-list {
margin: 0;
padding: 0;
list-style: none;
}
.transaction-list li {
padding: 14px 0;
border-bottom: 1px solid rgba(255, 255, 255, 0.1);
}
.address-box {
word-break: break-all;
background: rgba(41, 182, 246, 0.15);
border-radius: 6px;
padding: 12px;
margin: 12px 0;
}
.status {
border-radius: 6px;
margin-top: 16px;
padding: 12px 14px;
}
.status.success {
background: rgba(76, 175, 80, 0.2);
color: #c8e6c9;
}
.status.error {
background: rgba(244, 67, 54, 0.2);
color: #ffcdd2;
}
.token-display {
background: rgba(9, 19, 30, 0.8);
padding: 12px;
border-radius: 6px;
font-size: 13px;
word-break: break-all;
}
.sidebar h2 {
margin-top: 0;
}
.sidebar button {
width: 100%;
}
@media (max-width: 900px) {
body {
flex-direction: column;
}
.sidebar {
width: 100%;
flex-direction: row;
gap: 16px;
flex-wrap: wrap;
}
.sidebar section {
flex: 1 1 200px;
}
.content {
padding: 24px;
}
}
</style>
</head>
<body>
<aside class="sidebar">
<section>
<h2>Session</h2>
<div class="token-display" id="tokenDisplay">Loading…</div>
</section>
<section>
<button id="refreshButton">Refresh Data</button>
</section>
<section>
<button id="generateButton">Generate Address</button>
</section>
</aside>
<main class="content">
<h1>RinCoin Wallet</h1>
<div class="card">
<div class="muted">Confirmed Balance</div>
<div class="balance" id="confirmedBalance"></div>
<div class="muted" id="totalBalance">Total: —</div>
</div>
<div class="card">
<h2>Send RinCoin</h2>
<label for="sendAddress">Recipient Address</label>
<input id="sendAddress" type="text" placeholder="rin1..." />
<label for="sendAmount">Amount (RIN)</label>
<input id="sendAmount" type="number" step="0.00000001" min="0" />
<button id="sendButton">Send</button>
<div id="sendStatus" class="status" style="display:none;"></div>
</div>
<div class="card" id="generatedAddressCard" style="display:none;">
<h2>New Address</h2>
<div class="address-box" id="generatedAddress"></div>
</div>
<div class="card">
<h2>Recent Activity</h2>
<ul class="transaction-list" id="txList"></ul>
</div>
</main>
<script>
const tokenDisplay = document.getElementById('tokenDisplay');
const confirmedBalance = document.getElementById('confirmedBalance');
const totalBalance = document.getElementById('totalBalance');
const txList = document.getElementById('txList');
const sendStatus = document.getElementById('sendStatus');
const generatedAddressCard = document.getElementById('generatedAddressCard');
const generatedAddress = document.getElementById('generatedAddress');
let apiToken = localStorage.getItem('rinWebWalletToken');
async function fetchSession() {
const res = await fetch('/api/session');
const data = await res.json();
if (!apiToken) {
apiToken = data.token;
localStorage.setItem('rinWebWalletToken', apiToken);
}
tokenDisplay.textContent = `Wallet: ${data.wallet}`;
}
function authHeaders() {
return { Authorization: `Bearer ${apiToken}`, 'Content-Type': 'application/json' };
}
async function fetchBalances() {
const res = await fetch('/api/balance', { headers: authHeaders() });
const data = await res.json();
if (data.error) {
confirmedBalance.textContent = 'Error';
totalBalance.textContent = data.error;
return;
}
confirmedBalance.textContent = `${Number(data.confirmed).toFixed(8)} RIN`;
totalBalance.textContent = `Total: ${Number(data.total).toFixed(8)} RIN`;
}
function renderTx(tx) {
const li = document.createElement('li');
const amount = Number(tx.amount).toFixed(8);
const type = tx.category === 'send' ? 'Sent' : 'Received';
li.innerHTML = `
<div><strong>${type}</strong> ${amount} RIN</div>
<div class="muted">${tx.time ? new Date(tx.time * 1000).toLocaleString() : ''}</div>
<div class="muted">${tx.txid}</div>
`;
return li;
}
async function fetchTransactions() {
const res = await fetch('/api/transactions', { headers: authHeaders() });
const data = await res.json();
txList.innerHTML = '';
if (data.error) {
txList.innerHTML = `<li class="muted">${data.error}</li>`;
return;
}
if (!data.transactions.length) {
txList.innerHTML = '<li class="muted">No transactions yet.</li>';
return;
}
data.transactions.forEach((tx) => txList.appendChild(renderTx(tx)));
}
async function generateAddress() {
const res = await fetch('/api/address', {
method: 'POST',
headers: authHeaders(),
body: JSON.stringify({}),
});
const data = await res.json();
if (data.error) {
generatedAddressCard.style.display = 'block';
generatedAddress.textContent = `Error: ${data.error}`;
return;
}
generatedAddressCard.style.display = 'block';
generatedAddress.textContent = data.address;
}
async function sendCoins() {
const address = document.getElementById('sendAddress').value.trim();
const amount = Number(document.getElementById('sendAmount').value);
if (!address || !amount) {
sendStatus.textContent = 'Destination and amount are required.';
sendStatus.className = 'status error';
sendStatus.style.display = 'block';
return;
}
const res = await fetch('/api/send', {
method: 'POST',
headers: authHeaders(),
body: JSON.stringify({ address, amount, subtractFee: true }),
});
const data = await res.json();
if (data.error) {
sendStatus.textContent = data.error;
sendStatus.className = 'status error';
sendStatus.style.display = 'block';
return;
}
sendStatus.textContent = `Transaction broadcast. TXID: ${data.txid}`;
sendStatus.className = 'status success';
sendStatus.style.display = 'block';
await refresh();
}
async function refresh() {
await Promise.all([fetchBalances(), fetchTransactions()]);
}
document.getElementById('refreshButton').addEventListener('click', refresh);
document.getElementById('generateButton').addEventListener('click', generateAddress);
document.getElementById('sendButton').addEventListener('click', sendCoins);
(async () => {
await fetchSession();
await refresh();
})();
</script>
</body>
</html>

1
MINE/stratum Submodule

Submodule MINE/stratum added at 829075155a

54
MINE/zergBench.sh Normal file
View File

@@ -0,0 +1,54 @@
#!/bin/bash
# Test top Zergpool algorithms - save as test_zergpool.sh
BTC_WALLET="bc1qjn4m6rmrveuxhk02a5qhe4r6kdcsvvt3vhdn9j"
echo "=== Testing Top Zergpool Algorithms ==="
echo "Wallet: $BTC_WALLET"
echo "======================================"
# Top algorithms by profitability
declare -A algos=(
["rinhash"]="rinhash.mine.zergpool.com:7148 c=RIN"
["kawpow"]="kawpow.mine.zergpool.com:3638 c=BTC"
["evrprogpow"]="evrprogpow.mine.zergpool.com:3002 c=BTC"
["equihash125_4"]="equihash.mine.zergpool.com:2142 c=BTC"
["karlsenhashv2"]="karlsenhashv2.mine.zergpool.com:3200 c=BTC"
)
for algo in "${!algos[@]}"; do
echo ""
echo "Testing: $algo"
echo "------------------------"
# Parse config
read -r server pass <<< "${algos[$algo]}"
echo "Server: $server"
echo "Testing for 30 seconds..."
sudo docker exec -it amdopencl timeout 35s bash -c "/mnt/dl/gminer/miner --algo $algo --server $server --user '$BTC_WALLET' --pass '$pass'"
result=$?
if [ $result -eq 0 ]; then
echo "$algo: SUCCESS"
elif [ $result -eq 124 ]; then
echo "⏱️ $algo: TIMEOUT (likely working)"
else
echo "$algo: FAILED - trying alternative miner..."
# Try lolMiner for failed algorithms
sudo docker exec -it amdopencl timeout 35s bash -c "/mnt/dl/lolMiner_v1.88_Lin64/lolMiner --algo ${algo^^} --pool $server --user '$BTC_WALLET' --pass '$pass'" 2>/dev/null
if [ $? -eq 124 ]; then
echo "⏱️ $algo: WORKS with lolMiner"
else
echo "$algo: Not supported"
fi
fi
sleep 3
done
echo ""
echo "=== Zergpool Testing Complete ==="

1
build.log Normal file
View File

@@ -0,0 +1 @@
make: *** No targets specified and no makefile found. Stop.

10
mining_log.txt Normal file
View File

@@ -0,0 +1,10 @@
================================================================================
RinCoin Mining Log
================================================================================
Started: 2025-09-23 11:27:51
Target Address: rin1qahvvv9d5f3443wtckeqavwp9950wacxfmwv20q
Stratum: 0.0.0.0:3334
RPC: 127.0.0.1:9556
================================================================================
[2025-09-23 11:27:51] 💰 Wallet Balance: 25.00000000 RIN

View File

@@ -0,0 +1,4 @@
https://www.youtube.com/watch?v=wCBLMXgk3No
https://github.com/kyuz0/amd-strix-halo-toolboxes?tab=readme-ov-file#211-toolbox-refresh-script-automatic-updates
https://hub.docker.com/r/kyuz0/amd-strix-halo-toolboxes/tags

View File

@@ -0,0 +1,100 @@
version: '3.8'
services:
amd-strix-halo-llama-rocm:
image: kyuz0/amd-strix-halo-toolboxes:rocm-7rc-rocwmma
container_name: amd-strix-halo-llama-rocm
restart: unless-stopped
privileged: true
devices:
- /dev/kfd:/dev/kfd
- /dev/dri:/dev/dri
group_add:
- video
volumes:
- ./models:/models
- ./data:/data
- /home/${USER}:/home/${USER}:rslave
- /home/db/Downloads/:/mnt/dl
environment:
- DISPLAY=${DISPLAY}
- NVIDIA_VISIBLE_DEVICES=all
- NVIDIA_DRIVER_CAPABILITIES=all
ports:
- "8080:8080" # For web UI if available
working_dir: /models
command: /bin/bash
stdin_open: true
tty: true
# Alternative Vulkan backend
amd-strix-halo-llama-vulkan-radv:
image: kyuz0/amd-strix-halo-toolboxes:vulkan-radv
container_name: amd-strix-halo-llama-vulkan-radv
restart: unless-stopped
privileged: true
devices:
- /dev/dri:/dev/dri
group_add:
- video
volumes:
- ./models:/models
- ./data:/data
- /home/${USER}:/home/${USER}:rslave
- /home/db/Downloads/:/mnt/dl
environment:
- DISPLAY=${DISPLAY}
ports:
- "8081:8080" # Different port to avoid conflicts
working_dir: /models
command: /bin/bash
stdin_open: true
tty: true
# Alternative Vulkan AMDVLK backend
amd-strix-halo-llama-vulkan-amdvlk:
image: kyuz0/amd-strix-halo-toolboxes:vulkan-amdvlk
container_name: amd-strix-halo-llama-vulkan-amdvlk
restart: unless-stopped
privileged: true
devices:
- /dev/dri:/dev/dri
group_add:
- video
volumes:
- ./models:/models
- ./data:/data
- /home/${USER}:/home/${USER}:rslave
# - /home/db/Downloads/xmrig-6.21.0:/mnt/xmrig
- /home/db/Downloads/:/mnt/dl
environment:
- DISPLAY=${DISPLAY}
ports:
- "8082:8080" # Different port to avoid conflicts
working_dir: /models
command: /bin/bash
stdin_open: true
tty: true
amdopencl:
image: pbsprotest/amdopencl:24
container_name: amdopencl
devices:
- /dev/dri
- /dev/kfd
volumes:
- ./workspace:/workspace
# - /home/db/Downloads/xmrig-6.21.0:/mnt/xmrig
- /home/db/Downloads/:/mnt/dl
stdin_open: true
tty: true
volumes:
models:
driver: local
data:
driver: local
networks:
default:
name: amd-strix-halo-network

View File

@@ -0,0 +1,109 @@
# AMD Strix Halo Toolboxes Docker Compose
This Docker Compose setup provides pre-built containers for running LLMs on AMD Ryzen AI Max "Strix Halo" integrated GPUs.
## Prerequisites
- AMD Ryzen AI Max "Strix Halo" system (e.g., Ryzen AI MAX+ 395)
- Docker and Docker Compose installed
- At least 128GB RAM recommended for larger models
- Proper kernel configuration for unified memory
## Kernel Configuration
Add these boot parameters to `/etc/default/grub`:
```bash
amd_iommu=off amdgpu.gttsize=131072 ttm.pages_limit=33554432
```
Then apply:
```bash
sudo grub2-mkconfig -o /boot/grub2/grub.cfg
sudo reboot
```
## Usage
### Start all services
```bash
docker-compose up -d
```
### Start specific backend
```bash
# ROCm backend (best for prompt processing)
docker-compose up -d amd-strix-halo-llama-rocm
# Vulkan RADV backend (fastest token generation)
docker-compose up -d amd-strix-halo-llama-vulkan-radv
# Vulkan AMDVLK backend
docker-compose up -d amd-strix-halo-llama-vulkan-amdvlk
```
### Access containers
```bash
# Enter ROCm container
docker exec -it amd-strix-halo-llama-rocm bash
# Enter Vulkan RADV container
docker exec -it amd-strix-halo-llama-vulkan-radv bash
# Enter Vulkan AMDVLK container
docker exec -it amd-strix-halo-llama-vulkan-amdvlk bash
```
## Directory Structure
```
amd-strix-halo-toolboxes/
├── models/ # Mount point for GGUF models
├── data/ # Mount point for data
└── amd-strix-halo-toolboxes.yml
```
## Download Models
Inside the container, download GGUF models:
```bash
# Example: Download Llama-2-7B
wget https://huggingface.co/TheBloke/Llama-2-7B-Chat-GGUF/resolve/main/llama-2-7b-chat.Q4_K_M.gguf
# Run the model
./llama.cpp/main -m llama-2-7b-chat.Q4_K_M.gguf -n 128 --repeat_penalty 1.1
```
## Backend Performance
Based on benchmarks:
- **ROCm 6.4.3 + ROCWMMA (hipBLASLt)**: Best for prompt processing
- **Vulkan RADV**: Fastest for token generation
- **Vulkan AMDVLK**: Good balance
## Memory Planning
Use the VRAM estimator inside containers:
```bash
python3 gguf-vram-estimator.py your-model.gguf --contexts 4096 32768 1048576
```
## Ports
- ROCm backend: `8080`
- Vulkan RADV backend: `8081`
- Vulkan AMDVLK backend: `8082`
## Troubleshooting
1. **Permission issues**: Ensure your user is in the `video` group
2. **GPU not detected**: Check kernel parameters and reboot
3. **Out of memory**: Use the VRAM estimator to plan model sizes
## References
- [Original Repository](https://github.com/kyuz0/amd-strix-halo-toolboxes)
- [Strix Halo Hardware Database](https://strixhalo-homelab.d7.wtf/)

View File

@@ -0,0 +1,100 @@
version: '3.8'
services:
amd-strix-halo-llama-rocm:
image: kyuz0/amd-strix-halo-toolboxes:rocm-7rc-rocwmma
container_name: amd-strix-halo-llama-rocm
restart: unless-stopped
privileged: true
devices:
- /dev/kfd:/dev/kfd
- /dev/dri:/dev/dri
group_add:
- video
volumes:
- ./models:/models
- ./data:/data
- /home/${USER}:/home/${USER}:rslave
- /home/db/Downloads/:/mnt/dl
environment:
- DISPLAY=${DISPLAY}
- NVIDIA_VISIBLE_DEVICES=all
- NVIDIA_DRIVER_CAPABILITIES=all
ports:
- "8080:8080" # For web UI if available
working_dir: /models
command: /bin/bash
stdin_open: true
tty: true
# Alternative Vulkan backend
amd-strix-halo-llama-vulkan-radv:
image: kyuz0/amd-strix-halo-toolboxes:vulkan-radv
container_name: amd-strix-halo-llama-vulkan-radv
restart: unless-stopped
privileged: true
devices:
- /dev/dri:/dev/dri
group_add:
- video
volumes:
- ./models:/models
- ./data:/data
- /home/${USER}:/home/${USER}:rslave
- /home/db/Downloads/:/mnt/dl
environment:
- DISPLAY=${DISPLAY}
ports:
- "8081:8080" # Different port to avoid conflicts
working_dir: /models
command: /bin/bash
stdin_open: true
tty: true
# Alternative Vulkan AMDVLK backend
amd-strix-halo-llama-vulkan-amdvlk:
image: kyuz0/amd-strix-halo-toolboxes:vulkan-amdvlk
container_name: amd-strix-halo-llama-vulkan-amdvlk
restart: unless-stopped
privileged: true
devices:
- /dev/dri:/dev/dri
group_add:
- video
volumes:
- ./models:/models
- ./data:/data
- /home/${USER}:/home/${USER}:rslave
# - /home/db/Downloads/xmrig-6.21.0:/mnt/xmrig
- /home/db/Downloads/:/mnt/dl
environment:
- DISPLAY=${DISPLAY}
ports:
- "8082:8080" # Different port to avoid conflicts
working_dir: /models
command: /bin/bash
stdin_open: true
tty: true
amdopencl:
image: pbsprotest/amdopencl:24
container_name: amdopencl
devices:
- /dev/dri
- /dev/kfd
volumes:
- ./workspace:/workspace
# - /home/db/Downloads/xmrig-6.21.0:/mnt/xmrig
- /home/db/Downloads/:/mnt/dl
stdin_open: true
tty: true
volumes:
models:
driver: local
data:
driver: local
networks:
default:
name: amd-strix-halo-network

View File

@@ -0,0 +1,40 @@
#!/bin/bash
# AMD Strix Halo Toolboxes Startup Script
echo "Starting AMD Strix Halo Toolboxes..."
# Check if Docker is running
if ! docker info > /dev/null 2>&1; then
echo "Error: Docker is not running. Please start Docker first."
exit 1
fi
# Check if we're in the right directory
if [ ! -f "amd-strix-halo-toolboxes.yml" ]; then
echo "Error: amd-strix-halo-toolboxes.yml not found. Please run this script from the amd-strix-halo-toolboxes directory."
exit 1
fi
# Pull the latest images
echo "Pulling latest images..."
docker-compose pull
# Start the services
echo "Starting services..."
docker-compose up -d
echo "Services started successfully!"
echo ""
echo "Available containers:"
echo "- amd-strix-halo-llama-rocm (ROCm backend)"
echo "- amd-strix-halo-llama-vulkan-radv (Vulkan RADV backend)"
echo "- amd-strix-halo-llama-vulkan-amdvlk (Vulkan AMDVLK backend)"
echo ""
echo "To access a container:"
echo "docker exec -it amd-strix-halo-llama-rocm bash"
echo ""
echo "To view logs:"
echo "docker-compose logs -f"

View File

@@ -0,0 +1,103 @@
version: "3.8"
services:
ubuntu-vnc-xfce-firefox-g3: # zelenakravapase
deploy:
replicas: 1
#https://hub.docker.com/r/accetto/ubuntu-vnc-xfce-firefox-g3
#https://github.com/accetto/ubuntu-vnc-xfce-g3/tree/master/docker/xfce-firefox
image: accetto/ubuntu-vnc-xfce-firefox-g3
container_name: desktop-ubuntu
restart: unless-stopped
#network_mode: "host"
ports:
- "6080:80"
- 6901:6901
environment:
- VNC_PW=tixooo! #zelenakravapase
- VNC_RESOLUTION=1900x1000
#1024x768 1600 x 900 HD+ 1920x1080
# - USER_UID=1000
# - USER_GID=1000
devices:
- "/dev/binder:/dev/binder"
- "/dev/ashmem:/dev/ashmem"
privileged: true
cap_add:
- NET_ADMIN
user: "0:0"
volumes:
- /mnt/apps/docker_volumes/DESK/firefox/:/headless/data
- /mnt/apps/docker_volumes/DESK/firefox/desktop:/home/headless/Desktop
- /mnt:/mnt
#- /dev/ppp:/dev/ppp # for VPN
#network_mode: "host"
winehq:
deploy:
replicas: 0
image: ich777/winehq-novnc-baseimage
ports:
- "8089:8080"
- "5909:5900"
volumes:
- /mnt/apps/docker_volumes/DESK/wine:/wine-data
environment:
- TZ=Europe/Sofia
restart: unless-stopped
android-container:
deploy:
replicas: 0
image: budtmo/docker-android:emulator_14.0 #budtmo/docker-android
container_name: android-container
ports:
- "6081:6080"
- "5554:5554"
- "5555:5555"
environment:
EMULATOR_DEVICE: Samsung Galaxy S10
WEB_VNC: "true"
devices:
- "/dev/kvm:/dev/kvm"
volumes:
- "/mnt/storage/DEV/workspace/repos/git.d-popov.com/ai-kevin/agent-mobile/artimobile:/home/androidusr/tmp"
- "/mnt/storage/DEV/workspace/repos/git.d-popov.com/ai-kevin/agent-mobile/jdk:/home/androidusr/download"
#- /mnt/apps/docker_volumes/android_emulator/opt:/opt/android
# - /mnt/apps/docker_volumes/android_emulator/jdk:/home/androidusr/jdk
#- "/mnt/apps/docker_volumes/android_emulator/data:/home/androidusr/.android"
#- /mnt/apps/docker_volumes/android_emulator/.android:/home/androidusr/.android
#- /mnt/apps/docker_volumes/android_emulator/android_emulator:/home/androidusr/emulator
working_dir: /home/androidusr/tmp
command: /bin/bash -c "cd /home/androidusr/tmp && npx react-native run-android"
# https://github.com/dockur/windows
windows:
image: dockurr/windows
container_name: windows
devices:
- /dev/kvm
cap_add:
- NET_ADMIN
ports:
- 8006:8006
- 3389:3389/tcp
- 3389:3389/udp
stop_grace_period: 2m
restart: on-failure
environment:
VERSION: "win11"
RAM_SIZE: "6G"
CPU_CORES: "6"
DISK_SIZE: "64G"
volumes:
- /mnt/apps/docker_volumes/win:/storage
- /mnt/storage/:/data
- /mnt/apps/docker_volumes/DESK/firefox/desktop:/data2
deploy:
resources:
reservations:
devices:
- driver: nvidia
count: all
capabilities: [gpu]

View File

@@ -0,0 +1,22 @@
services:
windows:
image: dockurr/windows # https://github.com/dockur/windows
container_name: windows
environment:
VERSION: "11"
devices:
- /dev/kvm
- /dev/net/tun
cap_add:
- NET_ADMIN
ports:
- 8006:8006
- 3389:3389/tcp
- 3389:3389/udp
volumes:
#- /dev/nvme0n1p7:/disk1 # blind mount - not working for now
- /mnt/data/docker_vol/windows:/storage # storage (img file)location
#- /mnt/data/:/data
restart: always
stop_grace_period: 2m

1
stratum_proxy.log Normal file
View File

@@ -0,0 +1 @@
nohup: ignoring input