Compare commits
18 Commits
48f8b31e13
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
| 7d6f7b7712 | |||
| 342f5a3d3e | |||
| a0303fe01c | |||
| 80132740bd | |||
| 60cabb03d4 | |||
| 7f2fdf250a | |||
| bf9dbdf870 | |||
| ad77348a8b | |||
| 96e495ba11 | |||
| 3540fb9058 | |||
| e23c91c1b1 | |||
| 08693746a6 | |||
| 97320dade3 | |||
| 3098bbfa7d | |||
| 9dda40ab87 | |||
| bbded38ea0 | |||
| 4faea99426 | |||
| 9dd990ec9c |
47
.gitignore
vendored
47
.gitignore
vendored
@@ -1,3 +1,7 @@
|
|||||||
|
# ===================================
|
||||||
|
# Python Scanner & Database
|
||||||
|
# ===================================
|
||||||
|
|
||||||
# Virtual Environment
|
# Virtual Environment
|
||||||
.venv/
|
.venv/
|
||||||
venv/
|
venv/
|
||||||
@@ -11,19 +15,58 @@ __pycache__/
|
|||||||
*.pyo
|
*.pyo
|
||||||
*.pyd
|
*.pyd
|
||||||
*$py.class
|
*$py.class
|
||||||
*.so
|
|
||||||
.Python
|
.Python
|
||||||
*.egg-info/
|
*.egg-info/
|
||||||
dist/
|
dist/
|
||||||
build/
|
build/
|
||||||
|
|
||||||
# IDE
|
*.html
|
||||||
|
|
||||||
|
# ===================================
|
||||||
|
# C++ Bruteforce
|
||||||
|
# ===================================
|
||||||
|
|
||||||
|
# Eseguibili compilati
|
||||||
|
bruteforce/p2pk_bruteforce
|
||||||
|
bruteforce/p2pk_bruteforce_debug
|
||||||
|
|
||||||
|
# File oggetto e compilazione
|
||||||
|
*.o
|
||||||
|
*.a
|
||||||
|
*.so
|
||||||
|
*.gcda
|
||||||
|
*.gcno
|
||||||
|
*.gch
|
||||||
|
|
||||||
|
# File generati dal bruteforce
|
||||||
|
bruteforce/found_keys.txt
|
||||||
|
bruteforce/progress.csv
|
||||||
|
bruteforce/target_keys.txt
|
||||||
|
|
||||||
|
# libsecp256k1 compilata localmente
|
||||||
|
bruteforce/secp256k1_build/
|
||||||
|
bruteforce/secp256k1/
|
||||||
|
|
||||||
|
# ===================================
|
||||||
|
# IDE & Editor
|
||||||
|
# ===================================
|
||||||
|
|
||||||
.vscode/
|
.vscode/
|
||||||
.idea/
|
.idea/
|
||||||
*.swp
|
*.swp
|
||||||
*.swo
|
*.swo
|
||||||
*~
|
*~
|
||||||
|
|
||||||
|
# ===================================
|
||||||
# OS
|
# OS
|
||||||
|
# ===================================
|
||||||
|
|
||||||
.DS_Store
|
.DS_Store
|
||||||
Thumbs.db
|
Thumbs.db
|
||||||
|
|
||||||
|
# ===================================
|
||||||
|
# Logs & Temp
|
||||||
|
# ===================================
|
||||||
|
|
||||||
|
*.log
|
||||||
|
*.tmp
|
||||||
518
README.md
518
README.md
@@ -1,313 +1,303 @@
|
|||||||
# Bitcoin P2PK Scanner
|
# Bitcoin P2PK Research Suite
|
||||||
|
|
||||||
Scanner di transazioni Bitcoin Pay-to-Public-Key (P2PK) per scopi educativi e di ricerca sulla blockchain.
|
Suite completa di strumenti per ricerca educativa su transazioni Bitcoin Pay-to-Public-Key (P2PK) e sicurezza crittografica.
|
||||||
|
|
||||||
## ⚠️ Disclaimer
|
## ⚠️ Disclaimer Importante
|
||||||
|
|
||||||
**Questo progetto è SOLO per scopi educativi e di ricerca.**
|
**Questo progetto è ESCLUSIVAMENTE per scopi educativi e di ricerca.**
|
||||||
- Non utilizzare per attività illegali
|
|
||||||
- Rispetta sempre i termini di servizio delle API utilizzate
|
|
||||||
- Lo scopo è studiare la struttura della blockchain Bitcoin
|
|
||||||
|
|
||||||
## Caratteristiche
|
- Non utilizzare per attività illegali o tentativi di accesso non autorizzato
|
||||||
|
- La ricerca di chiavi private altrui è illegale
|
||||||
|
- Lo scopo è dimostrare la sicurezza crittografica di Bitcoin (ECDSA secp256k1)
|
||||||
|
- La probabilità di trovare chiavi per bruteforce è praticamente zero (1 su 2^256)
|
||||||
|
|
||||||
- Scansione blocchi Bitcoin per identificare transazioni P2PK
|
## Panoramica
|
||||||
- Salvataggio dati in database SQLite
|
|
||||||
- Verifica UTXO (controllo se le chiavi hanno ancora saldo)
|
|
||||||
- Report HTML interattivo con ricerca e statistiche
|
|
||||||
- Esportazione dati in CSV
|
|
||||||
- Scansione incrementale (riprende da dove si è fermato)
|
|
||||||
- API mempool.space con rate limiting integrato
|
|
||||||
|
|
||||||
## Installazione
|
Questo repository contiene due componenti principali:
|
||||||
|
|
||||||
### 1. Clona il repository
|
1. **Scanner Python**: Analizza la blockchain Bitcoin per identificare e catalogare transazioni P2PK
|
||||||
|
2. **Bruteforce C++**: Dimostra l'impossibilità pratica di trovare chiavi private per bruteforce
|
||||||
|
|
||||||
### 2. Crea l'ambiente virtuale Python
|
## Quick Start
|
||||||
|
|
||||||
|
### 1. Scanner Python - Analisi Blockchain
|
||||||
|
|
||||||
|
Scansiona la blockchain Bitcoin per trovare transazioni P2PK e verificare UTXO.
|
||||||
|
|
||||||
**Linux/macOS:**
|
|
||||||
```bash
|
```bash
|
||||||
|
# Installa dipendenze Python
|
||||||
python3 -m venv .venv
|
python3 -m venv .venv
|
||||||
source .venv/bin/activate
|
source .venv/bin/activate # Linux/macOS
|
||||||
```
|
|
||||||
|
|
||||||
**Windows:**
|
|
||||||
```cmd
|
|
||||||
python -m venv .venv
|
|
||||||
.venv\Scripts\activate
|
|
||||||
```
|
|
||||||
|
|
||||||
### 3. Installa le dipendenze
|
|
||||||
|
|
||||||
```bash
|
|
||||||
pip install -r requirements.txt
|
pip install -r requirements.txt
|
||||||
|
|
||||||
|
# Esegui lo scanner
|
||||||
|
cd databases
|
||||||
|
python3 scan_blockchain.py
|
||||||
|
|
||||||
|
# Genera report HTML
|
||||||
|
python3 view_db.py
|
||||||
```
|
```
|
||||||
|
|
||||||
## Utilizzo
|
**Output**:
|
||||||
|
- Database SQLite con tutte le transazioni P2PK trovate
|
||||||
|
- Report HTML interattivo con statistiche e ricerca
|
||||||
|
- Esportazione CSV dei dati
|
||||||
|
|
||||||
### Scanner Principale
|
### 2. Bruteforce C++ - Dimostrazione Sicurezza
|
||||||
|
|
||||||
Avvia lo scanner interattivo:
|
Dimostra l'impossibilità pratica del bruteforce su chiavi Bitcoin.
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
python main.py
|
# Installa dipendenze C++
|
||||||
|
cd bruteforce
|
||||||
|
make install-deps
|
||||||
|
|
||||||
|
# Compila con ottimizzazioni massime
|
||||||
|
make
|
||||||
|
|
||||||
|
# Estrai chiavi P2PK non spese dal database
|
||||||
|
python3 extract_p2pk_utxo.py
|
||||||
|
|
||||||
|
# Esegui il bruteforce (dimostrazione)
|
||||||
|
./p2pk_bruteforce
|
||||||
```
|
```
|
||||||
|
|
||||||
Lo script ti chiederà:
|
**Performance**: ~50-100K keys/sec su hardware comune
|
||||||
- **Blocco di inizio**: da quale blocco iniziare (default: ultimo blocco + 1)
|
|
||||||
- **Blocco finale**: fino a quale blocco scansionare
|
|
||||||
- **Delay**: tempo tra richieste API in secondi (default: 1.0s, minimo: 0.1s)
|
|
||||||
|
|
||||||
Esempio di sessione:
|
## Documentazione Dettagliata
|
||||||
```
|
|
||||||
📊 Ultimo blocco scannerizzato: 0
|
|
||||||
💡 I primi blocchi di Bitcoin (1-10000) contengono molti P2PK
|
|
||||||
|
|
||||||
📍 Blocco di inizio scansione (default: 1): 1
|
Ogni componente ha la sua documentazione specifica:
|
||||||
📍 Blocco finale scansione (default: 1000): 100
|
|
||||||
⏱️ Delay tra richieste in secondi (default: 1.0): 1.0
|
|
||||||
|
|
||||||
🔧 Configurazione:
|
- **[Scanner Python](databases/README.md)**: Guida completa scanner blockchain
|
||||||
Primo blocco: 1
|
- **[Bruteforce C++](bruteforce/README.md)**: Guida completa programma bruteforce
|
||||||
Ultimo blocco: 100
|
|
||||||
Totale blocchi: 100
|
|
||||||
Delay richieste: 1.0s
|
|
||||||
Tempo stimato: ~1.7 minuti
|
|
||||||
|
|
||||||
▶️ Avviare la scansione? (s/n): s
|
## Componenti
|
||||||
```
|
|
||||||
|
|
||||||
### Visualizzatore Database
|
### Scanner Python (Database)
|
||||||
|
|
||||||
Genera un report HTML interattivo:
|
**Scopo**: Analizzare la blockchain Bitcoin per identificare transazioni P2PK storiche.
|
||||||
|
|
||||||
|
**Caratteristiche**:
|
||||||
|
- Scansione blocchi Bitcoin via API mempool.space
|
||||||
|
- Salvataggio in database SQLite normalizzato
|
||||||
|
- Verifica UTXO in tempo reale (speso/non speso)
|
||||||
|
- Report HTML interattivo con statistiche
|
||||||
|
- Scansione incrementale (riprende da dove si ferma)
|
||||||
|
- Rate limiting intelligente per rispettare le API
|
||||||
|
- Esportazione dati in CSV
|
||||||
|
|
||||||
|
**Tecnologie**:
|
||||||
|
- Python 3.x
|
||||||
|
- SQLite3
|
||||||
|
- Requests (HTTP API)
|
||||||
|
- HTML/CSS/JavaScript (report)
|
||||||
|
|
||||||
|
**Blocchi interessanti**:
|
||||||
|
- Blocchi 1-10000: Era di Satoshi, quasi tutti P2PK
|
||||||
|
- Blocchi 10000-100000: Transizione verso P2PKH
|
||||||
|
- Blocchi 100000+: Prevalentemente P2PKH, P2SH, SegWit
|
||||||
|
|
||||||
|
### Bruteforce C++ (Ricerca Chiavi)
|
||||||
|
|
||||||
|
**Scopo**: Dimostrare l'impossibilità pratica del bruteforce su Bitcoin.
|
||||||
|
|
||||||
|
**Caratteristiche**:
|
||||||
|
- Ottimizzato per massima velocità (C++ con secp256k1)
|
||||||
|
- Multi-threading automatico (usa tutti i core CPU)
|
||||||
|
- Partizionamento spazio chiavi 2^256 tra thread
|
||||||
|
- Generazione chiavi casuali con seed robusti
|
||||||
|
- Statistiche in tempo reale (velocità, tentativi, tempo)
|
||||||
|
- Logging progresso in CSV
|
||||||
|
- Zero sovrapposizione tra thread (efficienza massima)
|
||||||
|
- Formato numeri leggibile (K, M, G, T)
|
||||||
|
|
||||||
|
**Tecnologie**:
|
||||||
|
- C++11
|
||||||
|
- libsecp256k1 (libreria Bitcoin Core)
|
||||||
|
- pthread (multi-threading)
|
||||||
|
- Makefile (build system)
|
||||||
|
|
||||||
|
**Ottimizzazioni**:
|
||||||
|
- Compilazione nativa (`-march=native -mtune=native`)
|
||||||
|
- Link-Time Optimization (`-flto`)
|
||||||
|
- Profile-Guided Optimization (opzionale)
|
||||||
|
- Partitioning dello spazio delle chiavi
|
||||||
|
- Seed XOR-based per zero correlazione
|
||||||
|
|
||||||
|
## Statistiche e Performance
|
||||||
|
|
||||||
|
### Scanner Python
|
||||||
|
- **Velocità**: ~1-5 blocchi/sec (dipende da delay API)
|
||||||
|
- **API calls**: ~3-5 per blocco
|
||||||
|
- **Database size**: ~100KB per 1000 blocchi P2PK
|
||||||
|
- **Rate limit**: Configurabile (default 1.0s tra blocchi)
|
||||||
|
|
||||||
|
### Bruteforce C++
|
||||||
|
- **Raspberry Pi 5 (4 core)**: ~50-60K keys/sec
|
||||||
|
- **CPU moderna (8 core)**: ~100-500K keys/sec
|
||||||
|
- **CPU high-end (16+ core)**: ~1M+ keys/sec
|
||||||
|
- **Efficienza threading**: Lineare (100% utilizzo core)
|
||||||
|
|
||||||
|
### Matematica della (Im)Probabilità
|
||||||
|
|
||||||
|
**Spazio delle chiavi**: 2^256 ≈ 1.16 × 10^77 chiavi possibili
|
||||||
|
|
||||||
|
**Con 1 milione di chiavi/sec**:
|
||||||
|
- 1 anno = 3.15 × 10^13 tentativi
|
||||||
|
- Probabilità successo = 2.7 × 10^-64 (praticamente zero)
|
||||||
|
- Tempo per testare tutto = 3.67 × 10^63 anni
|
||||||
|
- Età dell'universo = 1.38 × 10^10 anni
|
||||||
|
|
||||||
|
**Conclusione**: È più probabile vincere la lotteria 10 volte consecutive che trovare una chiave per caso.
|
||||||
|
|
||||||
|
## Workflow Completo
|
||||||
|
|
||||||
|
### Ricerca Educativa Tipica
|
||||||
|
|
||||||
|
1. **Scansione Blockchain** (Scanner Python):
|
||||||
|
```bash
|
||||||
|
cd databases
|
||||||
|
python3 scan_blockchain.py
|
||||||
|
# Scansiona blocchi 1-10000 (ricchi di P2PK)
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Analisi Dati** (Visualizzatore):
|
||||||
|
```bash
|
||||||
|
python3 view_db.py
|
||||||
|
# Genera report HTML interattivo
|
||||||
|
```
|
||||||
|
|
||||||
|
3. **Estrazione Target** (Preparazione Bruteforce):
|
||||||
|
```bash
|
||||||
|
cd ../bruteforce
|
||||||
|
python3 extract_p2pk_utxo.py
|
||||||
|
# Estrae P2PK non spesi dal database
|
||||||
|
```
|
||||||
|
|
||||||
|
4. **Dimostrazione Sicurezza** (Bruteforce):
|
||||||
|
```bash
|
||||||
|
./p2pk_bruteforce
|
||||||
|
# Dimostra l'impossibilità pratica del bruteforce
|
||||||
|
```
|
||||||
|
|
||||||
|
## Scopo Educativo
|
||||||
|
|
||||||
|
Questo progetto insegna:
|
||||||
|
|
||||||
|
1. **Struttura Blockchain Bitcoin**:
|
||||||
|
- Differenza tra P2PK, P2PKH, P2SH, SegWit
|
||||||
|
- Formato transazioni e ScriptPubKey
|
||||||
|
- Concetto di UTXO (Unspent Transaction Output)
|
||||||
|
|
||||||
|
2. **Crittografia Bitcoin**:
|
||||||
|
- ECDSA su curva secp256k1
|
||||||
|
- Generazione chiavi pubbliche da private
|
||||||
|
- Spazio delle chiavi 2^256
|
||||||
|
|
||||||
|
3. **Sicurezza Crittografica**:
|
||||||
|
- Perché il bruteforce è impossibile
|
||||||
|
- Importanza dello spazio delle chiavi grande
|
||||||
|
- Scalabilità esponenziale (ogni bit raddoppia il tempo)
|
||||||
|
|
||||||
|
4. **Ottimizzazione Software**:
|
||||||
|
- Multi-threading efficiente
|
||||||
|
- Partizionamento dati
|
||||||
|
- Eliminazione race conditions
|
||||||
|
- Profiling e misurazione performance
|
||||||
|
|
||||||
|
## Requisiti di Sistema
|
||||||
|
|
||||||
|
### Minimi
|
||||||
|
- **OS**: Linux, macOS, Windows (WSL)
|
||||||
|
- **RAM**: 512MB
|
||||||
|
- **Storage**: 100MB
|
||||||
|
- **CPU**: Single core
|
||||||
|
- **Python**: 3.7+
|
||||||
|
- **Compiler**: GCC 7+ o Clang 8+
|
||||||
|
|
||||||
|
### Raccomandati
|
||||||
|
- **OS**: Linux (migliore performance)
|
||||||
|
- **RAM**: 2GB+
|
||||||
|
- **Storage**: 1GB+
|
||||||
|
- **CPU**: Multi-core (4+)
|
||||||
|
- **Python**: 3.10+
|
||||||
|
- **Compiler**: GCC 11+ o Clang 14+
|
||||||
|
|
||||||
|
## Collaborazione Multi-Utente
|
||||||
|
|
||||||
|
### Scansione Distribuita
|
||||||
|
|
||||||
|
Più utenti possono collaborare per scansionare blocchi diversi:
|
||||||
|
|
||||||
|
1. **Utente A**: Scansiona blocchi 1-5000
|
||||||
|
2. **Utente B**: Scansiona blocchi 5001-10000
|
||||||
|
3. **Utente C**: Scansiona blocchi 10001-15000
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
python view_db.py
|
# Sincronizza database via Git
|
||||||
|
git pull
|
||||||
|
python3 scan_blockchain.py # Scansiona il tuo range
|
||||||
|
git add databases/*.db
|
||||||
|
git commit -m "Scansionati blocchi X-Y"
|
||||||
|
git push
|
||||||
```
|
```
|
||||||
|
|
||||||
Questo genera il file `p2pk_report.html` con:
|
### Best Practices
|
||||||
- 📊 Statistiche generali (blocchi scansionati, P2PK trovati, valori)
|
- Coordinarsi sui range di blocchi
|
||||||
- 🟢 Stato UTXO (speso/non speso)
|
- Usare delay ≥ 1.0s per non sovraccaricare API
|
||||||
- 🔍 Ricerca per blocco, TXID o chiave pubblica
|
- Fare pull prima di ogni nuova scansione
|
||||||
- 📋 Pulsanti copia per chiavi pubbliche e TXID
|
- Committare database dopo scansioni complete
|
||||||
|
|
||||||
### Statistiche Rapide
|
|
||||||
|
|
||||||
Visualizza statistiche nel terminale:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
python view_db.py --stats
|
|
||||||
```
|
|
||||||
|
|
||||||
Output esempio:
|
|
||||||
```
|
|
||||||
============================================================
|
|
||||||
📁 Database: bitcoin_p2pk_study.db
|
|
||||||
📦 Ultimo blocco scansionato: 100
|
|
||||||
🔑 P2PK totali trovati: 45
|
|
||||||
📊 Blocchi unici con P2PK: 23
|
|
||||||
📈 Range blocchi: 1 - 100
|
|
||||||
💰 Valore totale: 1234.56789012 BTC (123456789012 sat)
|
|
||||||
📝 Transazioni uniche: 42
|
|
||||||
------------------------------------------------------------
|
|
||||||
💎 P2PK NON SPESI: 12
|
|
||||||
💵 Valore non speso: 567.89012345 BTC (56789012345 sat)
|
|
||||||
============================================================
|
|
||||||
```
|
|
||||||
|
|
||||||
## Struttura Database
|
|
||||||
|
|
||||||
Il database SQLite contiene due tabelle:
|
|
||||||
|
|
||||||
### `p2pk_addresses`
|
|
||||||
| Campo | Tipo | Descrizione |
|
|
||||||
|-------|------|-------------|
|
|
||||||
| id | INTEGER | ID univoco |
|
|
||||||
| block_height | INTEGER | Altezza blocco |
|
|
||||||
| txid | TEXT | Transaction ID |
|
|
||||||
| output_index | INTEGER | Indice output |
|
|
||||||
| scriptpubkey | TEXT | ScriptPubKey (chiave pubblica) |
|
|
||||||
| value_satoshi | INTEGER | Valore in satoshi |
|
|
||||||
| timestamp | INTEGER | Timestamp blocco |
|
|
||||||
| is_unspent | INTEGER | 1 = non speso, 0 = speso |
|
|
||||||
| last_checked | INTEGER | Ultimo controllo UTXO |
|
|
||||||
|
|
||||||
### `scan_progress`
|
|
||||||
| Campo | Tipo | Descrizione |
|
|
||||||
|-------|------|-------------|
|
|
||||||
| id | INTEGER | ID (sempre 1) |
|
|
||||||
| last_scanned_block | INTEGER | Ultimo blocco scansionato |
|
|
||||||
| total_p2pk_found | INTEGER | Totale P2PK trovati |
|
|
||||||
|
|
||||||
## Scansione Incrementale
|
|
||||||
|
|
||||||
Il database supporta scansioni incrementali:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Prima scansione: blocchi 1-100
|
|
||||||
python main.py
|
|
||||||
> Blocco di inizio: 1
|
|
||||||
> Blocco finale: 100
|
|
||||||
|
|
||||||
# Seconda scansione: blocchi 101-200 (continua automaticamente)
|
|
||||||
python main.py
|
|
||||||
> [ENTER] (usa default: ultimo + 1)
|
|
||||||
> Blocco finale: 200
|
|
||||||
|
|
||||||
# Puoi anche scansionare "indietro" o saltare range
|
|
||||||
python main.py
|
|
||||||
> Blocco di inizio: 500
|
|
||||||
> Blocco finale: 1000
|
|
||||||
```
|
|
||||||
|
|
||||||
Il database usa `UNIQUE(txid, output_index)` quindi non ci saranno duplicati.
|
|
||||||
|
|
||||||
## Performance e Rate Limiting
|
|
||||||
|
|
||||||
### Chiamate API per blocco
|
|
||||||
- 1x per ottenere hash blocco
|
|
||||||
- 1x per ottenere timestamp
|
|
||||||
- 1+ per transazioni (paginazione: 25 tx per pagina)
|
|
||||||
- 1x per ogni P2PK trovato (verifica UTXO)
|
|
||||||
|
|
||||||
**Totale**: ~3-5 chiamate API per blocco (più se ci sono molte transazioni o P2PK)
|
|
||||||
|
|
||||||
### Rate Limiting
|
|
||||||
- **Delay default**: 1.0 secondo tra blocchi
|
|
||||||
- **Minimo consigliato**: 0.1 secondi
|
|
||||||
- mempool.space non ha limiti ufficiali documentati, ma rispetta sempre l'API
|
|
||||||
|
|
||||||
**Tempo stimato**:
|
|
||||||
- 100 blocchi × 1.0s = ~1.7 minuti
|
|
||||||
- 1000 blocchi × 1.0s = ~17 minuti
|
|
||||||
- 10000 blocchi × 1.0s = ~2.8 ore
|
|
||||||
|
|
||||||
## Esportazione Dati
|
|
||||||
|
|
||||||
### Esportazione CSV
|
|
||||||
|
|
||||||
Dopo la scansione, lo script chiederà:
|
|
||||||
```
|
|
||||||
📤 Esportare dati in CSV? (s/n): s
|
|
||||||
```
|
|
||||||
|
|
||||||
Questo genera `p2pk_export.csv` con tutte le colonne del database.
|
|
||||||
|
|
||||||
### Formato CSV
|
|
||||||
|
|
||||||
```csv
|
|
||||||
id,block_height,txid,output_index,scriptpubkey,value_satoshi,timestamp,is_unspent,last_checked
|
|
||||||
1,9,0437cd7f8525ceed2324359c2d0ba26006d92d85,0,4104d46c4968bde02899d2d0d...,5000000000,1231469744,0,1234567890
|
|
||||||
```
|
|
||||||
|
|
||||||
## Collaborazione
|
|
||||||
|
|
||||||
Questo progetto supporta la collaborazione multi-utente:
|
|
||||||
|
|
||||||
### Condivisione Database
|
|
||||||
Il file `.gitignore` **non** esclude:
|
|
||||||
- `*.db` (database SQLite)
|
|
||||||
- `*.csv` (esportazioni)
|
|
||||||
- `*.html` (report)
|
|
||||||
|
|
||||||
Più persone possono:
|
|
||||||
1. Clonare il repository con il database esistente
|
|
||||||
2. Scansionare range di blocchi diversi
|
|
||||||
3. Push/pull per sincronizzare i dati raccolti
|
|
||||||
4. Collaborare per coprire più blocchi velocemente
|
|
||||||
|
|
||||||
### Best Practice per Collaborazione
|
|
||||||
1. Coordinarsi sui range di blocchi da scansionare
|
|
||||||
2. Fare pull prima di iniziare una nuova scansione
|
|
||||||
3. Committare e pushare il database dopo ogni scansione completata
|
|
||||||
4. Usare delay >= 1.0s per non sovraccaricare l'API
|
|
||||||
|
|
||||||
## Come Funziona
|
|
||||||
|
|
||||||
### Rilevamento P2PK
|
|
||||||
|
|
||||||
Lo scanner usa 4 metodi per identificare P2PK:
|
|
||||||
|
|
||||||
1. **Tipo esplicito**: `scriptpubkey_type == 'pubkey'`
|
|
||||||
2. **Lunghezza script**: 67 byte (non compresso) o 35 byte (compresso)
|
|
||||||
3. **Pattern ASM**: `<PUBKEY> OP_CHECKSIG`
|
|
||||||
4. **Pattern HEX**: `41<pubkey>ac` o `21<pubkey>ac`
|
|
||||||
|
|
||||||
### Formato ScriptPubKey
|
|
||||||
|
|
||||||
**P2PK non compresso** (134 caratteri hex):
|
|
||||||
```
|
|
||||||
41 + <65 byte pubkey> + ac
|
|
||||||
```
|
|
||||||
|
|
||||||
**P2PK compresso** (70 caratteri hex):
|
|
||||||
```
|
|
||||||
21 + <33 byte pubkey> + ac
|
|
||||||
```
|
|
||||||
|
|
||||||
### Verifica UTXO
|
|
||||||
|
|
||||||
Per ogni P2PK trovato, lo scanner verifica su mempool.space se l'output è ancora non speso:
|
|
||||||
|
|
||||||
```
|
|
||||||
GET /api/tx/{txid}/outspend/{vout}
|
|
||||||
```
|
|
||||||
|
|
||||||
Risposta:
|
|
||||||
```json
|
|
||||||
{
|
|
||||||
"spent": false, // true se speso
|
|
||||||
"txid": "...", // txid che ha speso (se spent=true)
|
|
||||||
"vin": 0 // input index (se spent=true)
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Risoluzione Problemi
|
## Risoluzione Problemi
|
||||||
|
|
||||||
### L'ambiente virtuale non si attiva
|
### Scanner Python
|
||||||
|
|
||||||
**Linux/macOS:**
|
**Errore: API timeout**
|
||||||
```bash
|
```bash
|
||||||
chmod +x .venv/bin/activate
|
# Aumenta il delay tra richieste
|
||||||
source .venv/bin/activate
|
# Nel prompt: delay = 2.0 secondi
|
||||||
```
|
```
|
||||||
|
|
||||||
**Windows PowerShell:**
|
**Errore: Database locked**
|
||||||
```powershell
|
|
||||||
Set-ExecutionPolicy -ExecutionPolicy RemoteSigned -Scope CurrentUser
|
|
||||||
.venv\Scripts\Activate.ps1
|
|
||||||
```
|
|
||||||
|
|
||||||
### Errori API / Timeout
|
|
||||||
|
|
||||||
- Verifica la connessione internet
|
|
||||||
- Aumenta il delay tra richieste: `delay: 2.0`
|
|
||||||
- mempool.space potrebbe essere temporaneamente non disponibile
|
|
||||||
|
|
||||||
### Database locked
|
|
||||||
|
|
||||||
Se più processi accedono al database:
|
|
||||||
```bash
|
```bash
|
||||||
# Chiudi tutti gli script Python in esecuzione
|
# Chiudi tutti gli script che usano il database
|
||||||
# Riavvia lo script
|
pkill -f scan_blockchain
|
||||||
```
|
```
|
||||||
|
|
||||||
## Blocchi Interessanti
|
### Bruteforce C++
|
||||||
|
|
||||||
I primi blocchi Bitcoin contengono molti P2PK:
|
**Errore: libsecp256k1 non trovata**
|
||||||
|
```bash
|
||||||
|
cd bruteforce
|
||||||
|
make install-secp256k1
|
||||||
|
```
|
||||||
|
|
||||||
- **Blocchi 1-10000**: Era di Satoshi, quasi tutti P2PK
|
**Velocità troppo bassa**
|
||||||
- **Blocchi 10000-100000**: Transizione verso P2PKH
|
```bash
|
||||||
- **Blocchi 100000+**: Prevalentemente P2PKH, P2SH, SegWit
|
# Verifica numero thread
|
||||||
|
htop # Controlla utilizzo CPU
|
||||||
|
|
||||||
**Consiglio**: Inizia dai blocchi 1-10000 per trovare più P2PK.
|
# Ricompila con ottimizzazioni
|
||||||
|
make clean
|
||||||
|
make
|
||||||
|
```
|
||||||
|
|
||||||
## Crediti
|
## Crediti
|
||||||
|
- **Bitcoin**: [bitcoin.org](https://bitcoin.org)
|
||||||
|
- **API**: [mempool.space](https://mempool.space)
|
||||||
|
- **libsecp256k1**: [Bitcoin Core](https://github.com/bitcoin-core/secp256k1)
|
||||||
|
- **Documentazione**: [Bitcoin Wiki](https://en.bitcoin.it/wiki)
|
||||||
|
|
||||||
- API: [mempool.space](https://mempool.space/docs/api/rest)
|
## Contributi
|
||||||
- Bitcoin: [bitcoin.org](https://bitcoin.org)
|
Sviluppato con assistenza di Claude Code (Anthropic).
|
||||||
- P2PK Reference: [Bitcoin Wiki](https://en.bitcoin.it/wiki/Transaction)
|
|
||||||
|
## Link Utili
|
||||||
|
|
||||||
|
- [Bitcoin Whitepaper](https://bitcoin.org/bitcoin.pdf)
|
||||||
|
- [secp256k1 Documentation](https://github.com/bitcoin-core/secp256k1)
|
||||||
|
- [Mempool.space API](https://mempool.space/docs/api)
|
||||||
|
- [P2PK Transactions](https://en.bitcoin.it/wiki/Transaction)
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
**⚠️ Ricorda**: Questo è un progetto educativo. Le chiavi pubbliche P2PK non devono essere utilizzate per scopi illeciti.
|
**⚠️ Ricorda**: La sicurezza di Bitcoin si basa sull'impossibilità computazionale del bruteforce. Questo progetto lo dimostra in pratica.
|
||||||
|
|||||||
205
bruteforce/Makefile
Normal file
205
bruteforce/Makefile
Normal file
@@ -0,0 +1,205 @@
|
|||||||
|
# Makefile per Bitcoin P2PK Bruteforce
|
||||||
|
|
||||||
|
CC = g++
|
||||||
|
|
||||||
|
# Ottimizzazioni aggressive per CPU moderna
|
||||||
|
# -O3: massima ottimizzazione
|
||||||
|
# -march=native: usa tutte le istruzioni del processore (AVX2, SSE4.2, etc)
|
||||||
|
# -mtune=native: ottimizza per il processore specifico
|
||||||
|
# -flto: Link Time Optimization
|
||||||
|
# -ffast-math: ottimizzazioni matematiche aggressive (safe per crypto)
|
||||||
|
# -funroll-loops: srotola loop piccoli
|
||||||
|
# -finline-functions: inline aggressivo
|
||||||
|
# -fprefetch-loop-arrays: prefetch automatico
|
||||||
|
# -faligned-new: supporto per aligned new (C++17)
|
||||||
|
CFLAGS = -O3 -march=native -mtune=native -flto -ffast-math \
|
||||||
|
-funroll-loops -finline-functions -fprefetch-loop-arrays \
|
||||||
|
-faligned-new -pthread -Wall -Wextra -std=c++17
|
||||||
|
|
||||||
|
# Librerie necessarie
|
||||||
|
LIBS = -lsecp256k1 -lgmp
|
||||||
|
|
||||||
|
# Target
|
||||||
|
TARGET = p2pk_bruteforce
|
||||||
|
SOURCE = p2pk_bruteforce.cpp
|
||||||
|
|
||||||
|
# Percorsi libreria
|
||||||
|
INCLUDE_PATH = -I/usr/local/include -I/usr/include
|
||||||
|
LIB_PATH = -L/usr/local/lib -L/usr/lib
|
||||||
|
|
||||||
|
# ============================================================================
|
||||||
|
# TARGET PRINCIPALI
|
||||||
|
# ============================================================================
|
||||||
|
|
||||||
|
all: build
|
||||||
|
|
||||||
|
# Compilazione standard
|
||||||
|
build: $(SOURCE)
|
||||||
|
@echo "========================================="
|
||||||
|
@echo " Bitcoin P2PK Bruteforce - Compilazione"
|
||||||
|
@echo "========================================="
|
||||||
|
@if [ -d "secp256k1" ]; then \
|
||||||
|
echo "[+] Compilazione con libsecp256k1 locale..."; \
|
||||||
|
$(CC) $(CFLAGS) \
|
||||||
|
-I./secp256k1/include \
|
||||||
|
-L./secp256k1/lib \
|
||||||
|
-Wl,-rpath,$(shell pwd)/secp256k1/lib \
|
||||||
|
-o $(TARGET) $(SOURCE) $(LIBS); \
|
||||||
|
else \
|
||||||
|
echo "[+] Compilazione con libsecp256k1 di sistema..."; \
|
||||||
|
$(CC) $(CFLAGS) $(INCLUDE_PATH) $(LIB_PATH) \
|
||||||
|
-o $(TARGET) $(SOURCE) $(LIBS); \
|
||||||
|
fi
|
||||||
|
@echo "[+] Compilazione completata!"
|
||||||
|
@echo "[+] Eseguibile: ./$(TARGET)"
|
||||||
|
@echo ""
|
||||||
|
@echo "OTTIMIZZAZIONI ATTIVE:"
|
||||||
|
@echo " ✓ Batch EC point addition (256 keys/iteration)"
|
||||||
|
@echo " ✓ Zero-copy lookup (no serialization)"
|
||||||
|
@echo " ✓ SIMD-optimized Bloom filter"
|
||||||
|
@echo " ✓ Cache-aligned memory"
|
||||||
|
@echo " ✓ CPU prefetching hints"
|
||||||
|
@echo " ✓ LTO & aggressive inlining"
|
||||||
|
@echo ""
|
||||||
|
@echo "PERFORMANCE ATTESE: 800K - 2M keys/sec"
|
||||||
|
@echo "========================================="
|
||||||
|
|
||||||
|
# ============================================================================
|
||||||
|
# PROFILE-GUIDED OPTIMIZATION (PGO)
|
||||||
|
# ============================================================================
|
||||||
|
|
||||||
|
pgo: pgo-generate pgo-run pgo-use
|
||||||
|
|
||||||
|
pgo-generate: $(SOURCE)
|
||||||
|
@echo "[+] Step 1/3: Compilazione con profile generation..."
|
||||||
|
@if [ -d "secp256k1" ]; then \
|
||||||
|
$(CC) $(CFLAGS) -fprofile-generate \
|
||||||
|
-I./secp256k1/include \
|
||||||
|
-L./secp256k1/lib \
|
||||||
|
-Wl,-rpath,$(shell pwd)/secp256k1/lib \
|
||||||
|
-o $(TARGET)_pgo $(SOURCE) $(LIBS); \
|
||||||
|
else \
|
||||||
|
$(CC) $(CFLAGS) -fprofile-generate $(INCLUDE_PATH) $(LIB_PATH) \
|
||||||
|
-o $(TARGET)_pgo $(SOURCE) $(LIBS); \
|
||||||
|
fi
|
||||||
|
@echo "[+] Pronto per eseguire il programma e generare profilo..."
|
||||||
|
@echo "[!] Esegui: timeout 30s ./$(TARGET)_pgo"
|
||||||
|
|
||||||
|
pgo-run:
|
||||||
|
@echo "[+] Step 2/3: Generazione profilo (30 secondi)..."
|
||||||
|
@timeout 30s ./$(TARGET)_pgo || true
|
||||||
|
@echo "[+] Profilo generato!"
|
||||||
|
|
||||||
|
pgo-use: $(SOURCE)
|
||||||
|
@echo "[+] Step 3/3: Ricompilazione con Profile-Guided Optimization..."
|
||||||
|
@if [ -d "secp256k1" ]; then \
|
||||||
|
$(CC) $(CFLAGS) -fprofile-use -fprofile-correction \
|
||||||
|
-I./secp256k1/include \
|
||||||
|
-L./secp256k1/lib \
|
||||||
|
-Wl,-rpath,$(shell pwd)/secp256k1/lib \
|
||||||
|
-o $(TARGET) $(SOURCE) $(LIBS); \
|
||||||
|
else \
|
||||||
|
$(CC) $(CFLAGS) -fprofile-use -fprofile-correction $(INCLUDE_PATH) $(LIB_PATH) \
|
||||||
|
-o $(TARGET) $(SOURCE) $(LIBS); \
|
||||||
|
fi
|
||||||
|
@echo "[+] PGO compilazione completata!"
|
||||||
|
@echo "[+] Eseguibile ottimizzato: ./$(TARGET)"
|
||||||
|
@echo "[!] Performance attese: +10-20% aggiuntivo"
|
||||||
|
@rm -f $(TARGET)_pgo
|
||||||
|
|
||||||
|
# ============================================================================
|
||||||
|
# UTILITÀ
|
||||||
|
# ============================================================================
|
||||||
|
|
||||||
|
# Versione debug
|
||||||
|
debug: $(SOURCE)
|
||||||
|
@echo "[+] Compilazione DEBUG..."
|
||||||
|
$(CC) -g -O0 -pthread -Wall -Wextra -std=c++17 \
|
||||||
|
$(INCLUDE_PATH) $(LIB_PATH) \
|
||||||
|
-o $(TARGET)_debug $(SOURCE) $(LIBS)
|
||||||
|
@echo "[+] Eseguibile debug: ./$(TARGET)_debug"
|
||||||
|
|
||||||
|
# Analisi assembly generato
|
||||||
|
asm: $(SOURCE)
|
||||||
|
@echo "[+] Generazione assembly..."
|
||||||
|
$(CC) $(CFLAGS) -S -fverbose-asm $(INCLUDE_PATH) \
|
||||||
|
-o $(TARGET).s $(SOURCE)
|
||||||
|
@echo "[+] Assembly salvato in: $(TARGET).s"
|
||||||
|
|
||||||
|
# Benchmark veloce (10 secondi)
|
||||||
|
bench: build
|
||||||
|
@echo "[+] Benchmark rapido (10 secondi)..."
|
||||||
|
@timeout 10s ./$(TARGET) || true
|
||||||
|
|
||||||
|
# Test con valgrind (memory leaks)
|
||||||
|
valgrind: debug
|
||||||
|
@echo "[+] Test con Valgrind..."
|
||||||
|
valgrind --leak-check=full --show-leak-kinds=all \
|
||||||
|
./$(TARGET)_debug
|
||||||
|
|
||||||
|
# Pulizia
|
||||||
|
clean:
|
||||||
|
@echo "[+] Pulizia file compilati..."
|
||||||
|
rm -f $(TARGET) $(TARGET)_debug $(TARGET)_pgo
|
||||||
|
rm -f *.o *.gcda *.gcno *.s
|
||||||
|
rm -f progress.csv
|
||||||
|
@echo "[+] Pulizia completata!"
|
||||||
|
|
||||||
|
clean-all: clean
|
||||||
|
@echo "[+] Pulizia completa..."
|
||||||
|
rm -rf secp256k1_build secp256k1
|
||||||
|
@echo "[+] Pulizia completa terminata!"
|
||||||
|
|
||||||
|
# ============================================================================
|
||||||
|
# DIPENDENZE
|
||||||
|
# ============================================================================
|
||||||
|
|
||||||
|
install-deps:
|
||||||
|
@echo "[+] Installazione dipendenze..."
|
||||||
|
@echo "[!] Richiede: build-essential, libsecp256k1-dev, libgmp-dev"
|
||||||
|
@read -p "Continuare? [y/N] " -n 1 -r; \
|
||||||
|
echo; \
|
||||||
|
if [[ $$REPLY =~ ^[Yy]$$ ]]; then \
|
||||||
|
sudo apt-get update && \
|
||||||
|
sudo apt-get install -y build-essential libsecp256k1-dev libgmp-dev \
|
||||||
|
git autoconf libtool pkg-config; \
|
||||||
|
echo "[+] Dipendenze installate!"; \
|
||||||
|
fi
|
||||||
|
|
||||||
|
build-secp256k1:
|
||||||
|
@echo "[+] Compilazione libsecp256k1 ottimizzata..."
|
||||||
|
@./build_secp256k1.sh
|
||||||
|
|
||||||
|
# ============================================================================
|
||||||
|
# HELP
|
||||||
|
# ============================================================================
|
||||||
|
|
||||||
|
help:
|
||||||
|
@echo "==================================================="
|
||||||
|
@echo " Bitcoin P2PK Bruteforce - Makefile"
|
||||||
|
@echo "==================================================="
|
||||||
|
@echo ""
|
||||||
|
@echo "Target disponibili:"
|
||||||
|
@echo " make - Compila il programma (default)"
|
||||||
|
@echo " make build - Compila il programma"
|
||||||
|
@echo " make pgo - Compila con Profile-Guided Optimization"
|
||||||
|
@echo " make debug - Compila versione debug"
|
||||||
|
@echo " make asm - Genera assembly per analisi"
|
||||||
|
@echo " make bench - Benchmark rapido (10s)"
|
||||||
|
@echo " make valgrind - Test memory leaks"
|
||||||
|
@echo " make clean - Rimuove file compilati"
|
||||||
|
@echo " make clean-all - Pulizia completa"
|
||||||
|
@echo " make install-deps - Installa dipendenze"
|
||||||
|
@echo " make build-secp256k1 - Compila libsecp256k1 locale"
|
||||||
|
@echo ""
|
||||||
|
@echo "Uso consigliato:"
|
||||||
|
@echo " 1. make # Compila"
|
||||||
|
@echo " 2. ./$(TARGET) # Esegui bruteforce"
|
||||||
|
@echo ""
|
||||||
|
@echo "Per massime performance:"
|
||||||
|
@echo " make pgo # Compila con PGO (+10-20% speed)"
|
||||||
|
@echo ""
|
||||||
|
@echo "==================================================="
|
||||||
|
|
||||||
|
.PHONY: all build pgo pgo-generate pgo-run pgo-use debug asm bench \
|
||||||
|
valgrind clean clean-all install-deps build-secp256k1 help
|
||||||
269
bruteforce/README.md
Normal file
269
bruteforce/README.md
Normal file
@@ -0,0 +1,269 @@
|
|||||||
|
# Bitcoin P2PK Bruteforce
|
||||||
|
|
||||||
|
Strumento di ricerca chiavi private Bitcoin P2PK compilato per massime performance su CPU (fino a **300K+ keys/sec**).
|
||||||
|
|
||||||
|
**⚠️ DISCLAIMER**: Solo per scopi educativi e di ricerca. Non utilizzare per attività illegali.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Indice
|
||||||
|
|
||||||
|
- [Requisiti](#requisiti)
|
||||||
|
- [Installazione Rapida](#installazione-rapida)
|
||||||
|
- [Utilizzo](#utilizzo)
|
||||||
|
- [Comandi Make](#comandi-make)
|
||||||
|
- [Troubleshooting](#troubleshooting)
|
||||||
|
- [Note Tecniche](#note-tecniche)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Requisiti
|
||||||
|
|
||||||
|
### Sistema Operativo
|
||||||
|
- **Linux**: Ubuntu/Debian (consigliato)
|
||||||
|
- **Architettura**: x86_64 o ARM64
|
||||||
|
|
||||||
|
### Hardware Minimo
|
||||||
|
- **CPU**: Multicore (consigliato 8+ core)
|
||||||
|
- **RAM**: 2GB minimo, 4GB+ consigliato
|
||||||
|
- **Disco**: 500MB per libreria locale
|
||||||
|
|
||||||
|
### Software
|
||||||
|
```bash
|
||||||
|
sudo apt-get update
|
||||||
|
sudo apt-get install -y build-essential git autoconf libtool pkg-config libgmp-dev
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Installazione Rapida
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd bruteforce
|
||||||
|
|
||||||
|
# Compila automaticamente (prima volta ~5 minuti)
|
||||||
|
make
|
||||||
|
|
||||||
|
# Esegui
|
||||||
|
./p2pk_bruteforce
|
||||||
|
```
|
||||||
|
|
||||||
|
**Risultato atteso**: ~300K keys/sec
|
||||||
|
|
||||||
|
> **Nota**: La prima volta `make` compilerà automaticamente libsecp256k1 locale (~5 minuti). Le volte successive sarà istantaneo.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Utilizzo
|
||||||
|
|
||||||
|
### 1. Prepara il File Target
|
||||||
|
|
||||||
|
Crea `target_keys.txt` con le chiavi pubbliche P2PK (una per riga):
|
||||||
|
|
||||||
|
```txt
|
||||||
|
pubkey_hex
|
||||||
|
04a1b2c3d4e5f6789... (130 caratteri hex)
|
||||||
|
04f9e8d7c6b5a49321...
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
**Formato**:
|
||||||
|
- Chiavi pubbliche **non compresse** (130 caratteri hex)
|
||||||
|
- Prefisso `04` (opzionale)
|
||||||
|
- Una chiave per riga
|
||||||
|
|
||||||
|
### 2. Esegui il Programma
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# File default (target_keys.txt)
|
||||||
|
./p2pk_bruteforce
|
||||||
|
|
||||||
|
# File custom
|
||||||
|
./p2pk_bruteforce mie_chiavi.txt
|
||||||
|
```
|
||||||
|
|
||||||
|
### 3. Output Esempio
|
||||||
|
|
||||||
|
```
|
||||||
|
========================================
|
||||||
|
Bitcoin P2PK Bruteforce v1.0
|
||||||
|
SOLO PER SCOPI EDUCATIVI
|
||||||
|
========================================
|
||||||
|
|
||||||
|
[+] Inizializzazione secp256k1...
|
||||||
|
[+] Bloom filter inizializzato: 64 MB
|
||||||
|
[+] Caricamento chiavi target da target_keys.txt...
|
||||||
|
[+] Caricate 2164 chiavi pubbliche target
|
||||||
|
[+] CPU rilevata: 11 thread disponibili
|
||||||
|
[+] Partizionamento spazio chiavi in 11 regioni
|
||||||
|
[+] Avvio 11 thread worker...
|
||||||
|
|
||||||
|
[+] Thread 0 avviato su core 0
|
||||||
|
Privkey iniziale: 0000000000000000a3f2b1c4d5e6f7a8b9c0d1e2f3a4b5c6d7e8f9a0b1c2d3e4
|
||||||
|
[+] Thread 1 avviato su core 1
|
||||||
|
Privkey iniziale: 1745d1741745d174f1e2d3c4b5a69788c9d0e1f2a3b4c5d6e7f8a9b0c1d2e3f4
|
||||||
|
...
|
||||||
|
|
||||||
|
[INFO] Tentativi: 3.30M | Velocità: 300.00K keys/sec | Tempo: 11s
|
||||||
|
[INFO] Tentativi: 6.60M | Velocità: 300.00K keys/sec | Tempo: 22s
|
||||||
|
```
|
||||||
|
|
||||||
|
### 4. Chiave Trovata (se succede)
|
||||||
|
|
||||||
|
```
|
||||||
|
========================================
|
||||||
|
🎯 CHIAVE TROVATA! 🎯
|
||||||
|
========================================
|
||||||
|
Private Key: a1b2c3d4e5f6...
|
||||||
|
Public Key: 04f9e8d7c6b5a4...
|
||||||
|
========================================
|
||||||
|
```
|
||||||
|
|
||||||
|
**Salvata in**: `found_keys.txt`
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Comandi Make
|
||||||
|
|
||||||
|
```bash
|
||||||
|
make # Compila (setup automatico prima volta)
|
||||||
|
make clean # Rimuove eseguibili
|
||||||
|
make clean-all # Pulizia completa (include secp256k1)
|
||||||
|
make help # Mostra tutti i comandi
|
||||||
|
```
|
||||||
|
|
||||||
|
### Workflow Consigliato
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Prima compilazione
|
||||||
|
make
|
||||||
|
|
||||||
|
# Modifiche successive al codice
|
||||||
|
make clean
|
||||||
|
make
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Troubleshooting
|
||||||
|
|
||||||
|
### Errore: Directory secp256k1 non trovata
|
||||||
|
|
||||||
|
```bash
|
||||||
|
make clean-all
|
||||||
|
make
|
||||||
|
```
|
||||||
|
|
||||||
|
### Velocità Bassa (<250K keys/sec)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# 1. Verifica libreria usata
|
||||||
|
ldd ./p2pk_bruteforce | grep secp256k1
|
||||||
|
# Deve mostrare: ./secp256k1/lib/libsecp256k1.so
|
||||||
|
|
||||||
|
# 2. Verifica carico CPU
|
||||||
|
htop # Ogni core dovrebbe essere al 100%
|
||||||
|
|
||||||
|
# 3. Verifica frequency scaling
|
||||||
|
lscpu | grep MHz
|
||||||
|
# Se bassa, disabilita power saving
|
||||||
|
```
|
||||||
|
|
||||||
|
### Errori di Compilazione
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Reinstalla dipendenze
|
||||||
|
sudo apt-get install -y build-essential git autoconf libtool pkg-config libgmp-dev
|
||||||
|
|
||||||
|
# Pulizia completa e ricompilazione
|
||||||
|
make clean-all
|
||||||
|
make
|
||||||
|
```
|
||||||
|
|
||||||
|
### Il Programma non Trova Nulla
|
||||||
|
|
||||||
|
**Questo è normale!** La probabilità di trovare una chiave è praticamente zero.
|
||||||
|
|
||||||
|
**Test funzionamento**:
|
||||||
|
1. Genera una chiave privata nota
|
||||||
|
2. Calcola la pubkey con Bitcoin Core
|
||||||
|
3. Aggiungi a `target_keys.txt`
|
||||||
|
4. Modifica codice per partire da quella chiave
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Note Tecniche
|
||||||
|
|
||||||
|
### Probabilità di Successo
|
||||||
|
|
||||||
|
**Spazio chiavi**: 2^256 ≈ 10^77
|
||||||
|
|
||||||
|
Con 300K keys/sec:
|
||||||
|
- **1 ora**: ~1 miliardo (10^9) chiavi
|
||||||
|
- **1 anno**: ~9.5 trilioni (10^15) chiavi
|
||||||
|
- **Universo**: Servirebbe **10^60 anni** per tutto lo spazio
|
||||||
|
|
||||||
|
**Conclusione**: Statisticamente impossibile trovare una chiave casuale.
|
||||||
|
|
||||||
|
### Partizionamento Spazio Chiavi
|
||||||
|
|
||||||
|
Ogni thread riceve un chunk univoco:
|
||||||
|
```
|
||||||
|
Thread 0: 0x0000000000000000... → 0x1745d1741745d174...
|
||||||
|
Thread 1: 0x1745d1741745d174... → 0x2e8ba2e82e8ba2e8...
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
All'interno del chunk:
|
||||||
|
1. Inizia con chiave casuale
|
||||||
|
2. Incrementa sequenzialmente +1
|
||||||
|
3. **Zero sovrapposizione** garantita
|
||||||
|
|
||||||
|
### Bloom Filter
|
||||||
|
|
||||||
|
- **Dimensione**: 64 MB (2^26 bits)
|
||||||
|
- **Hash functions**: 3 indipendenti
|
||||||
|
- **False positive**: ~0.01%
|
||||||
|
- **Speedup**: ~5-10%
|
||||||
|
|
||||||
|
### Algoritmo Incremento
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
// 4x più veloce di byte-per-byte
|
||||||
|
uint64_t* p64 = (uint64_t*)privkey;
|
||||||
|
if (++p64[3]) return; // 99.99% dei casi termina qui
|
||||||
|
if (++p64[2]) return;
|
||||||
|
if (++p64[1]) return;
|
||||||
|
++p64[0];
|
||||||
|
```
|
||||||
|
|
||||||
|
### libsecp256k1 Ottimizzata
|
||||||
|
|
||||||
|
Flags di compilazione:
|
||||||
|
```bash
|
||||||
|
-O3 # Ottimizzazione massima
|
||||||
|
-march=native # Istruzioni CPU native (AVX2, etc)
|
||||||
|
-mtune=native # Tuning per CPU specifica
|
||||||
|
-flto # Link-Time Optimization
|
||||||
|
-fomit-frame-pointer # Rimuove frame pointer
|
||||||
|
-funroll-loops # Loop unrolling aggressivo
|
||||||
|
```
|
||||||
|
|
||||||
|
Window size: **15** (standard ottimale per ECC)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Contributori
|
||||||
|
|
||||||
|
Sviluppato con **Claude Code** (Anthropic)
|
||||||
|
|
||||||
|
## Riferimenti
|
||||||
|
|
||||||
|
- [Bitcoin Core secp256k1](https://github.com/bitcoin-core/secp256k1)
|
||||||
|
- [SECP256k1 Specification](https://en.bitcoin.it/wiki/Secp256k1)
|
||||||
|
- [Bloom Filters](https://en.wikipedia.org/wiki/Bloom_filter)
|
||||||
|
- [Elliptic Curve Cryptography](https://en.wikipedia.org/wiki/Elliptic-curve_cryptography)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**⚠️ Ricorda**: Questo progetto dimostra l'impossibilità pratica del bruteforce su Bitcoin. **La crittografia funziona.**
|
||||||
78
bruteforce/build_secp256k1.sh
Executable file
78
bruteforce/build_secp256k1.sh
Executable file
@@ -0,0 +1,78 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# Script per compilare libsecp256k1 con ottimizzazioni massime
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
echo "[+] Compilazione libsecp256k1 ottimizzata per massime performance..."
|
||||||
|
|
||||||
|
# Directory di lavoro - locale nella cartella bruteforce
|
||||||
|
SCRIPT_DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )"
|
||||||
|
WORK_DIR="$SCRIPT_DIR/secp256k1_build"
|
||||||
|
INSTALL_DIR="$SCRIPT_DIR/secp256k1"
|
||||||
|
|
||||||
|
# Cleanup
|
||||||
|
echo "[+] Pulizia directory precedenti..."
|
||||||
|
rm -rf "$WORK_DIR"
|
||||||
|
rm -rf "$INSTALL_DIR"
|
||||||
|
mkdir -p "$WORK_DIR"
|
||||||
|
mkdir -p "$INSTALL_DIR"
|
||||||
|
|
||||||
|
cd "$WORK_DIR"
|
||||||
|
|
||||||
|
# Clone repository ufficiale
|
||||||
|
echo "[+] Download libsecp256k1 da Bitcoin Core..."
|
||||||
|
git clone --depth 1 https://github.com/bitcoin-core/secp256k1.git
|
||||||
|
cd secp256k1
|
||||||
|
|
||||||
|
# Configura con ottimizzazioni massime
|
||||||
|
echo "[+] Configurazione con ottimizzazioni massime..."
|
||||||
|
./autogen.sh
|
||||||
|
|
||||||
|
# FLAGS CRITICI PER PERFORMANCE:
|
||||||
|
# --with-ecmult-window=15: Window size standard ottimale (24 richiede rigenerazione tabelle)
|
||||||
|
# --enable-module-recovery: Abilita recovery
|
||||||
|
# --enable-benchmark: Per testare performance
|
||||||
|
# --disable-tests: Velocizza la compilazione
|
||||||
|
# --disable-exhaustive-tests: Velocizza la compilazione
|
||||||
|
|
||||||
|
echo "[+] Configurazione..."
|
||||||
|
CFLAGS="-O3 -march=native -mtune=native -flto -fomit-frame-pointer -funroll-loops" \
|
||||||
|
./configure \
|
||||||
|
--prefix="$INSTALL_DIR" \
|
||||||
|
--with-ecmult-window=15 \
|
||||||
|
--enable-module-recovery \
|
||||||
|
--enable-module-extrakeys \
|
||||||
|
--enable-module-schnorrsig \
|
||||||
|
--enable-benchmark \
|
||||||
|
--disable-tests \
|
||||||
|
--disable-exhaustive-tests
|
||||||
|
|
||||||
|
echo "[+] Compilazione (usando $(nproc) core)..."
|
||||||
|
make -j$(nproc)
|
||||||
|
|
||||||
|
echo "[+] Installazione in $INSTALL_DIR..."
|
||||||
|
make install
|
||||||
|
|
||||||
|
# Test rapido
|
||||||
|
echo "[+] Test performance..."
|
||||||
|
if [ -f "$INSTALL_DIR/bin/bench_ecmult" ]; then
|
||||||
|
echo "[+] Running benchmark..."
|
||||||
|
"$INSTALL_DIR/bin/bench_ecmult" || true
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Pulizia directory temporanea di build (28MB risparmiati)
|
||||||
|
echo "[+] Rimozione directory temporanea di build..."
|
||||||
|
cd "$SCRIPT_DIR"
|
||||||
|
rm -rf "$WORK_DIR"
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "=========================================="
|
||||||
|
echo " libsecp256k1 INSTALLATA"
|
||||||
|
echo "=========================================="
|
||||||
|
echo ""
|
||||||
|
echo "Directory: $INSTALL_DIR"
|
||||||
|
echo "Libreria: $INSTALL_DIR/lib/libsecp256k1.so"
|
||||||
|
echo "Headers: $INSTALL_DIR/include/secp256k1.h"
|
||||||
|
echo ""
|
||||||
|
echo "Performance atteso: 1.5-2x più veloce della versione di sistema"
|
||||||
|
echo "=========================================="
|
||||||
242
bruteforce/extract_p2pk_utxo.py
Executable file
242
bruteforce/extract_p2pk_utxo.py
Executable file
@@ -0,0 +1,242 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
"""
|
||||||
|
Script per estrarre chiavi pubbliche P2PK con UTXO non spesi
|
||||||
|
dal database SQLite generato dallo scanner
|
||||||
|
"""
|
||||||
|
|
||||||
|
import sqlite3
|
||||||
|
import sys
|
||||||
|
import os
|
||||||
|
|
||||||
|
def extract_p2pk_unspent(db_path, output_file):
|
||||||
|
"""
|
||||||
|
Estrae le chiavi pubbliche P2PK non spese dal database
|
||||||
|
e le salva in un file di testo (una per riga)
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Verifica esistenza database
|
||||||
|
if not os.path.exists(db_path):
|
||||||
|
print(f"[ERROR] Database non trovato: {db_path}")
|
||||||
|
print("[!] Esegui prima lo scanner Python per creare il database")
|
||||||
|
return False
|
||||||
|
|
||||||
|
try:
|
||||||
|
# Connetti al database
|
||||||
|
conn = sqlite3.connect(db_path)
|
||||||
|
cursor = conn.cursor()
|
||||||
|
|
||||||
|
# Query per ottenere solo P2PK non spesi
|
||||||
|
query = """
|
||||||
|
SELECT
|
||||||
|
scriptpubkey,
|
||||||
|
block_height,
|
||||||
|
txid,
|
||||||
|
output_index,
|
||||||
|
value_satoshi
|
||||||
|
FROM p2pk_addresses
|
||||||
|
WHERE is_unspent = 1
|
||||||
|
ORDER BY value_satoshi DESC
|
||||||
|
"""
|
||||||
|
|
||||||
|
cursor.execute(query)
|
||||||
|
results = cursor.fetchall()
|
||||||
|
|
||||||
|
if not results:
|
||||||
|
print("[!] Nessun P2PK non speso trovato nel database")
|
||||||
|
conn.close()
|
||||||
|
return False
|
||||||
|
|
||||||
|
print(f"[+] Trovati {len(results)} P2PK non spesi")
|
||||||
|
print("[+] Estrazione in corso...")
|
||||||
|
|
||||||
|
# Estrai le chiavi pubbliche
|
||||||
|
pubkeys = []
|
||||||
|
total_value = 0
|
||||||
|
|
||||||
|
with open(output_file, 'w') as f:
|
||||||
|
# Header
|
||||||
|
f.write("# Bitcoin P2PK Public Keys (Unspent)\n")
|
||||||
|
f.write("# Format: One public key per line (hex, uncompressed)\n")
|
||||||
|
f.write("# Generated from database\n")
|
||||||
|
f.write("#\n")
|
||||||
|
|
||||||
|
for row in results:
|
||||||
|
scriptpubkey, block_height, txid, output_index, value_sat = row
|
||||||
|
|
||||||
|
# Il scriptpubkey è nel formato: 41<pubkey>ac o 21<pubkey>ac
|
||||||
|
# Dobbiamo estrarre solo la pubkey
|
||||||
|
|
||||||
|
# Rimuovi il primo byte (41 o 21) e l'ultimo byte (ac - OP_CHECKSIG)
|
||||||
|
if len(scriptpubkey) == 134: # Non compresso: 41 + 130 char (65 bytes) + ac
|
||||||
|
pubkey = scriptpubkey[2:-2] # Rimuovi 41 e ac
|
||||||
|
elif len(scriptpubkey) == 70: # Compresso: 21 + 66 char (33 bytes) + ac
|
||||||
|
pubkey = scriptpubkey[2:-2] # Rimuovi 21 e ac
|
||||||
|
# NOTA: Il bruteforce genera solo pubkey non compresse
|
||||||
|
# quindi le chiavi compresse non verranno trovate
|
||||||
|
print(f"[!] SKIP chiave compressa: {txid}:{output_index}")
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
print(f"[!] SKIP formato sconosciuto: {scriptpubkey}")
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Aggiungi il prefisso 04 se non c'è (formato non compresso)
|
||||||
|
if not pubkey.startswith('04'):
|
||||||
|
pubkey = '04' + pubkey
|
||||||
|
|
||||||
|
# Verifica lunghezza corretta (130 caratteri hex = 65 bytes)
|
||||||
|
if len(pubkey) != 130:
|
||||||
|
print(f"[!] SKIP lunghezza errata ({len(pubkey)}): {txid}:{output_index}")
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Scrivi la pubkey nel file
|
||||||
|
f.write(f"{pubkey}\n")
|
||||||
|
pubkeys.append(pubkey)
|
||||||
|
total_value += value_sat
|
||||||
|
|
||||||
|
# Info dettagliate
|
||||||
|
btc_value = value_sat / 100000000.0
|
||||||
|
print(f" [{len(pubkeys)}] Block {block_height} | {btc_value:.8f} BTC | {txid[:16]}...:{output_index}")
|
||||||
|
|
||||||
|
conn.close()
|
||||||
|
|
||||||
|
# Statistiche finali
|
||||||
|
print("\n" + "="*60)
|
||||||
|
print(" ESTRAZIONE COMPLETATA")
|
||||||
|
print("="*60)
|
||||||
|
print(f"Chiavi pubbliche estratte: {len(pubkeys)}")
|
||||||
|
print(f"Valore totale: {total_value / 100000000.0:.8f} BTC")
|
||||||
|
print(f"Valore totale: {total_value:,} satoshi")
|
||||||
|
print(f"File output: {output_file}")
|
||||||
|
print("="*60)
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
except sqlite3.Error as e:
|
||||||
|
print(f"[ERROR] Errore database: {e}")
|
||||||
|
return False
|
||||||
|
except Exception as e:
|
||||||
|
print(f"[ERROR] Errore: {e}")
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def show_stats(db_path):
|
||||||
|
"""
|
||||||
|
Mostra statistiche sui P2PK nel database
|
||||||
|
"""
|
||||||
|
|
||||||
|
if not os.path.exists(db_path):
|
||||||
|
print(f"[ERROR] Database non trovato: {db_path}")
|
||||||
|
return
|
||||||
|
|
||||||
|
try:
|
||||||
|
conn = sqlite3.connect(db_path)
|
||||||
|
cursor = conn.cursor()
|
||||||
|
|
||||||
|
# Statistiche generali
|
||||||
|
cursor.execute("SELECT COUNT(*) FROM p2pk_addresses")
|
||||||
|
total_p2pk = cursor.fetchone()[0]
|
||||||
|
|
||||||
|
cursor.execute("SELECT COUNT(*) FROM p2pk_addresses WHERE is_unspent = 1")
|
||||||
|
unspent_p2pk = cursor.fetchone()[0]
|
||||||
|
|
||||||
|
cursor.execute("SELECT SUM(value_satoshi) FROM p2pk_addresses")
|
||||||
|
total_value = cursor.fetchone()[0] or 0
|
||||||
|
|
||||||
|
cursor.execute("SELECT SUM(value_satoshi) FROM p2pk_addresses WHERE is_unspent = 1")
|
||||||
|
unspent_value = cursor.fetchone()[0] or 0
|
||||||
|
|
||||||
|
print("\n" + "="*60)
|
||||||
|
print(" STATISTICHE DATABASE P2PK")
|
||||||
|
print("="*60)
|
||||||
|
print(f"Totale P2PK: {total_p2pk}")
|
||||||
|
print(f"P2PK non spesi: {unspent_p2pk}")
|
||||||
|
print(f"P2PK spesi: {total_p2pk - unspent_p2pk}")
|
||||||
|
print(f"-" * 60)
|
||||||
|
print(f"Valore totale: {total_value / 100000000.0:.8f} BTC")
|
||||||
|
print(f"Valore non speso: {unspent_value / 100000000.0:.8f} BTC")
|
||||||
|
print(f"Valore speso: {(total_value - unspent_value) / 100000000.0:.8f} BTC")
|
||||||
|
print("="*60)
|
||||||
|
|
||||||
|
# Top 10 P2PK non spesi per valore
|
||||||
|
print("\nTop 10 P2PK non spesi per valore:")
|
||||||
|
print("-" * 60)
|
||||||
|
|
||||||
|
cursor.execute("""
|
||||||
|
SELECT block_height, txid, output_index, value_satoshi, scriptpubkey
|
||||||
|
FROM p2pk_addresses
|
||||||
|
WHERE is_unspent = 1
|
||||||
|
ORDER BY value_satoshi DESC
|
||||||
|
LIMIT 10
|
||||||
|
""")
|
||||||
|
|
||||||
|
for i, row in enumerate(cursor.fetchall(), 1):
|
||||||
|
block, txid, vout, value, script = row
|
||||||
|
btc = value / 100000000.0
|
||||||
|
print(f"{i:2d}. Block {block:7d} | {btc:12.8f} BTC | {txid[:16]}...:{vout}")
|
||||||
|
|
||||||
|
print("="*60 + "\n")
|
||||||
|
|
||||||
|
conn.close()
|
||||||
|
|
||||||
|
except sqlite3.Error as e:
|
||||||
|
print(f"[ERROR] Errore database: {e}")
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
# Percorsi di default
|
||||||
|
db_path = "../databases/bitcoin_p2pk_study.db"
|
||||||
|
output_file = "target_keys.txt"
|
||||||
|
|
||||||
|
print("="*60)
|
||||||
|
print(" Bitcoin P2PK UTXO Extractor")
|
||||||
|
print(" Estrae chiavi pubbliche non spese per bruteforce")
|
||||||
|
print("="*60)
|
||||||
|
print()
|
||||||
|
|
||||||
|
# Controlla argomenti
|
||||||
|
if len(sys.argv) > 1:
|
||||||
|
if sys.argv[1] in ['-h', '--help']:
|
||||||
|
print("Uso:")
|
||||||
|
print(f" {sys.argv[0]} [database.db] [output.txt]")
|
||||||
|
print()
|
||||||
|
print("Esempi:")
|
||||||
|
print(f" {sys.argv[0]}")
|
||||||
|
print(f" {sys.argv[0]} custom.db keys.txt")
|
||||||
|
print(f" {sys.argv[0]} --stats")
|
||||||
|
return
|
||||||
|
elif sys.argv[1] == '--stats':
|
||||||
|
show_stats(db_path)
|
||||||
|
return
|
||||||
|
else:
|
||||||
|
db_path = sys.argv[1]
|
||||||
|
|
||||||
|
if len(sys.argv) > 2:
|
||||||
|
output_file = sys.argv[2]
|
||||||
|
|
||||||
|
# Mostra statistiche prima dell'estrazione
|
||||||
|
show_stats(db_path)
|
||||||
|
|
||||||
|
# Conferma utente
|
||||||
|
print(f"[?] Database: {db_path}")
|
||||||
|
print(f"[?] Output: {output_file}")
|
||||||
|
print()
|
||||||
|
response = input("Procedere con l'estrazione? (s/n): ")
|
||||||
|
|
||||||
|
if response.lower() not in ['s', 'y', 'si', 'yes']:
|
||||||
|
print("[!] Operazione annullata")
|
||||||
|
return
|
||||||
|
|
||||||
|
# Estrai le chiavi
|
||||||
|
print()
|
||||||
|
if extract_p2pk_unspent(db_path, output_file):
|
||||||
|
print()
|
||||||
|
print("[+] Estrazione completata con successo!")
|
||||||
|
print(f"[+] Ora puoi eseguire: ./p2pk_bruteforce {output_file}")
|
||||||
|
else:
|
||||||
|
print()
|
||||||
|
print("[!] Estrazione fallita")
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
713
bruteforce/p2pk_bruteforce.cpp
Normal file
713
bruteforce/p2pk_bruteforce.cpp
Normal file
@@ -0,0 +1,713 @@
|
|||||||
|
/*
|
||||||
|
* Bitcoin P2PK Bruteforce ULTRA-OPTIMIZED
|
||||||
|
* Versione CPU ottimizzata per massime prestazioni
|
||||||
|
*
|
||||||
|
* OTTIMIZZAZIONI IMPLEMENTATE:
|
||||||
|
* - Batch EC point addition (genera N chiavi con 1 moltiplicazione + N addizioni)
|
||||||
|
* - Zero-copy: niente serializzazione fino al match
|
||||||
|
* - Hash diretto su secp256k1_pubkey raw data
|
||||||
|
* - SIMD-friendly Bloom filter
|
||||||
|
* - Precomputed lookup tables
|
||||||
|
* - Cache-aligned memory
|
||||||
|
* - CPU prefetching hints
|
||||||
|
*
|
||||||
|
* DISCLAIMER: Solo per scopi educativi e di ricerca
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <time.h>
|
||||||
|
#include <signal.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <secp256k1.h>
|
||||||
|
#include <pthread.h>
|
||||||
|
#include <sys/time.h>
|
||||||
|
#include <vector>
|
||||||
|
#include <string>
|
||||||
|
#include <unordered_map>
|
||||||
|
#include <fstream>
|
||||||
|
#include <sstream>
|
||||||
|
#include <iostream>
|
||||||
|
#include <algorithm>
|
||||||
|
#include <cctype>
|
||||||
|
#include <sched.h>
|
||||||
|
#if defined(__x86_64__) || defined(__i386__) || defined(_M_X64) || defined(_M_IX86)
|
||||||
|
#include <immintrin.h> // Per SIMD intrinsics (SSE/AVX) su x86
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// CONFIGURAZIONE OTTIMIZZATA
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
#define EC_BATCH_SIZE 256 // Genera 256 chiavi consecutive con EC addition (+25% speed)
|
||||||
|
#define SYNC_BATCH 100000 // Sincronizza contatori ogni 100K chiavi
|
||||||
|
#define MAX_THREADS 256
|
||||||
|
#define BLOOM_SIZE_BITS 26 // 64MB Bloom filter
|
||||||
|
#define USE_BLOOM_FILTER 1
|
||||||
|
#define USE_EC_BATCH 1 // Abilita batch EC point addition
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// STRUTTURE DATI OTTIMIZZATE
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
// Struttura per memorizzare chiavi target
|
||||||
|
struct TargetKey {
|
||||||
|
uint8_t pubkey[65];
|
||||||
|
char hex[131];
|
||||||
|
};
|
||||||
|
|
||||||
|
// Hash ottimizzato per raw secp256k1_pubkey data (64 bytes)
|
||||||
|
struct PubkeyRawHash {
|
||||||
|
size_t operator()(const secp256k1_pubkey& key) const {
|
||||||
|
const uint64_t* p = reinterpret_cast<const uint64_t*>(key.data);
|
||||||
|
// XOR rapido dei primi 64 bit
|
||||||
|
return p[0] ^ p[1] ^ p[2];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct PubkeyRawEqual {
|
||||||
|
bool operator()(const secp256k1_pubkey& a, const secp256k1_pubkey& b) const {
|
||||||
|
return memcmp(a.data, b.data, 64) == 0;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
#if USE_BLOOM_FILTER
|
||||||
|
// Bloom Filter ottimizzato con prefetching e cache alignment
|
||||||
|
class __attribute__((aligned(64))) BloomFilter {
|
||||||
|
private:
|
||||||
|
uint64_t* bits;
|
||||||
|
size_t size_bits;
|
||||||
|
size_t size_words;
|
||||||
|
size_t mask;
|
||||||
|
|
||||||
|
// Hash functions ottimizzate - usa direttamente i 64 bytes interni
|
||||||
|
inline uint64_t hash1(const uint8_t* data) const {
|
||||||
|
const uint64_t* p = (const uint64_t*)data;
|
||||||
|
return p[0] ^ (p[1] << 7);
|
||||||
|
}
|
||||||
|
|
||||||
|
inline uint64_t hash2(const uint8_t* data) const {
|
||||||
|
const uint64_t* p = (const uint64_t*)data;
|
||||||
|
return p[2] ^ (p[3] << 13);
|
||||||
|
}
|
||||||
|
|
||||||
|
inline uint64_t hash3(const uint8_t* data) const {
|
||||||
|
const uint64_t* p = (const uint64_t*)data;
|
||||||
|
return p[4] ^ (p[5] << 19);
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
BloomFilter(size_t bits_exponent) {
|
||||||
|
size_bits = 1ULL << bits_exponent;
|
||||||
|
size_words = size_bits / 64;
|
||||||
|
mask = size_bits - 1;
|
||||||
|
|
||||||
|
// Alloca memoria allineata per cache lines (64 bytes)
|
||||||
|
int ret = posix_memalign((void**)&bits, 64, size_words * sizeof(uint64_t));
|
||||||
|
if (ret != 0) {
|
||||||
|
fprintf(stderr, "[ERROR] posix_memalign failed\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
memset(bits, 0, size_words * sizeof(uint64_t));
|
||||||
|
}
|
||||||
|
|
||||||
|
~BloomFilter() {
|
||||||
|
free(bits);
|
||||||
|
}
|
||||||
|
|
||||||
|
void add(const secp256k1_pubkey* pubkey) {
|
||||||
|
const uint8_t* data = pubkey->data;
|
||||||
|
uint64_t h1 = hash1(data) & mask;
|
||||||
|
uint64_t h2 = hash2(data) & mask;
|
||||||
|
uint64_t h3 = hash3(data) & mask;
|
||||||
|
|
||||||
|
bits[h1 >> 6] |= (1ULL << (h1 & 63));
|
||||||
|
bits[h2 >> 6] |= (1ULL << (h2 & 63));
|
||||||
|
bits[h3 >> 6] |= (1ULL << (h3 & 63));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verifica ultra-veloce con prefetching
|
||||||
|
inline bool might_contain(const secp256k1_pubkey* pubkey) const {
|
||||||
|
const uint8_t* data = pubkey->data;
|
||||||
|
uint64_t h1 = hash1(data) & mask;
|
||||||
|
uint64_t h2 = hash2(data) & mask;
|
||||||
|
uint64_t h3 = hash3(data) & mask;
|
||||||
|
|
||||||
|
// Prefetch delle cache lines
|
||||||
|
__builtin_prefetch(&bits[h1 >> 6], 0, 3);
|
||||||
|
__builtin_prefetch(&bits[h2 >> 6], 0, 3);
|
||||||
|
__builtin_prefetch(&bits[h3 >> 6], 0, 3);
|
||||||
|
|
||||||
|
return (bits[h1 >> 6] & (1ULL << (h1 & 63))) &&
|
||||||
|
(bits[h2 >> 6] & (1ULL << (h2 & 63))) &&
|
||||||
|
(bits[h3 >> 6] & (1ULL << (h3 & 63)));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
static BloomFilter* bloom_filter = NULL;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// VARIABILI GLOBALI
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
static volatile int keep_running = 1;
|
||||||
|
static secp256k1_context* ctx = NULL;
|
||||||
|
static std::vector<TargetKey> target_keys;
|
||||||
|
static std::unordered_map<secp256k1_pubkey, int, PubkeyRawHash, PubkeyRawEqual> target_map;
|
||||||
|
static uint64_t attempts_per_thread[MAX_THREADS] = {0};
|
||||||
|
static time_t start_time;
|
||||||
|
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
|
||||||
|
static FILE* log_file = NULL;
|
||||||
|
static int num_threads = 0;
|
||||||
|
|
||||||
|
#if USE_EC_BATCH
|
||||||
|
// Precomputed: G, 2G, 3G, ..., 256G per batch EC addition
|
||||||
|
static secp256k1_pubkey precomputed_G[EC_BATCH_SIZE];
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// STRUTTURA THREAD
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
struct ThreadData {
|
||||||
|
int thread_id;
|
||||||
|
uint64_t seed;
|
||||||
|
uint8_t range_start[32];
|
||||||
|
uint8_t range_end[32];
|
||||||
|
};
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// UTILITY FUNCTIONS
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
int get_num_threads() {
|
||||||
|
int num = (int)sysconf(_SC_NPROCESSORS_ONLN);
|
||||||
|
if (num < 1) num = 1;
|
||||||
|
if (num > 1) num--;
|
||||||
|
if (num > MAX_THREADS) num = MAX_THREADS;
|
||||||
|
return num;
|
||||||
|
}
|
||||||
|
|
||||||
|
void set_thread_affinity(int core_id) {
|
||||||
|
cpu_set_t cpuset;
|
||||||
|
CPU_ZERO(&cpuset);
|
||||||
|
CPU_SET(core_id, &cpuset);
|
||||||
|
pthread_t current_thread = pthread_self();
|
||||||
|
pthread_setaffinity_np(current_thread, sizeof(cpu_set_t), &cpuset);
|
||||||
|
}
|
||||||
|
|
||||||
|
void partition_keyspace(int thread_id, int total_threads, uint8_t* range_start, uint8_t* range_end) {
|
||||||
|
memset(range_start, 0, 32);
|
||||||
|
memset(range_end, 0xFF, 32);
|
||||||
|
|
||||||
|
uint64_t partition_size = UINT64_MAX / total_threads;
|
||||||
|
uint64_t start = partition_size * thread_id;
|
||||||
|
uint64_t end = (thread_id == total_threads - 1) ? UINT64_MAX : (partition_size * (thread_id + 1) - 1);
|
||||||
|
|
||||||
|
for (int i = 0; i < 8; i++) {
|
||||||
|
range_start[i] = (uint8_t)(start >> (56 - i * 8));
|
||||||
|
range_end[i] = (uint8_t)(end >> (56 - i * 8));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void sigint_handler(int sig) {
|
||||||
|
(void)sig;
|
||||||
|
keep_running = 0;
|
||||||
|
printf("\n\n[!] Interruzione rilevata, chiusura in corso...\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
void bytes_to_hex(const uint8_t* bytes, size_t len, char* hex) {
|
||||||
|
for (size_t i = 0; i < len; i++) {
|
||||||
|
sprintf(hex + (i * 2), "%02x", bytes[i]);
|
||||||
|
}
|
||||||
|
hex[len * 2] = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
int hex_to_bytes(const char* hex, uint8_t* bytes, size_t len) {
|
||||||
|
if (strlen(hex) != len * 2) return 0;
|
||||||
|
for (size_t i = 0; i < len; i++) {
|
||||||
|
sscanf(hex + (i * 2), "%2hhx", &bytes[i]);
|
||||||
|
}
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// CARICAMENTO TARGET KEYS
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
int load_target_keys(const char* filename) {
|
||||||
|
#if USE_BLOOM_FILTER
|
||||||
|
bloom_filter = new BloomFilter(BLOOM_SIZE_BITS);
|
||||||
|
printf("[+] Bloom filter inizializzato: %llu MB\n",
|
||||||
|
(unsigned long long)((1ULL << BLOOM_SIZE_BITS) / 8 / 1024 / 1024));
|
||||||
|
#endif
|
||||||
|
|
||||||
|
std::ifstream file(filename);
|
||||||
|
if (!file.is_open()) {
|
||||||
|
fprintf(stderr, "[ERROR] Impossibile aprire %s\n", filename);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string line;
|
||||||
|
int count = 0;
|
||||||
|
|
||||||
|
std::getline(file, line); // Skip header
|
||||||
|
|
||||||
|
while (std::getline(file, line)) {
|
||||||
|
if (line.empty()) continue;
|
||||||
|
|
||||||
|
std::string pubkey_hex = line;
|
||||||
|
pubkey_hex.erase(remove_if(pubkey_hex.begin(), pubkey_hex.end(), isspace), pubkey_hex.end());
|
||||||
|
|
||||||
|
if (pubkey_hex.length() != 130 && pubkey_hex.length() != 128) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pubkey_hex.length() == 128) {
|
||||||
|
pubkey_hex = "04" + pubkey_hex;
|
||||||
|
}
|
||||||
|
|
||||||
|
TargetKey key;
|
||||||
|
if (hex_to_bytes(pubkey_hex.c_str(), key.pubkey, 65)) {
|
||||||
|
strcpy(key.hex, pubkey_hex.c_str());
|
||||||
|
target_keys.push_back(key);
|
||||||
|
|
||||||
|
// Converti in secp256k1_pubkey per lookup diretto
|
||||||
|
secp256k1_pubkey pubkey_obj;
|
||||||
|
if (secp256k1_ec_pubkey_parse(ctx, &pubkey_obj, key.pubkey, 65)) {
|
||||||
|
target_map[pubkey_obj] = count;
|
||||||
|
|
||||||
|
#if USE_BLOOM_FILTER
|
||||||
|
bloom_filter->add(&pubkey_obj);
|
||||||
|
#endif
|
||||||
|
count++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
file.close();
|
||||||
|
printf("[+] Caricate %d chiavi pubbliche target\n", count);
|
||||||
|
printf("[+] Target map size: %zu entries\n", target_map.size());
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// PRECOMPUTE EC GENERATOR MULTIPLES
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
#if USE_EC_BATCH
|
||||||
|
void precompute_generator_multiples() {
|
||||||
|
printf("[+] Precomputing EC generator multiples (1G, 2G, ..., %dG)...\n", EC_BATCH_SIZE);
|
||||||
|
|
||||||
|
uint8_t privkey[32];
|
||||||
|
|
||||||
|
for (int i = 0; i < EC_BATCH_SIZE; i++) {
|
||||||
|
memset(privkey, 0, 32);
|
||||||
|
|
||||||
|
// Imposta il valore (i+1) come privkey
|
||||||
|
// Per i=0: privkey=1, per i=255: privkey=256 (0x0100)
|
||||||
|
uint16_t value = i + 1;
|
||||||
|
privkey[31] = (uint8_t)(value & 0xFF); // byte basso
|
||||||
|
privkey[30] = (uint8_t)((value >> 8) & 0xFF); // byte alto
|
||||||
|
|
||||||
|
if (!secp256k1_ec_pubkey_create(ctx, &precomputed_G[i], privkey)) {
|
||||||
|
fprintf(stderr, "[ERROR] Failed to precompute %dG\n", i+1);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("[+] Precomputation complete!\n");
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// CHIAVE PRIVATA RANDOMIZZATA
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
void init_random_privkey_in_range(uint8_t* privkey, uint64_t* seed,
|
||||||
|
const uint8_t* range_start, const uint8_t* /*range_end*/) {
|
||||||
|
for (int i = 0; i < 32; i++) {
|
||||||
|
*seed ^= *seed << 13;
|
||||||
|
*seed ^= *seed >> 7;
|
||||||
|
*seed ^= *seed << 17;
|
||||||
|
privkey[i] = (uint8_t)(*seed & 0xFF);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = 0; i < 8; i++) {
|
||||||
|
privkey[i] = range_start[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Incremento ottimizzato a 64-bit
|
||||||
|
static inline void increment_privkey(uint8_t* privkey) {
|
||||||
|
uint64_t* p64 = (uint64_t*)privkey;
|
||||||
|
if (++p64[3]) return;
|
||||||
|
if (++p64[2]) return;
|
||||||
|
if (++p64[1]) return;
|
||||||
|
++p64[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Incremento di N
|
||||||
|
static inline void add_to_privkey(uint8_t* privkey, uint64_t n) {
|
||||||
|
uint64_t* p64 = (uint64_t*)privkey;
|
||||||
|
|
||||||
|
// Add to least significant word (little-endian)
|
||||||
|
uint64_t old = p64[3];
|
||||||
|
p64[3] += n;
|
||||||
|
|
||||||
|
// Handle carry
|
||||||
|
if (p64[3] < old) {
|
||||||
|
if (++p64[2] == 0) {
|
||||||
|
if (++p64[1] == 0) {
|
||||||
|
++p64[0];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// MATCH CHECKING OTTIMIZZATO
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
static inline int check_match_fast(const secp256k1_pubkey* pubkey) {
|
||||||
|
#if USE_BLOOM_FILTER
|
||||||
|
// Prima passa: Bloom filter
|
||||||
|
if (!bloom_filter->might_contain(pubkey)) {
|
||||||
|
return -1; // Sicuramente non presente
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Lookup diretto nella hash map (zero copy!)
|
||||||
|
auto it = target_map.find(*pubkey);
|
||||||
|
if (it != target_map.end()) {
|
||||||
|
return it->second; // Indice nella lista target_keys
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// SALVATAGGIO CHIAVE TROVATA
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
void save_found_key(const uint8_t* privkey, int target_index) {
|
||||||
|
pthread_mutex_lock(&mutex);
|
||||||
|
|
||||||
|
char priv_hex[65];
|
||||||
|
bytes_to_hex(privkey, 32, priv_hex);
|
||||||
|
|
||||||
|
printf("\n\n");
|
||||||
|
printf("========================================\n");
|
||||||
|
printf("🎯 CHIAVE TROVATA! 🎯\n");
|
||||||
|
printf("========================================\n");
|
||||||
|
printf("Private Key: %s\n", priv_hex);
|
||||||
|
printf("Public Key: %s\n", target_keys[target_index].hex);
|
||||||
|
printf("========================================\n\n");
|
||||||
|
|
||||||
|
FILE* found_file = fopen("found_keys.txt", "a");
|
||||||
|
if (found_file) {
|
||||||
|
time_t now = time(NULL);
|
||||||
|
fprintf(found_file, "\n=== FOUND at %s", ctime(&now));
|
||||||
|
fprintf(found_file, "Private Key: %s\n", priv_hex);
|
||||||
|
fprintf(found_file, "Public Key: %s\n", target_keys[target_index].hex);
|
||||||
|
fprintf(found_file, "========================================\n");
|
||||||
|
fclose(found_file);
|
||||||
|
}
|
||||||
|
|
||||||
|
pthread_mutex_unlock(&mutex);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// LOGGING
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
void format_number(uint64_t num, char* buffer) {
|
||||||
|
if (num >= 1000000000000ULL) {
|
||||||
|
sprintf(buffer, "%.2fT", num / 1000000000000.0);
|
||||||
|
} else if (num >= 1000000000ULL) {
|
||||||
|
sprintf(buffer, "%.2fG", num / 1000000000.0);
|
||||||
|
} else if (num >= 1000000ULL) {
|
||||||
|
sprintf(buffer, "%.2fM", num / 1000000.0);
|
||||||
|
} else if (num >= 1000ULL) {
|
||||||
|
sprintf(buffer, "%.2fK", num / 1000.0);
|
||||||
|
} else {
|
||||||
|
sprintf(buffer, "%lu", num);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void log_progress() {
|
||||||
|
pthread_mutex_lock(&mutex);
|
||||||
|
|
||||||
|
time_t now = time(NULL);
|
||||||
|
double elapsed = difftime(now, start_time);
|
||||||
|
if (elapsed < 1) elapsed = 1;
|
||||||
|
|
||||||
|
uint64_t total = 0;
|
||||||
|
for (int i = 0; i < num_threads; i++) {
|
||||||
|
total += attempts_per_thread[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
double rate = total / elapsed;
|
||||||
|
|
||||||
|
char total_str[32];
|
||||||
|
char rate_str[32];
|
||||||
|
format_number(total, total_str);
|
||||||
|
format_number((uint64_t)rate, rate_str);
|
||||||
|
|
||||||
|
printf("[INFO] Tentativi: %s | Velocità: %s keys/sec | Tempo: %.0fs\n",
|
||||||
|
total_str, rate_str, elapsed);
|
||||||
|
|
||||||
|
if (log_file) {
|
||||||
|
fprintf(log_file, "%ld,%lu,%.2f\n", now, total, rate);
|
||||||
|
fflush(log_file);
|
||||||
|
}
|
||||||
|
|
||||||
|
pthread_mutex_unlock(&mutex);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// WORKER THREAD - VERSIONE ULTRA-OTTIMIZZATA
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
void* worker_thread(void* arg) {
|
||||||
|
ThreadData* data = (ThreadData*)arg;
|
||||||
|
int thread_id = data->thread_id;
|
||||||
|
uint64_t seed = data->seed;
|
||||||
|
|
||||||
|
set_thread_affinity(thread_id);
|
||||||
|
|
||||||
|
// Pre-alloca buffer
|
||||||
|
uint8_t privkey[32];
|
||||||
|
secp256k1_pubkey pubkey_batch[EC_BATCH_SIZE];
|
||||||
|
uint64_t local_attempts = 0;
|
||||||
|
|
||||||
|
init_random_privkey_in_range(privkey, &seed, data->range_start, data->range_end);
|
||||||
|
|
||||||
|
char privkey_start_hex[65];
|
||||||
|
bytes_to_hex(privkey, 32, privkey_start_hex);
|
||||||
|
printf("[+] Thread %d avviato su core %d\n", thread_id, thread_id);
|
||||||
|
printf(" Privkey iniziale: %s\n", privkey_start_hex);
|
||||||
|
|
||||||
|
// ========================================================================
|
||||||
|
// LOOP PRINCIPALE CON EC BATCH PROCESSING
|
||||||
|
// ========================================================================
|
||||||
|
|
||||||
|
#if USE_EC_BATCH
|
||||||
|
// VERSIONE CON BATCH EC POINT ADDITION
|
||||||
|
while (keep_running) {
|
||||||
|
// Step 1: Genera la prima pubkey del batch (P = privkey * G)
|
||||||
|
if (!secp256k1_ec_pubkey_create(ctx, &pubkey_batch[0], privkey)) {
|
||||||
|
increment_privkey(privkey);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Step 2: Check prima chiave
|
||||||
|
int match_idx = check_match_fast(&pubkey_batch[0]);
|
||||||
|
if (__builtin_expect(match_idx >= 0, 0)) {
|
||||||
|
save_found_key(privkey, match_idx);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Step 3: Genera le restanti (EC_BATCH_SIZE - 1) chiavi usando EC addition
|
||||||
|
// P1 = P + G, P2 = P + 2G, P3 = P + 3G, ...
|
||||||
|
// Questo è MOLTO più veloce di fare EC_BATCH_SIZE moltiplicazioni!
|
||||||
|
uint8_t temp_privkey[32];
|
||||||
|
memcpy(temp_privkey, privkey, 32);
|
||||||
|
|
||||||
|
for (int i = 1; i < EC_BATCH_SIZE && keep_running; i++) {
|
||||||
|
increment_privkey(temp_privkey);
|
||||||
|
|
||||||
|
// EC point addition: pubkey_batch[i] = pubkey_batch[0] + precomputed_G[i-1]
|
||||||
|
// Usa EC pubkey combine (somma di due punti)
|
||||||
|
const secp256k1_pubkey* pubkeys_to_add[2] = {&pubkey_batch[0], &precomputed_G[i]};
|
||||||
|
|
||||||
|
if (secp256k1_ec_pubkey_combine(ctx, &pubkey_batch[i], pubkeys_to_add, 2)) {
|
||||||
|
match_idx = check_match_fast(&pubkey_batch[i]);
|
||||||
|
if (__builtin_expect(match_idx >= 0, 0)) {
|
||||||
|
save_found_key(temp_privkey, match_idx);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
local_attempts += EC_BATCH_SIZE;
|
||||||
|
add_to_privkey(privkey, EC_BATCH_SIZE);
|
||||||
|
|
||||||
|
// Aggiorna contatore globale periodicamente
|
||||||
|
if ((local_attempts & (SYNC_BATCH - 1)) == 0) {
|
||||||
|
attempts_per_thread[thread_id] = local_attempts;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
// VERSIONE STANDARD (fallback senza batch)
|
||||||
|
while (keep_running) {
|
||||||
|
secp256k1_pubkey pubkey_obj;
|
||||||
|
|
||||||
|
for (int batch = 0; batch < SYNC_BATCH; batch++) {
|
||||||
|
if (__builtin_expect(secp256k1_ec_pubkey_create(ctx, &pubkey_obj, privkey), 1)) {
|
||||||
|
int match_idx = check_match_fast(&pubkey_obj);
|
||||||
|
if (__builtin_expect(match_idx >= 0, 0)) {
|
||||||
|
save_found_key(privkey, match_idx);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
increment_privkey(privkey);
|
||||||
|
}
|
||||||
|
|
||||||
|
local_attempts += SYNC_BATCH;
|
||||||
|
attempts_per_thread[thread_id] = local_attempts;
|
||||||
|
|
||||||
|
if (__builtin_expect(!keep_running, 0)) break;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
printf("[+] Thread %d terminato (%lu tentativi)\n", thread_id, local_attempts);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// MAIN
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
int main(int argc, char** argv) {
|
||||||
|
printf("========================================\n");
|
||||||
|
printf(" Bitcoin P2PK Bruteforce v2.0 ULTRA\n");
|
||||||
|
printf(" CPU-Optimized Edition\n");
|
||||||
|
printf(" SOLO PER SCOPI EDUCATIVI\n");
|
||||||
|
printf("========================================\n\n");
|
||||||
|
|
||||||
|
const char* target_file = "target_keys.txt";
|
||||||
|
if (argc > 1) {
|
||||||
|
target_file = argv[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Inizializza secp256k1
|
||||||
|
printf("[+] Inizializzazione secp256k1...\n");
|
||||||
|
ctx = secp256k1_context_create(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY);
|
||||||
|
if (!ctx) {
|
||||||
|
fprintf(stderr, "[ERROR] Impossibile creare contesto secp256k1\n");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Randomizza contesto
|
||||||
|
unsigned char random_seed[32];
|
||||||
|
FILE* urandom = fopen("/dev/urandom", "rb");
|
||||||
|
if (urandom) {
|
||||||
|
size_t bytes_read = fread(random_seed, 1, 32, urandom);
|
||||||
|
fclose(urandom);
|
||||||
|
if (bytes_read == 32) {
|
||||||
|
if (secp256k1_context_randomize(ctx, random_seed) != 1) {
|
||||||
|
fprintf(stderr, "[WARNING] secp256k1_context_randomize failed\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Precompute EC multiples
|
||||||
|
#if USE_EC_BATCH
|
||||||
|
precompute_generator_multiples();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Carica target keys
|
||||||
|
printf("[+] Caricamento chiavi target da %s...\n", target_file);
|
||||||
|
if (load_target_keys(target_file) == 0) {
|
||||||
|
fprintf(stderr, "[ERROR] Nessuna chiave target caricata\n");
|
||||||
|
secp256k1_context_destroy(ctx);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Setup signal handler
|
||||||
|
signal(SIGINT, sigint_handler);
|
||||||
|
signal(SIGTERM, sigint_handler);
|
||||||
|
|
||||||
|
// Apri file di log
|
||||||
|
log_file = fopen("progress.csv", "w");
|
||||||
|
if (log_file) {
|
||||||
|
fprintf(log_file, "timestamp,attempts,keys_per_sec\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Rileva numero di thread
|
||||||
|
num_threads = get_num_threads();
|
||||||
|
printf("[+] CPU rilevata: %d thread disponibili\n", num_threads);
|
||||||
|
printf("[+] Batch size: %d keys per iteration\n", EC_BATCH_SIZE);
|
||||||
|
|
||||||
|
start_time = time(NULL);
|
||||||
|
srand(time(NULL));
|
||||||
|
|
||||||
|
// Crea threads
|
||||||
|
pthread_t threads[MAX_THREADS];
|
||||||
|
ThreadData thread_data[MAX_THREADS];
|
||||||
|
|
||||||
|
printf("[+] Avvio %d thread worker...\n", num_threads);
|
||||||
|
|
||||||
|
for (int i = 0; i < num_threads; i++) {
|
||||||
|
thread_data[i].thread_id = i;
|
||||||
|
|
||||||
|
uint64_t base_seed = (uint64_t)time(NULL);
|
||||||
|
uint64_t thread_offset = ((uint64_t)i << 48);
|
||||||
|
uint64_t random_part = ((uint64_t)rand() << 32) | rand();
|
||||||
|
thread_data[i].seed = base_seed ^ thread_offset ^ random_part;
|
||||||
|
|
||||||
|
partition_keyspace(i, num_threads, thread_data[i].range_start, thread_data[i].range_end);
|
||||||
|
|
||||||
|
printf(" Thread %d: range 0x%02x%02x%02x%02x... (seed: %016lx)\n",
|
||||||
|
i,
|
||||||
|
thread_data[i].range_start[0], thread_data[i].range_start[1],
|
||||||
|
thread_data[i].range_start[2], thread_data[i].range_start[3],
|
||||||
|
thread_data[i].seed);
|
||||||
|
|
||||||
|
pthread_create(&threads[i], NULL, worker_thread, &thread_data[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("\n");
|
||||||
|
|
||||||
|
// Loop principale
|
||||||
|
while (keep_running) {
|
||||||
|
sleep(10);
|
||||||
|
log_progress();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Attendi terminazione threads
|
||||||
|
printf("[+] Attesa terminazione threads...\n");
|
||||||
|
for (int i = 0; i < num_threads; i++) {
|
||||||
|
pthread_join(threads[i], NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Statistiche finali
|
||||||
|
printf("\n========================================\n");
|
||||||
|
printf(" STATISTICHE FINALI\n");
|
||||||
|
printf("========================================\n");
|
||||||
|
|
||||||
|
uint64_t total = 0;
|
||||||
|
char thread_str[32];
|
||||||
|
for (int i = 0; i < num_threads; i++) {
|
||||||
|
total += attempts_per_thread[i];
|
||||||
|
format_number(attempts_per_thread[i], thread_str);
|
||||||
|
printf("Thread %d: %s tentativi\n", i, thread_str);
|
||||||
|
}
|
||||||
|
|
||||||
|
time_t end_time = time(NULL);
|
||||||
|
double elapsed = difftime(end_time, start_time);
|
||||||
|
if (elapsed < 1) elapsed = 1;
|
||||||
|
|
||||||
|
char total_str[32];
|
||||||
|
char rate_str[32];
|
||||||
|
format_number(total, total_str);
|
||||||
|
format_number((uint64_t)(total / elapsed), rate_str);
|
||||||
|
|
||||||
|
printf("----------------------------------------\n");
|
||||||
|
printf("Totale tentativi: %s\n", total_str);
|
||||||
|
printf("Tempo totale: %.0f secondi\n", elapsed);
|
||||||
|
printf("Velocità media: %s keys/sec\n", rate_str);
|
||||||
|
printf("========================================\n\n");
|
||||||
|
|
||||||
|
// Cleanup
|
||||||
|
if (log_file) fclose(log_file);
|
||||||
|
secp256k1_context_destroy(ctx);
|
||||||
|
|
||||||
|
#if USE_BLOOM_FILTER
|
||||||
|
delete bloom_filter;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
printf("[+] Programma terminato\n");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
313
databases/README.md
Normal file
313
databases/README.md
Normal file
@@ -0,0 +1,313 @@
|
|||||||
|
# Bitcoin P2PK Scanner
|
||||||
|
|
||||||
|
Scanner di transazioni Bitcoin Pay-to-Public-Key (P2PK) per scopi educativi e di ricerca sulla blockchain.
|
||||||
|
|
||||||
|
## ⚠️ Disclaimer
|
||||||
|
|
||||||
|
**Questo progetto è SOLO per scopi educativi e di ricerca.**
|
||||||
|
- Non utilizzare per attività illegali
|
||||||
|
- Rispetta sempre i termini di servizio delle API utilizzate
|
||||||
|
- Lo scopo è studiare la struttura della blockchain Bitcoin
|
||||||
|
|
||||||
|
## Caratteristiche
|
||||||
|
|
||||||
|
- Scansione blocchi Bitcoin per identificare transazioni P2PK
|
||||||
|
- Salvataggio dati in database SQLite
|
||||||
|
- Verifica UTXO (controllo se le chiavi hanno ancora saldo)
|
||||||
|
- Report HTML interattivo con ricerca e statistiche
|
||||||
|
- Esportazione dati in CSV
|
||||||
|
- Scansione incrementale (riprende da dove si è fermato)
|
||||||
|
- API mempool.space con rate limiting integrato
|
||||||
|
|
||||||
|
## Installazione
|
||||||
|
|
||||||
|
### 1. Clona il repository
|
||||||
|
|
||||||
|
### 2. Crea l'ambiente virtuale Python
|
||||||
|
|
||||||
|
**Linux/macOS:**
|
||||||
|
```bash
|
||||||
|
python3 -m venv .venv
|
||||||
|
source .venv/bin/activate
|
||||||
|
```
|
||||||
|
|
||||||
|
**Windows:**
|
||||||
|
```cmd
|
||||||
|
python -m venv .venv
|
||||||
|
.venv\Scripts\activate
|
||||||
|
```
|
||||||
|
|
||||||
|
### 3. Installa le dipendenze
|
||||||
|
|
||||||
|
```bash
|
||||||
|
pip install -r requirements.txt
|
||||||
|
```
|
||||||
|
|
||||||
|
## Utilizzo
|
||||||
|
|
||||||
|
### Scanner Principale
|
||||||
|
|
||||||
|
Avvia lo scanner interattivo:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
python main.py
|
||||||
|
```
|
||||||
|
|
||||||
|
Lo script ti chiederà:
|
||||||
|
- **Blocco di inizio**: da quale blocco iniziare (default: ultimo blocco + 1)
|
||||||
|
- **Blocco finale**: fino a quale blocco scansionare
|
||||||
|
- **Delay**: tempo tra richieste API in secondi (default: 1.0s, minimo: 0.1s)
|
||||||
|
|
||||||
|
Esempio di sessione:
|
||||||
|
```
|
||||||
|
📊 Ultimo blocco scannerizzato: 0
|
||||||
|
💡 I primi blocchi di Bitcoin (1-10000) contengono molti P2PK
|
||||||
|
|
||||||
|
📍 Blocco di inizio scansione (default: 1): 1
|
||||||
|
📍 Blocco finale scansione (default: 1000): 100
|
||||||
|
⏱️ Delay tra richieste in secondi (default: 1.0): 1.0
|
||||||
|
|
||||||
|
🔧 Configurazione:
|
||||||
|
Primo blocco: 1
|
||||||
|
Ultimo blocco: 100
|
||||||
|
Totale blocchi: 100
|
||||||
|
Delay richieste: 1.0s
|
||||||
|
Tempo stimato: ~1.7 minuti
|
||||||
|
|
||||||
|
▶️ Avviare la scansione? (s/n): s
|
||||||
|
```
|
||||||
|
|
||||||
|
### Visualizzatore Database
|
||||||
|
|
||||||
|
Genera un report HTML interattivo:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
python view_db.py
|
||||||
|
```
|
||||||
|
|
||||||
|
Questo genera il file `p2pk_report.html` con:
|
||||||
|
- 📊 Statistiche generali (blocchi scansionati, P2PK trovati, valori)
|
||||||
|
- 🟢 Stato UTXO (speso/non speso)
|
||||||
|
- 🔍 Ricerca per blocco, TXID o chiave pubblica
|
||||||
|
- 📋 Pulsanti copia per chiavi pubbliche e TXID
|
||||||
|
|
||||||
|
### Statistiche Rapide
|
||||||
|
|
||||||
|
Visualizza statistiche nel terminale:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
python view_db.py --stats
|
||||||
|
```
|
||||||
|
|
||||||
|
Output esempio:
|
||||||
|
```
|
||||||
|
============================================================
|
||||||
|
📁 Database: bitcoin_p2pk_study.db
|
||||||
|
📦 Ultimo blocco scansionato: 100
|
||||||
|
🔑 P2PK totali trovati: 45
|
||||||
|
📊 Blocchi unici con P2PK: 23
|
||||||
|
📈 Range blocchi: 1 - 100
|
||||||
|
💰 Valore totale: 1234.56789012 BTC (123456789012 sat)
|
||||||
|
📝 Transazioni uniche: 42
|
||||||
|
------------------------------------------------------------
|
||||||
|
💎 P2PK NON SPESI: 12
|
||||||
|
💵 Valore non speso: 567.89012345 BTC (56789012345 sat)
|
||||||
|
============================================================
|
||||||
|
```
|
||||||
|
|
||||||
|
## Struttura Database
|
||||||
|
|
||||||
|
Il database SQLite contiene due tabelle:
|
||||||
|
|
||||||
|
### `p2pk_addresses`
|
||||||
|
| Campo | Tipo | Descrizione |
|
||||||
|
|-------|------|-------------|
|
||||||
|
| id | INTEGER | ID univoco |
|
||||||
|
| block_height | INTEGER | Altezza blocco |
|
||||||
|
| txid | TEXT | Transaction ID |
|
||||||
|
| output_index | INTEGER | Indice output |
|
||||||
|
| scriptpubkey | TEXT | ScriptPubKey (chiave pubblica) |
|
||||||
|
| value_satoshi | INTEGER | Valore in satoshi |
|
||||||
|
| timestamp | INTEGER | Timestamp blocco |
|
||||||
|
| is_unspent | INTEGER | 1 = non speso, 0 = speso |
|
||||||
|
| last_checked | INTEGER | Ultimo controllo UTXO |
|
||||||
|
|
||||||
|
### `scan_progress`
|
||||||
|
| Campo | Tipo | Descrizione |
|
||||||
|
|-------|------|-------------|
|
||||||
|
| id | INTEGER | ID (sempre 1) |
|
||||||
|
| last_scanned_block | INTEGER | Ultimo blocco scansionato |
|
||||||
|
| total_p2pk_found | INTEGER | Totale P2PK trovati |
|
||||||
|
|
||||||
|
## Scansione Incrementale
|
||||||
|
|
||||||
|
Il database supporta scansioni incrementali:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Prima scansione: blocchi 1-100
|
||||||
|
python main.py
|
||||||
|
> Blocco di inizio: 1
|
||||||
|
> Blocco finale: 100
|
||||||
|
|
||||||
|
# Seconda scansione: blocchi 101-200 (continua automaticamente)
|
||||||
|
python main.py
|
||||||
|
> [ENTER] (usa default: ultimo + 1)
|
||||||
|
> Blocco finale: 200
|
||||||
|
|
||||||
|
# Puoi anche scansionare "indietro" o saltare range
|
||||||
|
python main.py
|
||||||
|
> Blocco di inizio: 500
|
||||||
|
> Blocco finale: 1000
|
||||||
|
```
|
||||||
|
|
||||||
|
Il database usa `UNIQUE(txid, output_index)` quindi non ci saranno duplicati.
|
||||||
|
|
||||||
|
## Performance e Rate Limiting
|
||||||
|
|
||||||
|
### Chiamate API per blocco
|
||||||
|
- 1x per ottenere hash blocco
|
||||||
|
- 1x per ottenere timestamp
|
||||||
|
- 1+ per transazioni (paginazione: 25 tx per pagina)
|
||||||
|
- 1x per ogni P2PK trovato (verifica UTXO)
|
||||||
|
|
||||||
|
**Totale**: ~3-5 chiamate API per blocco (più se ci sono molte transazioni o P2PK)
|
||||||
|
|
||||||
|
### Rate Limiting
|
||||||
|
- **Delay default**: 1.0 secondo tra blocchi
|
||||||
|
- **Minimo consigliato**: 0.1 secondi
|
||||||
|
- mempool.space non ha limiti ufficiali documentati, ma rispetta sempre l'API
|
||||||
|
|
||||||
|
**Tempo stimato**:
|
||||||
|
- 100 blocchi × 1.0s = ~1.7 minuti
|
||||||
|
- 1000 blocchi × 1.0s = ~17 minuti
|
||||||
|
- 10000 blocchi × 1.0s = ~2.8 ore
|
||||||
|
|
||||||
|
## Esportazione Dati
|
||||||
|
|
||||||
|
### Esportazione CSV
|
||||||
|
|
||||||
|
Dopo la scansione, lo script chiederà:
|
||||||
|
```
|
||||||
|
📤 Esportare dati in CSV? (s/n): s
|
||||||
|
```
|
||||||
|
|
||||||
|
Questo genera `p2pk_export.csv` con tutte le colonne del database.
|
||||||
|
|
||||||
|
### Formato CSV
|
||||||
|
|
||||||
|
```csv
|
||||||
|
id,block_height,txid,output_index,scriptpubkey,value_satoshi,timestamp,is_unspent,last_checked
|
||||||
|
1,9,0437cd7f8525ceed2324359c2d0ba26006d92d85,0,4104d46c4968bde02899d2d0d...,5000000000,1231469744,0,1234567890
|
||||||
|
```
|
||||||
|
|
||||||
|
## Collaborazione
|
||||||
|
|
||||||
|
Questo progetto supporta la collaborazione multi-utente:
|
||||||
|
|
||||||
|
### Condivisione Database
|
||||||
|
Il file `.gitignore` **non** esclude:
|
||||||
|
- `*.db` (database SQLite)
|
||||||
|
- `*.csv` (esportazioni)
|
||||||
|
- `*.html` (report)
|
||||||
|
|
||||||
|
Più persone possono:
|
||||||
|
1. Clonare il repository con il database esistente
|
||||||
|
2. Scansionare range di blocchi diversi
|
||||||
|
3. Push/pull per sincronizzare i dati raccolti
|
||||||
|
4. Collaborare per coprire più blocchi velocemente
|
||||||
|
|
||||||
|
### Best Practice per Collaborazione
|
||||||
|
1. Coordinarsi sui range di blocchi da scansionare
|
||||||
|
2. Fare pull prima di iniziare una nuova scansione
|
||||||
|
3. Committare e pushare il database dopo ogni scansione completata
|
||||||
|
4. Usare delay >= 1.0s per non sovraccaricare l'API
|
||||||
|
|
||||||
|
## Come Funziona
|
||||||
|
|
||||||
|
### Rilevamento P2PK
|
||||||
|
|
||||||
|
Lo scanner usa 4 metodi per identificare P2PK:
|
||||||
|
|
||||||
|
1. **Tipo esplicito**: `scriptpubkey_type == 'pubkey'`
|
||||||
|
2. **Lunghezza script**: 67 byte (non compresso) o 35 byte (compresso)
|
||||||
|
3. **Pattern ASM**: `<PUBKEY> OP_CHECKSIG`
|
||||||
|
4. **Pattern HEX**: `41<pubkey>ac` o `21<pubkey>ac`
|
||||||
|
|
||||||
|
### Formato ScriptPubKey
|
||||||
|
|
||||||
|
**P2PK non compresso** (134 caratteri hex):
|
||||||
|
```
|
||||||
|
41 + <65 byte pubkey> + ac
|
||||||
|
```
|
||||||
|
|
||||||
|
**P2PK compresso** (70 caratteri hex):
|
||||||
|
```
|
||||||
|
21 + <33 byte pubkey> + ac
|
||||||
|
```
|
||||||
|
|
||||||
|
### Verifica UTXO
|
||||||
|
|
||||||
|
Per ogni P2PK trovato, lo scanner verifica su mempool.space se l'output è ancora non speso:
|
||||||
|
|
||||||
|
```
|
||||||
|
GET /api/tx/{txid}/outspend/{vout}
|
||||||
|
```
|
||||||
|
|
||||||
|
Risposta:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"spent": false, // true se speso
|
||||||
|
"txid": "...", // txid che ha speso (se spent=true)
|
||||||
|
"vin": 0 // input index (se spent=true)
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Risoluzione Problemi
|
||||||
|
|
||||||
|
### L'ambiente virtuale non si attiva
|
||||||
|
|
||||||
|
**Linux/macOS:**
|
||||||
|
```bash
|
||||||
|
chmod +x .venv/bin/activate
|
||||||
|
source .venv/bin/activate
|
||||||
|
```
|
||||||
|
|
||||||
|
**Windows PowerShell:**
|
||||||
|
```powershell
|
||||||
|
Set-ExecutionPolicy -ExecutionPolicy RemoteSigned -Scope CurrentUser
|
||||||
|
.venv\Scripts\Activate.ps1
|
||||||
|
```
|
||||||
|
|
||||||
|
### Errori API / Timeout
|
||||||
|
|
||||||
|
- Verifica la connessione internet
|
||||||
|
- Aumenta il delay tra richieste: `delay: 2.0`
|
||||||
|
- mempool.space potrebbe essere temporaneamente non disponibile
|
||||||
|
|
||||||
|
### Database locked
|
||||||
|
|
||||||
|
Se più processi accedono al database:
|
||||||
|
```bash
|
||||||
|
# Chiudi tutti gli script Python in esecuzione
|
||||||
|
# Riavvia lo script
|
||||||
|
```
|
||||||
|
|
||||||
|
## Blocchi Interessanti
|
||||||
|
|
||||||
|
I primi blocchi Bitcoin contengono molti P2PK:
|
||||||
|
|
||||||
|
- **Blocchi 1-10000**: Era di Satoshi, quasi tutti P2PK
|
||||||
|
- **Blocchi 10000-100000**: Transizione verso P2PKH
|
||||||
|
- **Blocchi 100000+**: Prevalentemente P2PKH, P2SH, SegWit
|
||||||
|
|
||||||
|
**Consiglio**: Inizia dai blocchi 1-10000 per trovare più P2PK.
|
||||||
|
|
||||||
|
## Crediti
|
||||||
|
|
||||||
|
- API: [mempool.space](https://mempool.space/docs/api/rest)
|
||||||
|
- Bitcoin: [bitcoin.org](https://bitcoin.org)
|
||||||
|
- P2PK Reference: [Bitcoin Wiki](https://en.bitcoin.it/wiki/Transaction)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**⚠️ Ricorda**: Questo è un progetto educativo. Le chiavi pubbliche P2PK non devono essere utilizzate per scopi illeciti.
|
||||||
Binary file not shown.
File diff suppressed because it is too large
Load Diff
@@ -75,6 +75,20 @@ class P2PKDatabaseViewer:
|
|||||||
stats['unspent_value_btc'] = unspent_sat / 100000000.0
|
stats['unspent_value_btc'] = unspent_sat / 100000000.0
|
||||||
stats['unspent_value_sat'] = int(unspent_sat)
|
stats['unspent_value_sat'] = int(unspent_sat)
|
||||||
|
|
||||||
|
# Conta chiavi compresse vs non compresse
|
||||||
|
cursor.execute('SELECT scriptpubkey FROM p2pk_addresses')
|
||||||
|
all_scripts = cursor.fetchall()
|
||||||
|
compressed_count = 0
|
||||||
|
uncompressed_count = 0
|
||||||
|
for (script,) in all_scripts:
|
||||||
|
if script and script.startswith('41') and len(script) == 134:
|
||||||
|
uncompressed_count += 1
|
||||||
|
elif script and script.startswith('21') and len(script) == 70:
|
||||||
|
compressed_count += 1
|
||||||
|
|
||||||
|
stats['compressed_count'] = compressed_count
|
||||||
|
stats['uncompressed_count'] = uncompressed_count
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
print(f"⚠️ Errore nel calcolo statistiche: {e}")
|
print(f"⚠️ Errore nel calcolo statistiche: {e}")
|
||||||
import traceback
|
import traceback
|
||||||
@@ -91,7 +105,9 @@ class P2PKDatabaseViewer:
|
|||||||
'unique_txs': 0,
|
'unique_txs': 0,
|
||||||
'unspent_count': 0,
|
'unspent_count': 0,
|
||||||
'unspent_value_btc': 0.0,
|
'unspent_value_btc': 0.0,
|
||||||
'unspent_value_sat': 0
|
'unspent_value_sat': 0,
|
||||||
|
'compressed_count': 0,
|
||||||
|
'uncompressed_count': 0
|
||||||
}
|
}
|
||||||
|
|
||||||
conn.close()
|
conn.close()
|
||||||
@@ -140,7 +156,8 @@ class P2PKDatabaseViewer:
|
|||||||
|
|
||||||
rows_html = []
|
rows_html = []
|
||||||
for row in p2pk_data:
|
for row in p2pk_data:
|
||||||
pubkey = self.extract_pubkey_from_script(row[4])
|
scriptpubkey = row[4]
|
||||||
|
pubkey = self.extract_pubkey_from_script(scriptpubkey)
|
||||||
txid_short = row[2][:16] if len(row[2]) > 16 else row[2]
|
txid_short = row[2][:16] if len(row[2]) > 16 else row[2]
|
||||||
timestamp_str = datetime.fromtimestamp(row[6]).strftime('%Y-%m-%d %H:%M') if row[6] else 'N/A'
|
timestamp_str = datetime.fromtimestamp(row[6]).strftime('%Y-%m-%d %H:%M') if row[6] else 'N/A'
|
||||||
# row[5] è già in satoshi, lo convertiamo in BTC dividendo per 100000000
|
# row[5] è già in satoshi, lo convertiamo in BTC dividendo per 100000000
|
||||||
@@ -152,8 +169,19 @@ class P2PKDatabaseViewer:
|
|||||||
utxo_status = '🟢 NON SPESO' if is_unspent else '🔴 SPESO'
|
utxo_status = '🟢 NON SPESO' if is_unspent else '🔴 SPESO'
|
||||||
utxo_class = 'unspent' if is_unspent else 'spent'
|
utxo_class = 'unspent' if is_unspent else 'spent'
|
||||||
|
|
||||||
|
# Determina se la chiave è compressa o non compressa
|
||||||
|
if scriptpubkey and scriptpubkey.startswith('41') and len(scriptpubkey) == 134:
|
||||||
|
key_type = 'uncompressed'
|
||||||
|
key_type_badge = '<span class="key-type-badge uncompressed">📜 Non Compressa (65 bytes)</span>'
|
||||||
|
elif scriptpubkey and scriptpubkey.startswith('21') and len(scriptpubkey) == 70:
|
||||||
|
key_type = 'compressed'
|
||||||
|
key_type_badge = '<span class="key-type-badge compressed">📦 Compressa (33 bytes)</span>'
|
||||||
|
else:
|
||||||
|
key_type = 'unknown'
|
||||||
|
key_type_badge = '<span class="key-type-badge unknown">❓ Sconosciuta</span>'
|
||||||
|
|
||||||
row_html = f'''
|
row_html = f'''
|
||||||
<tr class="{utxo_class}">
|
<tr class="{utxo_class} {key_type}">
|
||||||
<td>{row[0]}</td>
|
<td>{row[0]}</td>
|
||||||
<td><span class="block">{row[1]}</span></td>
|
<td><span class="block">{row[1]}</span></td>
|
||||||
<td>
|
<td>
|
||||||
@@ -163,7 +191,8 @@ class P2PKDatabaseViewer:
|
|||||||
<td>{row[3]}</td>
|
<td>{row[3]}</td>
|
||||||
<td>
|
<td>
|
||||||
<div class="pubkey">
|
<div class="pubkey">
|
||||||
{pubkey}
|
{key_type_badge}
|
||||||
|
<div style="margin-top: 5px;">{pubkey}</div>
|
||||||
<button class="copy-btn" onclick="copyToClipboard('{pubkey}')">📋</button>
|
<button class="copy-btn" onclick="copyToClipboard('{pubkey}')">📋</button>
|
||||||
</div>
|
</div>
|
||||||
</td>
|
</td>
|
||||||
@@ -462,6 +491,33 @@ class P2PKDatabaseViewer:
|
|||||||
tr.spent {{
|
tr.spent {{
|
||||||
opacity: 0.7;
|
opacity: 0.7;
|
||||||
}}
|
}}
|
||||||
|
|
||||||
|
.key-type-badge {{
|
||||||
|
padding: 4px 10px;
|
||||||
|
border-radius: 15px;
|
||||||
|
font-size: 0.8em;
|
||||||
|
font-weight: bold;
|
||||||
|
display: inline-block;
|
||||||
|
margin-bottom: 5px;
|
||||||
|
}}
|
||||||
|
|
||||||
|
.key-type-badge.uncompressed {{
|
||||||
|
background: #cfe2ff;
|
||||||
|
color: #084298;
|
||||||
|
border: 1px solid #9ec5fe;
|
||||||
|
}}
|
||||||
|
|
||||||
|
.key-type-badge.compressed {{
|
||||||
|
background: #d1e7dd;
|
||||||
|
color: #0a3622;
|
||||||
|
border: 1px solid #a3cfbb;
|
||||||
|
}}
|
||||||
|
|
||||||
|
.key-type-badge.unknown {{
|
||||||
|
background: #f8d7da;
|
||||||
|
color: #58151c;
|
||||||
|
border: 1px solid #f1aeb5;
|
||||||
|
}}
|
||||||
</style>
|
</style>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
@@ -505,6 +561,14 @@ class P2PKDatabaseViewer:
|
|||||||
<div class="label" style="color: #0c5460;">💎 Valore Non Speso</div>
|
<div class="label" style="color: #0c5460;">💎 Valore Non Speso</div>
|
||||||
<div class="value" style="color: #0c5460;">{stats['unspent_value_btc']:.8f} BTC</div>
|
<div class="value" style="color: #0c5460;">{stats['unspent_value_btc']:.8f} BTC</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="stat-card" style="background: #cfe2ff;">
|
||||||
|
<div class="label" style="color: #084298;">📜 Chiavi Non Compresse</div>
|
||||||
|
<div class="value" style="color: #084298;">{stats['uncompressed_count']:,}</div>
|
||||||
|
</div>
|
||||||
|
<div class="stat-card" style="background: #d1e7dd;">
|
||||||
|
<div class="label" style="color: #0a3622;">📦 Chiavi Compresse</div>
|
||||||
|
<div class="value" style="color: #0a3622;">{stats['compressed_count']:,}</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="content">
|
<div class="content">
|
||||||
@@ -518,6 +582,12 @@ class P2PKDatabaseViewer:
|
|||||||
<button class="filter-btn spent-filter" onclick="filterByStatus('spent')">🔴 Solo Spesi</button>
|
<button class="filter-btn spent-filter" onclick="filterByStatus('spent')">🔴 Solo Spesi</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="filter-buttons" style="margin-top: 10px;">
|
||||||
|
<button class="filter-btn" onclick="filterByKeyType('all-keys')">🔑 Tutti i Tipi</button>
|
||||||
|
<button class="filter-btn" onclick="filterByKeyType('uncompressed')">📜 Solo Non Compresse</button>
|
||||||
|
<button class="filter-btn" onclick="filterByKeyType('compressed')">📦 Solo Compresse</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
{self._generate_table_html(p2pk_data)}
|
{self._generate_table_html(p2pk_data)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -581,6 +651,32 @@ class P2PKDatabaseViewer:
|
|||||||
}}
|
}}
|
||||||
}}
|
}}
|
||||||
}}
|
}}
|
||||||
|
|
||||||
|
function filterByKeyType(keyType) {{
|
||||||
|
const table = document.getElementById('dataTable');
|
||||||
|
if (!table) return;
|
||||||
|
|
||||||
|
const tr = table.getElementsByTagName('tr');
|
||||||
|
|
||||||
|
// Aggiorna stato pulsanti del gruppo key-type
|
||||||
|
event.target.parentElement.querySelectorAll('.filter-btn').forEach(btn => {{
|
||||||
|
btn.classList.remove('active');
|
||||||
|
}});
|
||||||
|
event.target.classList.add('active');
|
||||||
|
|
||||||
|
// Filtra righe in base al tipo di chiave
|
||||||
|
for (let i = 1; i < tr.length; i++) {{
|
||||||
|
const row = tr[i];
|
||||||
|
|
||||||
|
if (keyType === 'all-keys') {{
|
||||||
|
row.style.display = '';
|
||||||
|
}} else if (keyType === 'uncompressed') {{
|
||||||
|
row.style.display = row.classList.contains('uncompressed') ? '' : 'none';
|
||||||
|
}} else if (keyType === 'compressed') {{
|
||||||
|
row.style.display = row.classList.contains('compressed') ? '' : 'none';
|
||||||
|
}}
|
||||||
|
}}
|
||||||
|
}}
|
||||||
</script>
|
</script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|||||||
Reference in New Issue
Block a user