1
1
use std:: {
2
2
collections:: BTreeMap , fs, io:: Write , net:: SocketAddr , path:: PathBuf , str:: FromStr , sync:: Arc ,
3
3
} ;
4
-
4
+ use std :: fs :: File ;
5
5
use anyhow:: { anyhow, Context } ;
6
6
use base64:: Engine ;
7
7
use bdk:: {
@@ -18,21 +18,12 @@ use jsonrpsee::{core::async_trait, proc_macros::rpc, server::Server, types::Erro
18
18
use log:: info;
19
19
use serde:: { Deserialize , Deserializer , Serialize , Serializer } ;
20
20
use spacedb:: { encode:: SubTreeEncoder , tx:: ProofType } ;
21
- use spaces_protocol:: {
22
- bitcoin,
23
- bitcoin:: {
24
- bip32:: Xpriv ,
25
- secp256k1,
26
- Network :: { Regtest , Testnet } ,
27
- OutPoint ,
28
- } ,
29
- constants:: ChainAnchor ,
30
- hasher:: { BaseHash , KeyHasher , OutpointKey , SpaceKey } ,
31
- prepare:: DataSource ,
32
- slabel:: SLabel ,
33
- validate:: TxChangeSet ,
34
- Bytes , FullSpaceOut , SpaceOut ,
35
- } ;
21
+ use spaces_protocol:: { bitcoin, bitcoin:: {
22
+ bip32:: Xpriv ,
23
+ secp256k1,
24
+ Network :: { Regtest , Testnet } ,
25
+ OutPoint ,
26
+ } , constants:: ChainAnchor , hasher:: { BaseHash , KeyHasher , OutpointKey , SpaceKey } , prepare:: DataSource , slabel:: SLabel , validate:: TxChangeSet , Bytes , Covenant , FullSpaceOut , SpaceOut } ;
36
27
use spaces_wallet:: {
37
28
bdk_wallet as bdk, bdk_wallet:: template:: Bip86 , bitcoin:: hashes:: Hash as BitcoinHash ,
38
29
export:: WalletExport , Balance , DoubleUtxo , Listing , SpacesWallet , WalletConfig ,
@@ -55,6 +46,7 @@ use crate::{
55
46
WalletResponse ,
56
47
} ,
57
48
} ;
49
+ use crate :: sync:: { TRUST_ANCHORS_COUNT , COMMIT_BLOCK_INTERVAL } ;
58
50
59
51
pub ( crate ) type Responder < T > = oneshot:: Sender < T > ;
60
52
@@ -77,9 +69,8 @@ pub struct TrustAnchor {
77
69
serialize_with = "serialize_hash" ,
78
70
deserialize_with = "deserialize_hash"
79
71
) ]
80
- pub state_root : spaces_protocol:: hasher:: Hash ,
81
- pub block_hash : BlockHash ,
82
- pub block_height : u32 ,
72
+ pub root : spaces_protocol:: hasher:: Hash ,
73
+ pub block : ChainAnchor
83
74
}
84
75
85
76
pub enum ChainStateCommand {
@@ -94,7 +85,6 @@ pub enum ChainStateCommand {
94
85
hash : SpaceKey ,
95
86
resp : Responder < anyhow:: Result < Option < FullSpaceOut > > > ,
96
87
} ,
97
-
98
88
GetSpaceout {
99
89
outpoint : OutPoint ,
100
90
resp : Responder < anyhow:: Result < Option < SpaceOut > > > ,
@@ -129,14 +119,15 @@ pub enum ChainStateCommand {
129
119
} ,
130
120
ProveSpaceout {
131
121
outpoint : OutPoint ,
122
+ oldest : bool ,
132
123
resp : Responder < anyhow:: Result < ProofResult > > ,
133
124
} ,
134
125
ProveSpaceOutpoint {
135
126
space_or_hash : String ,
136
127
resp : Responder < anyhow:: Result < ProofResult > > ,
137
128
} ,
138
- GetAnchor {
139
- resp : Responder < anyhow:: Result < TrustAnchor > > ,
129
+ GetTrustAnchors {
130
+ resp : Responder < anyhow:: Result < Vec < TrustAnchor > > > ,
140
131
} ,
141
132
}
142
133
@@ -256,16 +247,16 @@ pub trait Rpc {
256
247
async fn verify_listing ( & self , listing : Listing ) -> Result < ( ) , ErrorObjectOwned > ;
257
248
258
249
#[ method( name = "provespaceout" ) ]
259
- async fn prove_spaceout ( & self , outpoint : OutPoint ) -> Result < ProofResult , ErrorObjectOwned > ;
250
+ async fn prove_spaceout ( & self , outpoint : OutPoint , oldest : bool ) -> Result < ProofResult , ErrorObjectOwned > ;
260
251
261
252
#[ method( name = "provespaceoutpoint" ) ]
262
253
async fn prove_space_outpoint (
263
254
& self ,
264
255
space_or_hash : & str ,
265
256
) -> Result < ProofResult , ErrorObjectOwned > ;
266
257
267
- #[ method( name = "getanchor " ) ]
268
- async fn get_anchor ( & self ) -> Result < TrustAnchor , ErrorObjectOwned > ;
258
+ #[ method( name = "gettrustanchors " ) ]
259
+ async fn get_trust_anchors ( & self ) -> Result < Vec < TrustAnchor > , ErrorObjectOwned > ;
269
260
270
261
#[ method( name = "walletlisttransactions" ) ]
271
262
async fn wallet_list_transactions (
@@ -384,13 +375,15 @@ pub struct RpcServerImpl {
384
375
}
385
376
386
377
#[ derive( Clone , Serialize , Deserialize ) ]
387
- pub struct ProofResult (
378
+ pub struct ProofResult {
379
+ root : Bytes ,
388
380
#[ serde(
389
381
serialize_with = "serialize_base64" ,
390
382
deserialize_with = "deserialize_base64"
391
383
) ]
392
- Vec < u8 > ,
393
- ) ;
384
+ proof : Vec < u8 > ,
385
+
386
+ }
394
387
395
388
fn serialize_base64 < S > ( bytes : & Vec < u8 > , serializer : S ) -> Result < S :: Ok , S :: Error >
396
389
where
@@ -960,9 +953,9 @@ impl RpcServer for RpcServerImpl {
960
953
. map_err ( |error| ErrorObjectOwned :: owned ( -1 , error. to_string ( ) , None :: < String > ) )
961
954
}
962
955
963
- async fn prove_spaceout ( & self , outpoint : OutPoint ) -> Result < ProofResult , ErrorObjectOwned > {
956
+ async fn prove_spaceout ( & self , outpoint : OutPoint , oldest : bool ) -> Result < ProofResult , ErrorObjectOwned > {
964
957
self . store
965
- . prove_spaceout ( outpoint)
958
+ . prove_spaceout ( outpoint, oldest )
966
959
. await
967
960
. map_err ( |error| ErrorObjectOwned :: owned ( -1 , error. to_string ( ) , None :: < String > ) )
968
961
}
@@ -977,9 +970,9 @@ impl RpcServer for RpcServerImpl {
977
970
. map_err ( |error| ErrorObjectOwned :: owned ( -1 , error. to_string ( ) , None :: < String > ) )
978
971
}
979
972
980
- async fn get_anchor ( & self ) -> Result < TrustAnchor , ErrorObjectOwned > {
973
+ async fn get_trust_anchors ( & self ) -> Result < Vec < TrustAnchor > , ErrorObjectOwned > {
981
974
self . store
982
- . get_anchor ( )
975
+ . get_trust_anchors ( )
983
976
. await
984
977
. map_err ( |error| ErrorObjectOwned :: owned ( -1 , error. to_string ( ) , None :: < String > ) )
985
978
}
@@ -1124,6 +1117,7 @@ impl AsyncChainState {
1124
1117
pub async fn handle_command (
1125
1118
client : & reqwest:: Client ,
1126
1119
rpc : & BitcoinRpc ,
1120
+ anchors_path : & Option < PathBuf > ,
1127
1121
chain_state : & mut LiveSnapshot ,
1128
1122
block_index : & mut Option < LiveSnapshot > ,
1129
1123
cmd : ChainStateCommand ,
@@ -1196,11 +1190,11 @@ impl AsyncChainState {
1196
1190
msg. message . as_slice ( ) ,
1197
1191
& msg. signature ,
1198
1192
)
1199
- . map ( |_| ( ) ) ,
1193
+ . map ( |_| ( ) ) ,
1200
1194
) ;
1201
1195
}
1202
- ChainStateCommand :: ProveSpaceout { outpoint, resp } => {
1203
- _ = resp. send ( Self :: handle_prove_spaceout ( chain_state, outpoint) ) ;
1196
+ ChainStateCommand :: ProveSpaceout { oldest , outpoint, resp } => {
1197
+ _ = resp. send ( Self :: handle_prove_spaceout ( chain_state, outpoint, oldest ) ) ;
1204
1198
}
1205
1199
ChainStateCommand :: ProveSpaceOutpoint {
1206
1200
space_or_hash,
@@ -1211,21 +1205,30 @@ impl AsyncChainState {
1211
1205
& space_or_hash,
1212
1206
) ) ;
1213
1207
}
1214
- ChainStateCommand :: GetAnchor { resp } => {
1215
- _ = resp. send ( Self :: handle_get_anchor ( chain_state) ) ;
1208
+ ChainStateCommand :: GetTrustAnchors { resp } => {
1209
+ _ = resp. send ( Self :: handle_get_anchor ( anchors_path , chain_state) ) ;
1216
1210
}
1217
1211
}
1218
1212
}
1219
1213
1220
- fn handle_get_anchor ( state : & mut LiveSnapshot ) -> anyhow:: Result < TrustAnchor > {
1214
+ fn handle_get_anchor ( anchors_path : & Option < PathBuf > , state : & mut LiveSnapshot ) -> anyhow:: Result < Vec < TrustAnchor > > {
1215
+ if let Some ( anchors_path) = anchors_path {
1216
+ let anchors: Vec < TrustAnchor > = serde_json:: from_reader ( File :: open ( anchors_path)
1217
+ . or_else ( |e| Err ( anyhow ! ( "Could not open anchors file: {}" , e) ) ) ?)
1218
+ . or_else ( |e| Err ( anyhow ! ( "Could not read anchors file: {}" , e) ) ) ?;
1219
+ return Ok ( anchors) ;
1220
+ }
1221
+
1221
1222
let snapshot = state. inner ( ) ?;
1222
1223
let root = snapshot. compute_root ( ) ?;
1223
1224
let meta: ChainAnchor = snapshot. metadata ( ) . try_into ( ) ?;
1224
- Ok ( TrustAnchor {
1225
- state_root : root,
1226
- block_hash : meta. hash ,
1227
- block_height : meta. height ,
1228
- } )
1225
+ Ok ( vec ! [ TrustAnchor {
1226
+ root,
1227
+ block: ChainAnchor {
1228
+ hash: meta. hash,
1229
+ height: meta. height,
1230
+ } ,
1231
+ } ] )
1229
1232
}
1230
1233
1231
1234
fn handle_prove_space_outpoint (
@@ -1236,37 +1239,88 @@ impl AsyncChainState {
1236
1239
let snapshot = state. inner ( ) ?;
1237
1240
1238
1241
// warm up hash cache
1239
- _ = snapshot. compute_root ( ) ?;
1242
+ let root = snapshot. compute_root ( ) ?;
1240
1243
let proof = snapshot. prove ( & [ key. into ( ) ] , ProofType :: Standard ) ?;
1241
1244
1242
1245
let mut buf = vec ! [ 0u8 ; 4096 ] ;
1243
1246
let offset = proof. write_to_slice ( & mut buf) ?;
1244
1247
buf. truncate ( offset) ;
1245
1248
1246
- Ok ( ProofResult ( buf) )
1249
+ Ok ( ProofResult { proof : buf, root : Bytes :: new ( root. to_vec ( ) ) } )
1250
+ }
1251
+
1252
+ /// Determines the optimal snapshot block height for creating a Merkle proof.
1253
+ ///
1254
+ /// This function finds a suitable historical snapshot that:
1255
+ /// 1. Is not older than when the space was last updated.
1256
+ /// 2. Falls within [TRUST_ANCHORS_COUNT] range (for proof verification)
1257
+ /// 3. Skips the oldest trust anchors to prevent the proof from becoming stale too quickly.
1258
+ ///
1259
+ /// Parameters:
1260
+ /// - last_update: Block height when the space was last updated
1261
+ /// - tip: Current blockchain tip height
1262
+ ///
1263
+ /// Returns: Target block height aligned to [COMMIT_BLOCK_INTERVAL]
1264
+ fn compute_target_snapshot ( last_update : u32 , tip : u32 ) -> u32 {
1265
+ const SAFETY_MARGIN : u32 = 8 ; // Skip oldest trust anchors to prevent proof staleness
1266
+ const USABLE_ANCHORS : u32 = TRUST_ANCHORS_COUNT - SAFETY_MARGIN ;
1267
+
1268
+ // Align block heights to commit intervals
1269
+ let last_update_aligned = last_update. div_ceil ( COMMIT_BLOCK_INTERVAL )
1270
+ * COMMIT_BLOCK_INTERVAL ;
1271
+ let current_tip_aligned = ( tip / COMMIT_BLOCK_INTERVAL )
1272
+ * COMMIT_BLOCK_INTERVAL ;
1273
+
1274
+ // Calculate the oldest allowed snapshot while maintaining safety margin
1275
+ let lookback_window = USABLE_ANCHORS * COMMIT_BLOCK_INTERVAL ;
1276
+ let oldest_allowed_snapshot = current_tip_aligned. saturating_sub ( lookback_window) ;
1277
+
1278
+ // Choose the most recent of last update or oldest allowed snapshot
1279
+ // to ensure both data freshness and proof verifiability
1280
+ std:: cmp:: max ( last_update_aligned, oldest_allowed_snapshot)
1247
1281
}
1248
1282
1249
1283
fn handle_prove_spaceout (
1250
1284
state : & mut LiveSnapshot ,
1251
1285
outpoint : OutPoint ,
1286
+ oldest : bool ,
1252
1287
) -> anyhow:: Result < ProofResult > {
1253
1288
let key = OutpointKey :: from_outpoint :: < Sha256 > ( outpoint) ;
1254
- let snapshot = state. inner ( ) ?;
1255
1289
1256
- // warm up hash cache
1257
- _ = snapshot. compute_root ( ) ?;
1258
- let proof = snapshot. prove ( & [ key. into ( ) ] , ProofType :: Standard ) ?;
1290
+ let proof = if oldest {
1291
+ let spaceout = match state. get_spaceout ( & outpoint) ? {
1292
+ Some ( spaceot) => spaceot,
1293
+ None => return Err ( anyhow ! ( "Cannot find older proofs for a non-existent utxo (try with oldest: false)" ) ) ,
1294
+ } ;
1295
+ let target_snapshot = match spaceout. space . as_ref ( ) {
1296
+ None => return Ok ( ProofResult { proof : vec ! [ ] , root : Bytes :: new ( vec ! [ ] ) } ) ,
1297
+ Some ( space) => match space. covenant {
1298
+ Covenant :: Transfer { expire_height, .. } => {
1299
+ let tip = state. tip . read ( ) . expect ( "read lock" ) . height ;
1300
+ let last_update = expire_height. saturating_sub ( spaces_protocol:: constants:: RENEWAL_INTERVAL ) ;
1301
+ Self :: compute_target_snapshot ( last_update, tip)
1302
+ } ,
1303
+ _ => return Err ( anyhow ! ( "Cannot find older proofs for a non-registered space (try with oldest: false)" ) ) ,
1304
+ }
1305
+ } ;
1306
+ state. prove_with_snapshot ( & [ key. into ( ) ] , target_snapshot) ?
1307
+ } else {
1308
+ let snapshot = state. inner ( ) ?;
1309
+ snapshot. prove ( & [ key. into ( ) ] , ProofType :: Standard ) ?
1310
+ } ;
1259
1311
1312
+ let root = proof. compute_root ( ) ?. to_vec ( ) ;
1260
1313
let mut buf = vec ! [ 0u8 ; 4096 ] ;
1261
1314
let offset = proof. write_to_slice ( & mut buf) ?;
1262
1315
buf. truncate ( offset) ;
1263
1316
1264
- Ok ( ProofResult ( buf) )
1317
+ Ok ( ProofResult { proof : buf, root : Bytes :: new ( root ) } )
1265
1318
}
1266
1319
1267
1320
pub async fn handler (
1268
1321
client : & reqwest:: Client ,
1269
1322
rpc : BitcoinRpc ,
1323
+ anchors_path : Option < PathBuf > ,
1270
1324
mut chain_state : LiveSnapshot ,
1271
1325
mut block_index : Option < LiveSnapshot > ,
1272
1326
mut rx : mpsc:: Receiver < ChainStateCommand > ,
@@ -1278,7 +1332,7 @@ impl AsyncChainState {
1278
1332
break ;
1279
1333
}
1280
1334
Some ( cmd) = rx. recv( ) => {
1281
- Self :: handle_command( client, & rpc, & mut chain_state, & mut block_index, cmd) . await ;
1335
+ Self :: handle_command( client, & rpc, & anchors_path , & mut chain_state, & mut block_index, cmd) . await ;
1282
1336
}
1283
1337
}
1284
1338
}
@@ -1310,10 +1364,10 @@ impl AsyncChainState {
1310
1364
resp_rx. await ?
1311
1365
}
1312
1366
1313
- pub async fn prove_spaceout ( & self , outpoint : OutPoint ) -> anyhow:: Result < ProofResult > {
1367
+ pub async fn prove_spaceout ( & self , outpoint : OutPoint , oldest : bool ) -> anyhow:: Result < ProofResult > {
1314
1368
let ( resp, resp_rx) = oneshot:: channel ( ) ;
1315
1369
self . sender
1316
- . send ( ChainStateCommand :: ProveSpaceout { outpoint, resp } )
1370
+ . send ( ChainStateCommand :: ProveSpaceout { outpoint, oldest , resp } )
1317
1371
. await ?;
1318
1372
resp_rx. await ?
1319
1373
}
@@ -1329,10 +1383,10 @@ impl AsyncChainState {
1329
1383
resp_rx. await ?
1330
1384
}
1331
1385
1332
- pub async fn get_anchor ( & self ) -> anyhow:: Result < TrustAnchor > {
1386
+ pub async fn get_trust_anchors ( & self ) -> anyhow:: Result < Vec < TrustAnchor > > {
1333
1387
let ( resp, resp_rx) = oneshot:: channel ( ) ;
1334
1388
self . sender
1335
- . send ( ChainStateCommand :: GetAnchor { resp } )
1389
+ . send ( ChainStateCommand :: GetTrustAnchors { resp } )
1336
1390
. await ?;
1337
1391
resp_rx. await ?
1338
1392
}
0 commit comments