5
5
module StellarCoreCfg
6
6
7
7
open FSharp.Data
8
+ open Logging
8
9
open Nett
10
+ open System
11
+ open System.Collections
9
12
open System.Text .RegularExpressions
10
13
open StellarCoreSet
11
14
open StellarNetworkCfg
@@ -141,6 +144,7 @@ type StellarCoreCfg =
141
144
networkPassphrase: NetworkPassphrase
142
145
nodeSeed: KeyPair
143
146
nodeIsValidator: bool
147
+ homeDomain: string option
144
148
runStandalone: bool
145
149
image: string
146
150
preferredPeers: PeerDnsName list
@@ -156,6 +160,7 @@ type StellarCoreCfg =
156
160
unsafeQuorum: bool
157
161
failureSafety: int
158
162
quorumSet: QuorumSet
163
+ forceOldStyleLeaderElection: bool
159
164
historyNodes: Map < PeerShortName , PeerDnsName >
160
165
historyGetCommands: Map < PeerShortName , string >
161
166
localHistory: bool
@@ -208,6 +213,10 @@ type StellarCoreCfg =
208
213
t.Add( " COMMANDS" , logLevelCommands) |> ignore
209
214
t.Add( " CATCHUP_COMPLETE" , self.catchupMode = CatchupComplete) |> ignore
210
215
216
+ match self.homeDomain with
217
+ | None -> ()
218
+ | Some hd -> t.Add( " NODE_HOME_DOMAIN" , hd) |> ignore
219
+
211
220
match self.network.missionContext.peerReadingCapacity, self.network.missionContext.peerFloodCapacity with
212
221
| None, None -> ()
213
222
| Some read, Some flood ->
@@ -306,6 +315,9 @@ type StellarCoreCfg =
306
315
t.Add( " QUORUM_INTERSECTION_CHECKER" , false ) |> ignore
307
316
t.Add( " MANUAL_CLOSE" , self.manualClose) |> ignore
308
317
318
+ if self.forceOldStyleLeaderElection then
319
+ t.Add( " FORCE_OLD_STYLE_LEADER_ELECTION" , true ) |> ignore
320
+
309
321
let invList =
310
322
match self.invariantChecks with
311
323
| AllInvariants -> [ " .*" ]
@@ -321,7 +333,7 @@ type StellarCoreCfg =
321
333
| Some duration -> t.Add( " ARTIFICIALLY_SET_SURVEY_PHASE_DURATION_FOR_TESTING" , duration) |> ignore
322
334
323
335
// Add tables (and subtables, recursively) for qsets.
324
- let rec addQsetAt ( label : string ) ( qs : QuorumSet ) =
336
+ let rec addExplicitQsetAt ( label : string ) ( qs : ExplicitQuorumSet ) =
325
337
let validators : string array =
326
338
Map.toArray qs.validators
327
339
|> Array.map ( fun ( n : PeerShortName , k : KeyPair ) -> sprintf " %s %s " k.Address n.StringName)
@@ -333,11 +345,54 @@ type StellarCoreCfg =
333
345
| None -> ()
334
346
| Some ( pct) -> innerTab.Add( " THRESHOLD_PERCENT" , pct) |> ignore
335
347
336
- Array.iteri ( fun ( i : int ) ( qs : QuorumSet ) -> addQsetAt ( sprintf " %s .sub%d " label i) qs) qs.innerQuorumSets
348
+ Array.iteri
349
+ ( fun ( i : int ) ( qs : ExplicitQuorumSet ) -> addExplicitQsetAt ( sprintf " %s .sub%d " label i) qs)
350
+ qs.innerQuorumSets
351
+
352
+ let homeDomainToTable ( homeDomain : HomeDomain ) =
353
+ let ret = Toml.Create()
354
+ ret.Add( " HOME_DOMAIN" , homeDomain.name) |> ignore
355
+
356
+ ret.Add( " QUALITY" , homeDomain.quality.ToString() |> String.map Char.ToUpper)
357
+ |> ignore
358
+
359
+ ret
360
+
361
+ let autoValidatorToTable ( autoValidator : AutoValidator ) =
362
+ let ret = Toml.Create()
363
+ ret.Add( " NAME" , autoValidator.name.StringName) |> ignore
364
+ ret.Add( " HOME_DOMAIN" , autoValidator.homeDomain) |> ignore
365
+ ret.Add( " PUBLIC_KEY" , autoValidator.keys.Address) |> ignore
366
+
367
+ match Map.tryFind autoValidator.name self.historyGetCommands with
368
+ | Some cmd -> ret.Add( " HISTORY" , cmd) |> ignore
369
+ | None ->
370
+ match Map.tryFind autoValidator.name self.historyNodes with
371
+ | Some dnsName -> ret.Add( " HISTORY" , Map.find " get" ( remoteHist dnsName)) |> ignore
372
+ | None -> ()
337
373
338
- addQsetAt " QUORUM_SET " self.quorumSet
374
+ ret
339
375
340
376
let localTab = t.Add( " HISTORY" , Toml.Create(), TomlObjectFactory.RequireTomlObject()) .Added
377
+
378
+ match self.quorumSet with
379
+ | ExplicitQuorumSet qs ->
380
+ addExplicitQsetAt " QUORUM_SET" qs
381
+
382
+ for historyNode in self.historyNodes do
383
+ localTab.Add( historyNode.Key.StringName, remoteHist historyNode.Value) |> ignore
384
+
385
+ for historyGetCommand in self.historyGetCommands do
386
+ localTab.Add( historyGetCommand.Key.StringName, getHist historyGetCommand.Value)
387
+ |> ignore
388
+ | AutoQuorumSet qs ->
389
+ let homeDomainsTab = t.Add( " HOME_DOMAINS" , ([]: IDictionary list)) .Added
390
+ List.iter ( fun hd -> homeDomainsTab.Add( homeDomainToTable hd) |> ignore) qs.homeDomains
391
+ let validatorsTab = t.Add( " VALIDATORS" , ([]: IDictionary list)) .Added
392
+ // Filter out local node
393
+ let validators = List.filter ( fun ( v : AutoValidator ) -> v.keys <> self.nodeSeed) qs.validators
394
+ List.iter ( fun v -> validatorsTab.Add( autoValidatorToTable v) |> ignore) validators
395
+
341
396
// When simulateApplyWeight = Some _, stellar-core sets MODE_STORES_HISTORY
342
397
// which is used for simulations that only test consensus.
343
398
// In such cases, we should not pass put and mkdir commands.
@@ -351,12 +406,6 @@ type StellarCoreCfg =
351
406
)
352
407
|> ignore
353
408
354
- for historyNode in self.historyNodes do
355
- localTab.Add( historyNode.Key.StringName, remoteHist historyNode.Value) |> ignore
356
-
357
- for historyGetCommand in self.historyGetCommands do
358
- localTab.Add( historyGetCommand.Key.StringName, getHist historyGetCommand.Value)
359
- |> ignore
360
409
361
410
t
362
411
@@ -425,17 +474,60 @@ type NetworkCfg with
425
474
self.CoreSetList |> List.map processCoreSet |> List.concat |> Map.ofList
426
475
427
476
member self.QuorumSet ( o : CoreSetOptions ) : QuorumSet =
428
- let ofNameKeyList ( nks : ( PeerShortName * KeyPair ) array ) ( threshold : int option ) : QuorumSet =
429
- { thresholdPercent = threshold
430
- validators = Map.ofArray nks
431
- innerQuorumSets = [||] }
477
+ let toExplicitQSet ( nks : ( PeerShortName * KeyPair ) array ) ( threshold : int option ) : QuorumSet =
478
+ LogInfo " Using explicit quorum set configuration"
479
+
480
+ ExplicitQuorumSet
481
+ { thresholdPercent = threshold
482
+ validators = Map.ofArray nks
483
+ innerQuorumSets = [||] }
484
+
485
+ let toAutoQSet ( nks : ( PeerShortName * KeyPair ) list ) ( homeDomain : string ) =
486
+ LogInfo " Using auto quorum set configuration"
487
+ let homeDomains = [ { name = homeDomain; quality = High } ]
488
+
489
+ let validators =
490
+ List.map ( fun ( n : PeerShortName , k ) -> { name = n; homeDomain = homeDomain; keys = k }) nks
491
+
492
+ AutoQuorumSet { homeDomains = homeDomains; validators = validators }
493
+
494
+ // Generate a QuorumSet from an array of (PeerShortName, KeyPair) pairs.
495
+ // Produces a simple flat qset of nodes. Uses auto quorum set
496
+ // configuration if possible.
497
+ let simpleQuorum ( nks : ( PeerShortName * KeyPair ) array ) =
498
+ match o.homeDomain with
499
+ | Some hd ->
500
+ if nks.Length >= 3 then
501
+ // There are enough validators to use auto quorum set config
502
+ toAutoQSet ( List.ofArray nks) hd
503
+ else if o.requireAutoQset then
504
+ failwith " Auto quorum set configuration requires at least 3 validators"
505
+ else
506
+ // Fall back on manual quorum set configuration
507
+ toExplicitQSet nks None
508
+ | None ->
509
+ if o.requireAutoQset then
510
+ failwith " Auto quorum set configuration requires a home domain"
511
+ else
512
+ toExplicitQSet nks None
513
+
514
+ let checkAutoQSetIncompatability ( mode : string ) =
515
+ if o.requireAutoQset then
516
+ failwithf " Auto quorum set configuration is incompatible with %s " mode
517
+ else
518
+ ()
432
519
433
520
match o.quorumSet with
434
- | AllPeersQuorum -> ofNameKeyList ( self.GetNameKeyListAll()) None
435
- | CoreSetQuorum ( ns) -> ofNameKeyList ( self.GetNameKeyList [ ns ]) None
436
- | CoreSetQuorumList ( q) -> ofNameKeyList ( self.GetNameKeyList q) None
437
- | CoreSetQuorumListWithThreshold ( q, t) -> ofNameKeyList ( self.GetNameKeyList q) ( Some( t))
438
- | ExplicitQuorum ( e) -> e
521
+ | AllPeersQuorum -> simpleQuorum ( self.GetNameKeyListAll())
522
+ | CoreSetQuorum ( ns) -> simpleQuorum ( self.GetNameKeyList [ ns ])
523
+ | CoreSetQuorumList ( q) -> simpleQuorum ( self.GetNameKeyList q)
524
+ | CoreSetQuorumListWithThreshold ( q, t) ->
525
+ checkAutoQSetIncompatability " CoreSetQuorumListWithThreshold"
526
+ toExplicitQSet ( self.GetNameKeyList q) ( Some( t))
527
+ | ExplicitQuorum ( e) ->
528
+ checkAutoQSetIncompatability " ExplicitQuorum"
529
+ ExplicitQuorumSet e
530
+ | AutoQuorum q -> AutoQuorumSet q
439
531
440
532
member self.HistoryNodes ( o : CoreSetOptions ) : Map < PeerShortName , PeerDnsName > =
441
533
match o.historyNodes, o.quorumSet with
@@ -471,6 +563,7 @@ type NetworkCfg with
471
563
networkPassphrase = self.networkPassphrase
472
564
nodeSeed = KeyPair.Random()
473
565
nodeIsValidator = false
566
+ homeDomain = None
474
567
runStandalone = false
475
568
image = opts.image
476
569
preferredPeers = self.PreferredPeers opts
@@ -486,6 +579,7 @@ type NetworkCfg with
486
579
unsafeQuorum = opts.unsafeQuorum
487
580
failureSafety = 0
488
581
quorumSet = self.QuorumSet opts
582
+ forceOldStyleLeaderElection = opts.forceOldStyleLeaderElection
489
583
historyNodes = self.HistoryNodes opts
490
584
historyGetCommands = opts.historyGetCommands
491
585
localHistory = opts.localHistory
@@ -503,6 +597,7 @@ type NetworkCfg with
503
597
networkPassphrase = self.networkPassphrase
504
598
nodeSeed = c.keys.[ i]
505
599
nodeIsValidator = c.options.validate
600
+ homeDomain = c.options.homeDomain
506
601
runStandalone = false
507
602
image = c.options.image
508
603
preferredPeers =
@@ -524,6 +619,7 @@ type NetworkCfg with
524
619
unsafeQuorum = c.options.unsafeQuorum
525
620
failureSafety = 0
526
621
quorumSet = self.QuorumSet c.options
622
+ forceOldStyleLeaderElection = c.options.forceOldStyleLeaderElection
527
623
historyNodes = self.HistoryNodes c.options
528
624
historyGetCommands = c.options.historyGetCommands
529
625
localHistory = c.options.localHistory
0 commit comments