@@ -46,9 +46,6 @@ public PostgresIndexer(
46
46
CancellationTokenSource cts ;
47
47
Task _indexerLoop ;
48
48
Task _watchdogLoop ;
49
- Node _Node ;
50
- Channel < object > _Channel ;
51
- Channel < Block > _DownloadedBlocks ;
52
49
53
50
// This one will check if the indexer is "stuck" and disconnect the node if it is the case
54
51
async Task WatchdogLoop ( )
@@ -58,14 +55,14 @@ async Task WatchdogLoop()
58
55
try
59
56
{
60
57
await Task . Delay ( TimeSpan . FromMinutes ( 5.0 ) , cancellationToken ) ;
61
- var height = await SeemsStuck ( cancellationToken ) ;
62
- if ( height is null )
58
+ var lastBlock = await SeemsStuck ( cancellationToken ) ;
59
+ if ( lastBlock is null )
63
60
goto wait ;
64
61
await Task . Delay ( TimeSpan . FromMinutes ( 2.0 ) , cancellationToken ) ;
65
- var height2 = await SeemsStuck ( cancellationToken ) ;
66
- if ( height != height2 )
62
+ var lastBlock2 = await SeemsStuck ( cancellationToken ) ;
63
+ if ( lastBlock != lastBlock2 )
67
64
goto wait ;
68
- _Node ? . DisconnectAsync ( $ "Sync seems stuck at height { height . Value } , restarting the connection.") ;
65
+ _Connection ? . Dispose ( $ "Sync seems stuck after block { lastBlock . Hash } ( { lastBlock . Hash } ) , restarting the connection.") ;
69
66
goto wait ;
70
67
}
71
68
catch when ( cts . Token . IsCancellationRequested )
@@ -80,18 +77,17 @@ async Task WatchdogLoop()
80
77
end : ;
81
78
}
82
79
83
- async Task < long ? > SeemsStuck ( CancellationToken cancellationToken )
80
+ async Task < SlimChainedBlock > SeemsStuck ( CancellationToken cancellationToken )
84
81
{
85
82
if ( State is not ( BitcoinDWaiterState . NBXplorerSynching or BitcoinDWaiterState . Ready ) ||
86
- SyncHeight is not long syncHeight ||
83
+ lastIndexedBlock is not { } lastBlock ||
87
84
GetConnectedClient ( ) is not RPCClient rpc )
88
85
{
89
86
return null ;
90
87
}
88
+
91
89
var blockchainInfo = await rpc . GetBlockchainInfoAsyncEx ( cancellationToken ) ;
92
- if ( Math . Min ( blockchainInfo . Headers , blockchainInfo . Blocks ) > syncHeight )
93
- return syncHeight ;
94
- return null ;
90
+ return blockchainInfo . BestBlockHash != lastBlock . Hash ? lastBlock : null ;
95
91
}
96
92
97
93
async Task IndexerLoop ( )
@@ -121,24 +117,53 @@ async Task IndexerLoop()
121
117
}
122
118
}
123
119
120
+ class Connection : IDisposable
121
+ {
122
+ public Channel < Object > Events ;
123
+ public Channel < Block > Blocks ;
124
+ public Node Node ;
125
+ public Connection ( Node node )
126
+ {
127
+ Node = node ;
128
+ Events = Channel . CreateUnbounded < object > ( new ( ) { AllowSynchronousContinuations = false } ) ;
129
+ Blocks = Channel . CreateUnbounded < Block > ( new ( ) { AllowSynchronousContinuations = false } ) ;
130
+ }
131
+ bool _Disposed = false ;
132
+
133
+ public void Dispose ( )
134
+ {
135
+ Dispose ( null ) ;
136
+ }
137
+ public void Dispose ( string reason )
138
+ {
139
+ if ( _Disposed )
140
+ return ;
141
+ Node . DisconnectAsync ( reason ) ;
142
+ Events . Writer . TryComplete ( ) ;
143
+ Blocks . Writer . TryComplete ( ) ;
144
+ _Disposed = true ;
145
+ }
146
+ }
147
+ Connection _Connection ;
124
148
private async Task IndexerLoopCore ( CancellationToken token )
125
149
{
126
- await ConnectNode ( token , true ) ;
127
- await foreach ( var item in _Channel . Reader . ReadAllAsync ( token ) )
150
+ await ConnectNode ( token ) ;
151
+ var connection = _Connection ;
152
+ await foreach ( var item in connection . Events . Reader . ReadAllAsync ( token ) )
128
153
{
129
154
await using var conn = await ConnectionFactory . CreateConnectionHelper ( Network ) ;
130
155
if ( item is PullBlocks pb )
131
156
{
132
- var headers = ConsolidatePullBlocks ( _Channel . Reader , pb ) ;
157
+ var headers = ConsolidatePullBlocks ( connection . Events . Reader , pb ) ;
133
158
foreach ( var batch in headers . Chunk ( maxinflight ) )
134
159
{
135
- _ = _Node . SendMessageAsync (
160
+ _ = connection . Node . SendMessageAsync (
136
161
new GetDataPayload (
137
- batch . Select ( b => new InventoryVector ( _Node . AddSupportedOptions ( InventoryType . MSG_BLOCK ) , b . GetHash ( ) )
162
+ batch . Select ( b => new InventoryVector ( connection . Node . AddSupportedOptions ( InventoryType . MSG_BLOCK ) , b . GetHash ( ) )
138
163
) . ToArray ( ) ) ) ;
139
164
var remaining = batch . Select ( b => b . GetHash ( ) ) . ToHashSet ( ) ;
140
165
List < Block > unorderedBlocks = new List < Block > ( ) ;
141
- await foreach ( var block in _DownloadedBlocks . Reader . ReadAllAsync ( token ) )
166
+ await foreach ( var block in connection . Blocks . Reader . ReadAllAsync ( token ) )
142
167
{
143
168
if ( ! remaining . Remove ( block . Header . GetHash ( ) ) )
144
169
continue ;
@@ -188,17 +213,14 @@ private async Task IndexerLoopCore(CancellationToken token)
188
213
}
189
214
}
190
215
await SaveProgress ( conn ) ;
191
- await UpdateState ( ) ;
216
+ await UpdateState ( connection . Node ) ;
192
217
}
193
- await AskNextHeaders ( token ) ;
194
- }
195
- if ( item is NodeDisconnected )
196
- {
197
- await ConnectNode ( token , false ) ;
218
+ if ( connection . Node . State != NodeState . HandShaked )
219
+ await AskNextHeaders ( connection . Node , token ) ;
198
220
}
199
221
if ( item is Transaction tx )
200
222
{
201
- var txs = PullTransactions ( _Channel . Reader , tx ) ;
223
+ var txs = PullTransactions ( connection . Events . Reader , tx ) ;
202
224
await SaveMatches ( conn , txs , null , true ) ;
203
225
}
204
226
}
@@ -255,15 +277,8 @@ private IList<BlockHeader> ConsolidatePullBlocks(ChannelReader<object> reader, P
255
277
}
256
278
257
279
258
- private async Task ConnectNode ( CancellationToken token , bool forceRestart )
280
+ private async Task ConnectNode ( CancellationToken token )
259
281
{
260
- if ( _Node is not null )
261
- {
262
- if ( ! forceRestart && _Node . State == NodeState . HandShaked )
263
- return ;
264
- _Node . DisconnectAsync ( "Restarting" ) ;
265
- _Node = null ;
266
- }
267
282
State = BitcoinDWaiterState . NotStarted ;
268
283
using ( var handshakeTimeout = CancellationTokenSource . CreateLinkedTokenSource ( token ) )
269
284
{
@@ -344,35 +359,35 @@ private async Task ConnectNode(CancellationToken token, bool forceRestart)
344
359
State = BitcoinDWaiterState . NBXplorerSynching ;
345
360
// Refresh the NetworkInfo that may have become different while it was synching.
346
361
NetworkInfo = await RPCClient . GetNetworkInfoAsync ( ) ;
347
- _Node = node ;
348
- _Channel ? . Writer . Complete ( ) ;
349
- _Channel = Channel . CreateUnbounded < object > ( ) ;
350
- _DownloadedBlocks ? . Writer . Complete ( ) ;
351
- _DownloadedBlocks = Channel . CreateUnbounded < Block > ( ) ;
362
+
363
+ _Connection ? . Dispose ( "Creating new connection" ) ;
364
+ _Connection = new Connection ( node ) ;
352
365
node . MessageReceived += Node_MessageReceived ;
353
366
node . Disconnected += Node_Disconnected ;
354
-
355
- var locator = await AskNextHeaders ( token ) ;
367
+ var locator = await AskNextHeaders ( node , token ) ;
356
368
lastIndexedBlock = await Repository . GetLastIndexedSlimChainedBlock ( locator ) ;
357
369
if ( lastIndexedBlock is null )
358
370
{
359
371
var locatorTip = await RPCClient . GetBlockHeaderAsyncEx ( locator . Blocks [ 0 ] , token ) ;
360
372
lastIndexedBlock = locatorTip ? . ToSlimChainedBlock ( ) ;
361
373
}
362
- await UpdateState ( ) ;
374
+ await UpdateState ( node ) ;
363
375
}
364
376
}
365
377
366
-
367
378
bool firstConnect = true ;
368
- private async Task < BlockLocator > AskNextHeaders ( CancellationToken token )
379
+ private async Task < BlockLocator > AskNextHeaders ( Node node , CancellationToken token )
369
380
{
370
381
var indexProgress = await Repository . GetIndexProgress ( ) ;
371
382
if ( indexProgress is null )
372
383
{
373
384
indexProgress = await GetDefaultCurrentLocation ( token ) ;
374
385
}
375
- await _Node . SendMessageAsync ( new GetHeadersPayload ( indexProgress ) ) ;
386
+ foreach ( var block in indexProgress . Blocks )
387
+ {
388
+ Logger . LogInformation ( $ "Asking for block { block } ") ;
389
+ }
390
+ await node . SendMessageAsync ( new GetHeadersPayload ( indexProgress ) ) ;
376
391
return indexProgress ;
377
392
}
378
393
@@ -391,8 +406,10 @@ private async Task SaveProgress(DbConnectionHelper conn)
391
406
await Repository . SetIndexProgress ( conn . Connection , locator ) ;
392
407
}
393
408
394
- private async Task UpdateState ( )
409
+ private async Task UpdateState ( Node node )
395
410
{
411
+ if ( node . State != NodeState . HandShaked )
412
+ return ;
396
413
var blockchainInfo = await RPCClient . GetBlockchainInfoAsyncEx ( ) ;
397
414
if ( blockchainInfo . IsSynching ( Network ) )
398
415
{
@@ -508,18 +525,16 @@ private async Task SaveMatches(DbConnectionHelper conn, List<Transaction> transa
508
525
509
526
SlimChainedBlock lastIndexedBlock ;
510
527
record PullBlocks ( IList < BlockHeader > headers ) ;
511
- record NodeDisconnected ( ) ;
512
528
private void Node_MessageReceived ( Node node , IncomingMessage message )
513
529
{
514
- var channel = _Channel ;
515
- var downloadedBlocks = _DownloadedBlocks ;
530
+ var connection = _Connection ;
516
531
if ( message . Message . Payload is HeadersPayload h && h . Headers . Count != 0 )
517
532
{
518
- channel . Writer . TryWrite ( new PullBlocks ( h . Headers ) ) ;
533
+ connection . Events . Writer . TryWrite ( new PullBlocks ( h . Headers ) ) ;
519
534
}
520
535
else if ( message . Message . Payload is BlockPayload b )
521
536
{
522
- downloadedBlocks . Writer . TryWrite ( b . Object ) ;
537
+ connection . Blocks . Writer . TryWrite ( b . Object ) ;
523
538
}
524
539
else if ( message . Message . Payload is InvPayload invs )
525
540
{
@@ -535,41 +550,17 @@ private void Node_MessageReceived(Node node, IncomingMessage message)
535
550
{
536
551
node . SendMessageAsync ( data ) ;
537
552
}
538
- // DOGE coin doing doge things forget we want header first sync... reboot the connection
539
- else
540
- {
541
- if ( invs . Inventory . Where ( t => t . Type . HasFlag ( InventoryType . MSG_BLOCK ) ) . Any ( ) )
542
- {
543
- node . DisconnectAsync ( "Not sending headers first anymore" ) ;
544
- }
545
- }
546
553
}
547
554
else if ( message . Message . Payload is TxPayload tx )
548
555
{
549
- channel . Writer . TryWrite ( tx . Object ) ;
556
+ connection . Events . Writer . TryWrite ( tx . Object ) ;
550
557
}
551
558
}
552
559
553
560
private void Node_Disconnected ( Node node )
554
561
{
555
- var channel = _Channel ;
556
- if ( node . DisconnectReason . Reason != "Restarting" )
557
- {
558
- if ( ! cts . IsCancellationRequested )
559
- {
560
- var exception = node . DisconnectReason . Exception ? . Message ;
561
- if ( ! string . IsNullOrEmpty ( exception ) )
562
- exception = $ " ({ exception } )";
563
- else
564
- exception = String . Empty ;
565
- Logger . LogWarning ( $ "Node disconnected for reason: { node . DisconnectReason . Reason } { exception } ") ;
566
- }
567
- channel . Writer . TryWrite ( new NodeDisconnected ( ) ) ;
568
- }
569
- else
570
- {
571
- Logger . LogInformation ( $ "Restarting node connection...") ;
572
- }
562
+ Logger . LogInformation ( $ "Node disconnected ({ node . DisconnectReason . Reason } )") ;
563
+ _Connection ? . Dispose ( ) ;
573
564
node . MessageReceived -= Node_MessageReceived ;
574
565
node . Disconnected -= Node_Disconnected ;
575
566
State = BitcoinDWaiterState . NotStarted ;
@@ -589,12 +580,11 @@ public async Task StartAsync(CancellationToken cancellationToken)
589
580
public async Task StopAsync ( CancellationToken cancellationToken )
590
581
{
591
582
cts ? . Cancel ( ) ;
592
- _Channel . Writer . Complete ( ) ;
583
+ _Connection ? . Dispose ( "NBXplorer stopping..." ) ;
593
584
if ( _indexerLoop is not null )
594
585
await _indexerLoop ;
595
586
if ( _watchdogLoop is not null )
596
587
await _watchdogLoop ;
597
- _Node ? . DisconnectAsync ( ) ;
598
588
}
599
589
public NBXplorerNetwork Network => network ;
600
590
0 commit comments