trac-msb 0.2.4 → 0.2.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (239) hide show
  1. package/.dockerignore +16 -0
  2. package/.github/workflows/acceptance-tests.yml +7 -0
  3. package/.github/workflows/publish.yml +40 -0
  4. package/.github/workflows/{CI.yml → unit-tests.yml} +1 -1
  5. package/README.md +175 -50
  6. package/docker-compose.yml +16 -0
  7. package/dockerfile +41 -0
  8. package/docs/fee_distribution.md +89 -0
  9. package/msb.mjs +12 -14
  10. package/package.json +8 -4
  11. package/rpc/constants.mjs +4 -1
  12. package/rpc/handlers.mjs +109 -66
  13. package/rpc/routes/v1.mjs +3 -1
  14. package/rpc/rpc_services.js +126 -0
  15. package/rpc/utils/confirmedParameter.mjs +17 -0
  16. package/rpc/utils/url.mjs +38 -0
  17. package/src/core/network/Network.js +27 -10
  18. package/src/core/network/identity/NetworkWalletFactory.js +78 -0
  19. package/src/core/network/services/ConnectionManager.js +2 -2
  20. package/src/core/network/services/ValidatorObserverService.js +7 -4
  21. package/src/core/state/State.js +28 -22
  22. package/src/index.js +197 -385
  23. package/src/utils/cliCommands.js +280 -0
  24. package/src/utils/constants.js +3 -1
  25. package/tests/acceptance/v1/account/account.test.mjs +123 -0
  26. package/tests/acceptance/v1/balance/balance.test.mjs +55 -0
  27. package/tests/acceptance/v1/broadcast-transaction/broadcast-transaction.test.mjs +111 -0
  28. package/tests/acceptance/v1/confirmed-length/confirmed-length.test.mjs +19 -0
  29. package/tests/acceptance/v1/fee/fee.test.mjs +11 -0
  30. package/tests/acceptance/v1/rpc.test.mjs +62 -291
  31. package/tests/acceptance/v1/tx/tx.test.mjs +98 -0
  32. package/tests/acceptance/v1/tx-details/tx-details.test.mjs +195 -0
  33. package/tests/acceptance/v1/tx-hashes/tx-hashes.test.mjs +72 -0
  34. package/tests/acceptance/v1/tx-payloads-bulk/tx-payloads-bulk.test.mjs +27 -0
  35. package/tests/acceptance/v1/txv/txv.test.mjs +11 -0
  36. package/tests/acceptance/v1/unconfirmed-length/unconfirmed-length.test.mjs +11 -0
  37. package/tests/helpers/StateNetworkFactory.js +157 -0
  38. package/tests/helpers/autobaseTestHelpers.js +369 -0
  39. package/tests/helpers/createTestSignature.js +12 -0
  40. package/tests/helpers/transactionPayloads.mjs +78 -0
  41. package/tests/unit/network/NetworkWalletFactory.test.js +156 -0
  42. package/tests/unit/state/apply/addAdmin/addAdminHappyPathScenario.js +38 -0
  43. package/tests/unit/state/apply/addAdmin/addAdminScenarioHelpers.js +273 -0
  44. package/tests/unit/state/apply/addAdmin/adminEntryEncodingFailureScenario.js +30 -0
  45. package/tests/unit/state/apply/addAdmin/adminEntryExistsScenario.js +78 -0
  46. package/tests/unit/state/apply/addAdmin/nodeEntryInitializationFailureScenario.js +30 -0
  47. package/tests/unit/state/apply/addAdmin/nonBootstrapNodeScenario.js +68 -0
  48. package/tests/unit/state/apply/addAdmin/state.apply.addAdmin.test.js +155 -0
  49. package/tests/unit/state/apply/addIndexer/addIndexerHappyPathScenario.js +39 -0
  50. package/tests/unit/state/apply/addIndexer/addIndexerMultipleIndexersInTheNetworkScenario.js +167 -0
  51. package/tests/unit/state/apply/addIndexer/addIndexerPretenderAlreadyIndexerScenario.js +21 -0
  52. package/tests/unit/state/apply/addIndexer/addIndexerPretenderNotWriterScenario.js +21 -0
  53. package/tests/unit/state/apply/addIndexer/addIndexerRemoveAndReAddScenario.js +186 -0
  54. package/tests/unit/state/apply/addIndexer/addIndexerScenarioHelpers.js +445 -0
  55. package/tests/unit/state/apply/addIndexer/addIndexerWriterKeyAlreadyRegisteredScenario.js +32 -0
  56. package/tests/unit/state/apply/addIndexer/state.apply.addIndexer.test.js +297 -0
  57. package/tests/unit/state/apply/addWriter/addWriterHappyPathScenario.js +41 -0
  58. package/tests/unit/state/apply/addWriter/addWriterInvalidValidatorSignatureScenario.js +32 -0
  59. package/tests/unit/state/apply/addWriter/addWriterNewWkScenario.js +149 -0
  60. package/tests/unit/state/apply/addWriter/addWriterRequesterAlreadyWriterScenario.js +21 -0
  61. package/tests/unit/state/apply/addWriter/addWriterRequesterBalanceInsufficientScenario.js +21 -0
  62. package/tests/unit/state/apply/addWriter/addWriterRequesterEntryDecodeFailureScenario.js +19 -0
  63. package/tests/unit/state/apply/addWriter/addWriterRequesterEntryMissingScenario.js +19 -0
  64. package/tests/unit/state/apply/addWriter/addWriterRequesterIndexerScenario.js +21 -0
  65. package/tests/unit/state/apply/addWriter/addWriterRequesterNotWhitelistedScenario.js +21 -0
  66. package/tests/unit/state/apply/addWriter/addWriterScenarioHelpers.js +757 -0
  67. package/tests/unit/state/apply/addWriter/addWriterStakeBalanceUpdateFailureScenario.js +50 -0
  68. package/tests/unit/state/apply/addWriter/addWriterStakeInsufficientBalanceScenario.js +29 -0
  69. package/tests/unit/state/apply/addWriter/addWriterStakeInvalidBalanceScenario.js +29 -0
  70. package/tests/unit/state/apply/addWriter/addWriterStakeInvalidEntryScenario.js +21 -0
  71. package/tests/unit/state/apply/addWriter/addWriterStakeStakedBalanceFailureScenario.js +37 -0
  72. package/tests/unit/state/apply/addWriter/addWriterStakeSubtractFailureScenario.js +42 -0
  73. package/tests/unit/state/apply/addWriter/addWriterValidatorRewardScenario.js +105 -0
  74. package/tests/unit/state/apply/addWriter/addWriterWriterKeyMismatchScenario.js +54 -0
  75. package/tests/unit/state/apply/addWriter/addWriterWriterKeyOwnershipScenario.js +54 -0
  76. package/tests/unit/state/apply/addWriter/addWriterZeroWriterKeyScenario.js +29 -0
  77. package/tests/unit/state/apply/addWriter/state.apply.addWriter.test.js +309 -0
  78. package/tests/unit/state/apply/adminRecovery/adminRecoveryHappyPathScenario.js +30 -0
  79. package/tests/unit/state/apply/adminRecovery/adminRecoveryScenarioHelpers.js +866 -0
  80. package/tests/unit/state/apply/adminRecovery/state.apply.adminRecovery.test.js +439 -0
  81. package/tests/unit/state/apply/appendWhitelist/appendWhitelistBanAndReapplyScenario.js +78 -0
  82. package/tests/unit/state/apply/appendWhitelist/appendWhitelistExistingReaderHappyPathScenario.js +98 -0
  83. package/tests/unit/state/apply/appendWhitelist/appendWhitelistFeeAfterDisableScenario.js +66 -0
  84. package/tests/unit/state/apply/appendWhitelist/appendWhitelistHappyPathScenario.js +55 -0
  85. package/tests/unit/state/apply/appendWhitelist/appendWhitelistInsufficientAdminBalanceScenario.js +103 -0
  86. package/tests/unit/state/apply/appendWhitelist/appendWhitelistNodeAlreadyWhitelistedScenario.js +60 -0
  87. package/tests/unit/state/apply/appendWhitelist/appendWhitelistScenarioHelpers.js +191 -0
  88. package/tests/unit/state/apply/appendWhitelist/state.apply.appendWhitelist.test.js +220 -0
  89. package/tests/unit/state/apply/balanceInitialization/balanceInitializationHappyPathScenario.js +82 -0
  90. package/tests/unit/state/apply/balanceInitialization/balanceInitializationScenarioHelpers.js +106 -0
  91. package/tests/unit/state/apply/balanceInitialization/invalidAmountScenario.js +45 -0
  92. package/tests/unit/state/apply/balanceInitialization/nodeEntryBalanceUpdateFailureScenario.js +81 -0
  93. package/tests/unit/state/apply/balanceInitialization/state.apply.balanceInitialization.test.js +189 -0
  94. package/tests/unit/state/apply/banValidator/banValidatorBanAndReWhitelistScenario.js +155 -0
  95. package/tests/unit/state/apply/banValidator/banValidatorHappyPathScenario.js +36 -0
  96. package/tests/unit/state/apply/banValidator/banValidatorScenarioHelpers.js +534 -0
  97. package/tests/unit/state/apply/banValidator/banValidatorSequentialBansScenario.js +74 -0
  98. package/tests/unit/state/apply/banValidator/banValidatorTargetDecodeFailureScenario.js +19 -0
  99. package/tests/unit/state/apply/banValidator/banValidatorTargetIndexerScenario.js +32 -0
  100. package/tests/unit/state/apply/banValidator/banValidatorTargetNodeEntryMissingScenario.js +19 -0
  101. package/tests/unit/state/apply/banValidator/banValidatorTargetRoleUpdateFailureScenario.js +19 -0
  102. package/tests/unit/state/apply/banValidator/banValidatorWhitelistedNonWriterScenario.js +38 -0
  103. package/tests/unit/state/apply/banValidator/banValidatorWhitelistedZeroBalanceScenario.js +91 -0
  104. package/tests/unit/state/apply/banValidator/banValidatorWithdrawFailureScenario.js +19 -0
  105. package/tests/unit/state/apply/banValidator/state.apply.banValidator.test.js +266 -0
  106. package/tests/unit/state/apply/bootstrapDeployment/bootstrapDeploymentDuplicateRegistrationScenario.js +142 -0
  107. package/tests/unit/state/apply/bootstrapDeployment/bootstrapDeploymentHappyPathScenario.js +26 -0
  108. package/tests/unit/state/apply/bootstrapDeployment/bootstrapDeploymentIncompleteOperationScenario.js +94 -0
  109. package/tests/unit/state/apply/bootstrapDeployment/bootstrapDeploymentInvalidDeploymentEntryScenario.js +37 -0
  110. package/tests/unit/state/apply/bootstrapDeployment/bootstrapDeploymentMultipleBootstrapScenario.js +86 -0
  111. package/tests/unit/state/apply/bootstrapDeployment/bootstrapDeploymentScenarioHelpers.js +344 -0
  112. package/tests/unit/state/apply/bootstrapDeployment/invalidValidatorNodeEntryScenario.js +57 -0
  113. package/tests/unit/state/apply/bootstrapDeployment/state.apply.bootstrapDeployment.test.js +429 -0
  114. package/tests/unit/state/apply/common/access-control/adminConsistencyMismatchScenario.js +119 -0
  115. package/tests/unit/state/apply/common/access-control/adminEntryDecodeFailureScenario.js +130 -0
  116. package/tests/unit/state/apply/common/access-control/adminEntryExistsScenario.js +93 -0
  117. package/tests/unit/state/apply/common/access-control/adminEntryMissingScenario.js +108 -0
  118. package/tests/unit/state/apply/common/access-control/adminOnlyGuardScenario.js +126 -0
  119. package/tests/unit/state/apply/common/access-control/adminPublicKeyDecodeFailureScenario.js +120 -0
  120. package/tests/unit/state/apply/common/access-control/roleAccessOperationValidationScenario.js +50 -0
  121. package/tests/unit/state/apply/common/adminControlOperationValidationScenario.js +56 -0
  122. package/tests/unit/state/apply/common/balances/adminEntryUpdateFailureScenario.js +52 -0
  123. package/tests/unit/state/apply/common/balances/base/requesterBalanceScenarioBase.js +197 -0
  124. package/tests/unit/state/apply/common/balances/feeDecodeFailureScenario.js +52 -0
  125. package/tests/unit/state/apply/common/balances/requesterBalanceDecodeFailureScenario.js +15 -0
  126. package/tests/unit/state/apply/common/balances/requesterBalanceFeeApplicationFailureScenario.js +11 -0
  127. package/tests/unit/state/apply/common/balances/requesterBalanceInsufficientScenario.js +15 -0
  128. package/tests/unit/state/apply/common/balances/requesterBalanceUpdateFailureScenario.js +11 -0
  129. package/tests/unit/state/apply/common/balances/validatorEntryRewardFailureScenario.js +11 -0
  130. package/tests/unit/state/apply/common/balances/validatorEntryUpdateFailureScenario.js +11 -0
  131. package/tests/unit/state/apply/common/balances/validatorNodeEntryDecodeFailureScenario.js +40 -0
  132. package/tests/unit/state/apply/common/base/OperationValidationScenarioBase.js +114 -0
  133. package/tests/unit/state/apply/common/commonScenarioHelper.js +103 -0
  134. package/tests/unit/state/apply/common/indexer/indexerNodeEntryDecodeFailureScenario.js +36 -0
  135. package/tests/unit/state/apply/common/indexer/indexerNodeEntryMissingScenario.js +36 -0
  136. package/tests/unit/state/apply/common/indexer/indexerRoleUpdateFailureScenario.js +29 -0
  137. package/tests/unit/state/apply/common/indexer/indexerSequenceStateInvalidScenario.js +66 -0
  138. package/tests/unit/state/apply/common/invalidMessageComponentValidationScenario.js +84 -0
  139. package/tests/unit/state/apply/common/nodeEntryInitializationFailureScenario.js +47 -0
  140. package/tests/unit/state/apply/common/operationAlreadyAppliedScenario.js +85 -0
  141. package/tests/unit/state/apply/common/payload-structure/addressWithInvalidPublicKeyScenario.js +52 -0
  142. package/tests/unit/state/apply/common/payload-structure/initializationDisabledScenario.js +49 -0
  143. package/tests/unit/state/apply/common/payload-structure/invalidAddressValidationScenario.js +73 -0
  144. package/tests/unit/state/apply/common/payload-structure/invalidHashValidationScenario.js +71 -0
  145. package/tests/unit/state/apply/common/payload-structure/invalidPayloadValidationScenario.js +31 -0
  146. package/tests/unit/state/apply/common/payload-structure/invalidSignatureValidationScenario.js +142 -0
  147. package/tests/unit/state/apply/common/payload-structure/partialOperationValidationScenario.js +87 -0
  148. package/tests/unit/state/apply/common/requester/requesterNodeEntryBufferMissingScenario.js +70 -0
  149. package/tests/unit/state/apply/common/requester/requesterNodeEntryDecodeFailureScenario.js +72 -0
  150. package/tests/unit/state/apply/common/requester/requesterNodeEntryMissingScenario.js +36 -0
  151. package/tests/unit/state/apply/common/requesterAddressValidationScenario.js +44 -0
  152. package/tests/unit/state/apply/common/requesterPublicKeyValidationScenario.js +25 -0
  153. package/tests/unit/state/apply/common/transactionValidityMismatchScenario.js +98 -0
  154. package/tests/unit/state/apply/common/validatorConsistency/base/validatorConsistencyScenarioBase.js +201 -0
  155. package/tests/unit/state/apply/common/validatorConsistency/validatorEntryDecodeFailureScenario.js +17 -0
  156. package/tests/unit/state/apply/common/validatorConsistency/validatorEntryMissingScenario.js +44 -0
  157. package/tests/unit/state/apply/common/validatorConsistency/validatorInactiveScenario.js +19 -0
  158. package/tests/unit/state/apply/common/validatorConsistency/validatorWriterKeyMismatchScenario.js +18 -0
  159. package/tests/unit/state/apply/common/validatorEntryValidation/base/validatorEntryValidationScenarioBase.js +314 -0
  160. package/tests/unit/state/apply/common/validatorEntryValidation/validatorEntryInvalidBalanceScenario.js +18 -0
  161. package/tests/unit/state/apply/common/writerKeyExistsValidationScenario.js +43 -0
  162. package/tests/unit/state/apply/disableInitialization/disableInitializationAlreadyDisabledScenario.js +53 -0
  163. package/tests/unit/state/apply/disableInitialization/disableInitializationHappyPathScenario.js +24 -0
  164. package/tests/unit/state/apply/disableInitialization/disableInitializationScenarioHelpers.js +197 -0
  165. package/tests/unit/state/apply/disableInitialization/state.apply.disableInitialization.test.js +161 -0
  166. package/tests/unit/state/apply/missing-tests.md +18 -0
  167. package/tests/unit/state/apply/removeIndexer/removeIndexerHappyPathScenario.js +58 -0
  168. package/tests/unit/state/apply/removeIndexer/removeIndexerReAddAndRemoveAgainScenario.js +98 -0
  169. package/tests/unit/state/apply/removeIndexer/removeIndexerRemoveMultipleIndexersScenario.js +167 -0
  170. package/tests/unit/state/apply/removeIndexer/removeIndexerScenarioHelpers.js +428 -0
  171. package/tests/unit/state/apply/removeIndexer/removeIndexerTargetNotIndexerScenario.js +22 -0
  172. package/tests/unit/state/apply/removeIndexer/removeIndexerWriterKeyMissingScenario.js +20 -0
  173. package/tests/unit/state/apply/removeIndexer/state.apply.removeIndexer.test.js +291 -0
  174. package/tests/unit/state/apply/removeWriter/removeWriterAndAddWriterAgainScenario.js +87 -0
  175. package/tests/unit/state/apply/removeWriter/removeWriterHappyPathScenario.js +38 -0
  176. package/tests/unit/state/apply/removeWriter/removeWriterInvalidValidatorSignatureScenario.js +32 -0
  177. package/tests/unit/state/apply/removeWriter/removeWriterRequesterBalanceInsufficientScenario.js +21 -0
  178. package/tests/unit/state/apply/removeWriter/removeWriterRequesterEntryDecodeFailureScenario.js +19 -0
  179. package/tests/unit/state/apply/removeWriter/removeWriterRequesterEntryMissingScenario.js +19 -0
  180. package/tests/unit/state/apply/removeWriter/removeWriterRequesterIndexerScenario.js +21 -0
  181. package/tests/unit/state/apply/removeWriter/removeWriterRequesterNotWriterScenario.js +21 -0
  182. package/tests/unit/state/apply/removeWriter/removeWriterRequesterRoleUpdateFailureScenario.js +19 -0
  183. package/tests/unit/state/apply/removeWriter/removeWriterScenarioHelpers.js +344 -0
  184. package/tests/unit/state/apply/removeWriter/removeWriterThroughWriterValidatorScenario.js +113 -0
  185. package/tests/unit/state/apply/removeWriter/removeWriterUnstakeFailureScenario.js +33 -0
  186. package/tests/unit/state/apply/removeWriter/removeWriterWriterKeyMismatchScenario.js +21 -0
  187. package/tests/unit/state/apply/removeWriter/removeWriterWriterKeyOwnershipScenario.js +26 -0
  188. package/tests/unit/state/apply/removeWriter/removeWriterWriterKeyRegistryMissingScenario.js +22 -0
  189. package/tests/unit/state/apply/removeWriter/state.apply.removeWriter.test.js +307 -0
  190. package/tests/unit/state/apply/state.apply.test.js +24 -0
  191. package/tests/unit/state/apply/transfer/state.apply.transfer.test.js +819 -0
  192. package/tests/unit/state/apply/transfer/transferContractSchemaValidationScenario.js +22 -0
  193. package/tests/unit/state/apply/transfer/transferDoubleSpendAcrossValidatorsScenario.js +137 -0
  194. package/tests/unit/state/apply/transfer/transferDoubleSpendSameBatchScenario.js +63 -0
  195. package/tests/unit/state/apply/transfer/transferDoubleSpendSingleValidatorScenario.js +67 -0
  196. package/tests/unit/state/apply/transfer/transferExistingRecipientAmountScenario.js +31 -0
  197. package/tests/unit/state/apply/transfer/transferExistingRecipientZeroAmountScenario.js +31 -0
  198. package/tests/unit/state/apply/transfer/transferHandlerGuardScenarios.js +22 -0
  199. package/tests/unit/state/apply/transfer/transferHappyPathScenario.js +8 -0
  200. package/tests/unit/state/apply/transfer/transferInvalidIncomingDataScenario.js +66 -0
  201. package/tests/unit/state/apply/transfer/transferNewRecipientAmountScenario.js +31 -0
  202. package/tests/unit/state/apply/transfer/transferNewRecipientZeroAmountScenario.js +31 -0
  203. package/tests/unit/state/apply/transfer/transferScenarioHelpers.js +1167 -0
  204. package/tests/unit/state/apply/transfer/transferSelfTransferAmountScenario.js +38 -0
  205. package/tests/unit/state/apply/transfer/transferSelfTransferZeroAmountScenario.js +38 -0
  206. package/tests/unit/state/apply/transfer/transferValidatorRecipientAmountScenario.js +38 -0
  207. package/tests/unit/state/apply/transfer/transferValidatorRecipientZeroAmountScenario.js +38 -0
  208. package/tests/unit/state/apply/txOperation/state.apply.txOperation.test.js +318 -0
  209. package/tests/unit/state/apply/txOperation/txOperationBootstrapNotRegisteredScenario.js +70 -0
  210. package/tests/unit/state/apply/txOperation/txOperationDifferentValidatorCreatorHappyPathScenario.js +23 -0
  211. package/tests/unit/state/apply/txOperation/txOperationInvalidDeploymentEntryScenario.js +48 -0
  212. package/tests/unit/state/apply/txOperation/txOperationInvalidFeeAmountScenario.js +39 -0
  213. package/tests/unit/state/apply/txOperation/txOperationInvalidSubnetCreatorAddressScenario.js +46 -0
  214. package/tests/unit/state/apply/txOperation/txOperationRequesterCreatorHappyPathScenario.js +21 -0
  215. package/tests/unit/state/apply/txOperation/txOperationScenarioHelpers.js +429 -0
  216. package/tests/unit/state/apply/txOperation/txOperationStandardHappyPathScenario.js +21 -0
  217. package/tests/unit/state/apply/txOperation/txOperationTransferFeeAddCreatorBalanceFailureScenario.js +26 -0
  218. package/tests/unit/state/apply/txOperation/txOperationTransferFeeAddValidatorBalanceFailureScenario.js +25 -0
  219. package/tests/unit/state/apply/txOperation/txOperationTransferFeeAddValidatorBonusFailureScenario.js +27 -0
  220. package/tests/unit/state/apply/txOperation/txOperationTransferFeeDecodeCreatorEntryScenario.js +18 -0
  221. package/tests/unit/state/apply/txOperation/txOperationTransferFeeDecodeRequesterEntryScenario.js +17 -0
  222. package/tests/unit/state/apply/txOperation/txOperationTransferFeeDecodeValidatorEntryScenario.js +31 -0
  223. package/tests/unit/state/apply/txOperation/txOperationTransferFeeGuardBypassScenario.js +49 -0
  224. package/tests/unit/state/apply/txOperation/txOperationTransferFeeGuardScenarioFactory.js +92 -0
  225. package/tests/unit/state/apply/txOperation/txOperationTransferFeeInsufficientRequesterBalanceScenario.js +28 -0
  226. package/tests/unit/state/apply/txOperation/txOperationTransferFeeInvalidCreatorBalanceScenario.js +29 -0
  227. package/tests/unit/state/apply/txOperation/txOperationTransferFeeInvalidRequesterBalanceScenario.js +28 -0
  228. package/tests/unit/state/apply/txOperation/txOperationTransferFeeInvalidRequesterEntryScenario.js +17 -0
  229. package/tests/unit/state/apply/txOperation/txOperationTransferFeeInvalidValidatorBalanceScenario.js +33 -0
  230. package/tests/unit/state/apply/txOperation/txOperationTransferFeeMissingCreatorEntryScenario.js +18 -0
  231. package/tests/unit/state/apply/txOperation/txOperationTransferFeeSubtractFailureScenario.js +25 -0
  232. package/tests/unit/state/apply/txOperation/txOperationTransferFeeUpdateCreatorBalanceFailureScenario.js +26 -0
  233. package/tests/unit/state/apply/txOperation/txOperationTransferFeeUpdateFailureScenario.js +25 -0
  234. package/tests/unit/state/apply/txOperation/txOperationTransferFeeUpdateValidatorBalanceFailureScenario.js +26 -0
  235. package/tests/unit/state/apply/txOperation/txOperationTransferFeeUpdateValidatorBonusFailureScenario.js +27 -0
  236. package/tests/unit/state/apply/txOperation/txOperationValidatorCreatorHappyPathScenario.js +21 -0
  237. package/tests/unit/state/stateModule.test.js +1 -0
  238. package/tests/unit/state/stateTestUtils.js +5 -1
  239. package/.env +0 -3
@@ -0,0 +1,19 @@
1
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
2
+ import {
3
+ setupBanValidatorScenario,
4
+ buildBanValidatorPayload,
5
+ assertBanValidatorFailureState,
6
+ applyWithTargetNodeEntryRemoval
7
+ } from './banValidatorScenarioHelpers.js';
8
+
9
+ export default function banValidatorTargetNodeEntryMissingScenario() {
10
+ new OperationValidationScenarioBase({
11
+ title: 'State.apply banValidator rejects payloads when target node entry is missing',
12
+ setupScenario: setupBanValidatorScenario,
13
+ buildValidPayload: context => buildBanValidatorPayload(context),
14
+ mutatePayload: (_t, payload) => payload,
15
+ applyInvalidPayload: applyWithTargetNodeEntryRemoval,
16
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
17
+ expectedLogs: ['Failed to verify target node entry.']
18
+ }).performScenario();
19
+ }
@@ -0,0 +1,19 @@
1
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
2
+ import {
3
+ setupBanValidatorScenario,
4
+ buildBanValidatorPayload,
5
+ assertBanValidatorFailureState,
6
+ applyWithBanValidatorRoleUpdateFailure
7
+ } from './banValidatorScenarioHelpers.js';
8
+
9
+ export default function banValidatorTargetRoleUpdateFailureScenario() {
10
+ new OperationValidationScenarioBase({
11
+ title: 'State.apply banValidator rejects payloads when target node role update fails',
12
+ setupScenario: setupBanValidatorScenario,
13
+ buildValidPayload: context => buildBanValidatorPayload(context),
14
+ mutatePayload: (_t, payload) => payload,
15
+ applyInvalidPayload: applyWithBanValidatorRoleUpdateFailure,
16
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
17
+ expectedLogs: ['Failed to update target node role.']
18
+ }).performScenario();
19
+ }
@@ -0,0 +1,38 @@
1
+ import { test } from 'brittle';
2
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
3
+ import { selectWriterPeer } from '../addWriter/addWriterScenarioHelpers.js';
4
+ import {
5
+ setupBanValidatorScenario,
6
+ buildBanValidatorPayload,
7
+ assertBanValidatorSuccessState
8
+ } from './banValidatorScenarioHelpers.js';
9
+
10
+ export default function banValidatorWhitelistedNonWriterScenario() {
11
+ test('State.apply banValidator removes privileges for whitelisted non-writer - happy path', async t => {
12
+ const context = await setupBanValidatorScenario(t, { promoteToWriter: false });
13
+ const adminPeer = context.adminBootstrap;
14
+ const validatorPeer = context.banValidatorScenario?.validatorPeer ?? selectWriterPeer(context);
15
+
16
+ const validatorEntryBefore = await adminPeer.base.view.get(validatorPeer.wallet.address);
17
+ t.ok(validatorEntryBefore, 'whitelisted node entry exists before banValidator');
18
+
19
+ const adminEntryBefore = await adminPeer.base.view.get(adminPeer.wallet.address);
20
+ t.ok(adminEntryBefore, 'admin entry exists before banValidator');
21
+
22
+ const payload = await buildBanValidatorPayload(context, { adminPeer, validatorPeer });
23
+
24
+ await adminPeer.base.append(payload);
25
+ await adminPeer.base.update();
26
+ await eventFlush();
27
+
28
+ await assertBanValidatorSuccessState(t, context, {
29
+ validatorPeer,
30
+ adminPeer,
31
+ validatorEntryBefore,
32
+ adminEntryBefore,
33
+ payload,
34
+ expectedInitialRoles: { isWhitelisted: true, isWriter: false, isIndexer: false },
35
+ expectWriterRegistry: false
36
+ });
37
+ });
38
+ }
@@ -0,0 +1,91 @@
1
+ import { test } from 'brittle';
2
+ import b4a from 'b4a';
3
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
4
+ import { selectWriterPeer } from '../addWriter/addWriterScenarioHelpers.js';
5
+ import {
6
+ setupBanValidatorScenario,
7
+ buildBanValidatorPayload,
8
+ assertBanValidatorSuccessState
9
+ } from './banValidatorScenarioHelpers.js';
10
+ import nodeEntryUtils, { ZERO_LICENSE } from '../../../../../src/core/state/utils/nodeEntry.js';
11
+ import { BALANCE_ZERO, toBalance } from '../../../../../src/core/state/utils/balance.js';
12
+
13
+ export default function banValidatorWhitelistedZeroBalanceScenario() {
14
+ test('State.apply banValidator handles zero-balance whitelisted non-writer without unstake', async t => {
15
+ const context = await setupBanValidatorScenario(t, { promoteToWriter: false });
16
+ const adminPeer = context.adminBootstrap;
17
+ const targetPeer = context.banValidatorScenario?.validatorPeer ?? selectWriterPeer(context);
18
+
19
+ const entryBefore = await adminPeer.base.view.get(targetPeer.wallet.address);
20
+ t.ok(entryBefore, 'whitelisted node entry exists before banValidator');
21
+ const decodedBefore = nodeEntryUtils.decode(entryBefore?.value);
22
+ t.ok(decodedBefore, 'whitelisted node entry decodes before banValidator');
23
+ if (!decodedBefore) return;
24
+
25
+ const balanceBefore = toBalance(decodedBefore.balance);
26
+ const stakedBefore = toBalance(decodedBefore.stakedBalance);
27
+ t.ok(balanceBefore, 'balance before banValidator decodes');
28
+ t.ok(stakedBefore, 'staked balance before banValidator decodes');
29
+ t.ok(b4a.equals(balanceBefore.value, BALANCE_ZERO.value), 'balance is zero before ban');
30
+ t.ok(b4a.equals(stakedBefore.value, BALANCE_ZERO.value), 'staked balance is zero before ban');
31
+ t.ok(!b4a.equals(decodedBefore.license, ZERO_LICENSE), 'license present before ban');
32
+
33
+ const adminEntryBefore = await adminPeer.base.view.get(adminPeer.wallet.address);
34
+ t.ok(adminEntryBefore, 'admin entry exists before banValidator');
35
+
36
+ const payload = await buildBanValidatorPayload(context, { adminPeer, validatorPeer: targetPeer });
37
+
38
+ await adminPeer.base.append(payload);
39
+ await adminPeer.base.update();
40
+ await eventFlush();
41
+
42
+ await assertBanValidatorSuccessState(t, context, {
43
+ validatorPeer: targetPeer,
44
+ adminPeer,
45
+ validatorEntryBefore: entryBefore,
46
+ adminEntryBefore,
47
+ payload,
48
+ expectedInitialRoles: { isWhitelisted: true, isWriter: false, isIndexer: false },
49
+ expectWriterRegistry: false
50
+ });
51
+
52
+ const entryAfter = await adminPeer.base.view.get(targetPeer.wallet.address);
53
+ t.ok(entryAfter, 'node entry exists after banValidator');
54
+ const decodedAfter = nodeEntryUtils.decode(entryAfter?.value);
55
+ t.ok(decodedAfter, 'node entry decodes after banValidator');
56
+ if (decodedAfter) {
57
+ t.ok(
58
+ b4a.equals(decodedAfter.balance, BALANCE_ZERO.value),
59
+ 'balance remains zero after banValidator'
60
+ );
61
+ t.ok(
62
+ b4a.equals(decodedAfter.stakedBalance, BALANCE_ZERO.value),
63
+ 'staked balance remains zero after banValidator'
64
+ );
65
+ t.ok(
66
+ b4a.equals(decodedAfter.license, decodedBefore.license),
67
+ 'license preserved after banValidator'
68
+ );
69
+ }
70
+
71
+ await context.sync();
72
+ const replicated = await targetPeer.base.view.get(targetPeer.wallet.address);
73
+ t.ok(replicated, 'replicated entry exists after banValidator');
74
+ const decodedReplicated = nodeEntryUtils.decode(replicated?.value);
75
+ t.ok(decodedReplicated, 'replicated entry decodes after banValidator');
76
+ if (decodedReplicated) {
77
+ t.ok(
78
+ b4a.equals(decodedReplicated.balance, BALANCE_ZERO.value),
79
+ 'replicated balance remains zero'
80
+ );
81
+ t.ok(
82
+ b4a.equals(decodedReplicated.stakedBalance, BALANCE_ZERO.value),
83
+ 'replicated staked balance remains zero'
84
+ );
85
+ t.ok(
86
+ b4a.equals(decodedReplicated.license, decodedBefore.license),
87
+ 'replicated license unchanged'
88
+ );
89
+ }
90
+ });
91
+ }
@@ -0,0 +1,19 @@
1
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
2
+ import {
3
+ setupBanValidatorScenario,
4
+ buildBanValidatorPayload,
5
+ assertBanValidatorFailureState,
6
+ applyWithBanValidatorWithdrawFailure
7
+ } from './banValidatorScenarioHelpers.js';
8
+
9
+ export default function banValidatorWithdrawFailureScenario() {
10
+ new OperationValidationScenarioBase({
11
+ title: 'State.apply banValidator rejects payloads when staked balance cannot be withdrawn',
12
+ setupScenario: setupBanValidatorScenario,
13
+ buildValidPayload: context => buildBanValidatorPayload(context),
14
+ mutatePayload: (_t, payload) => payload,
15
+ applyInvalidPayload: applyWithBanValidatorWithdrawFailure,
16
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
17
+ expectedLogs: ['Failed to withdraw staked balance.']
18
+ }).performScenario();
19
+ }
@@ -0,0 +1,266 @@
1
+ import banValidatorHappyPathScenario from './banValidatorHappyPathScenario.js';
2
+ import banValidatorWhitelistedNonWriterScenario from './banValidatorWhitelistedNonWriterScenario.js';
3
+ import banValidatorWhitelistedZeroBalanceScenario from './banValidatorWhitelistedZeroBalanceScenario.js';
4
+ import banValidatorSequentialBansScenario from './banValidatorSequentialBansScenario.js';
5
+ import banValidatorBanAndReWhitelistScenario from './banValidatorBanAndReWhitelistScenario.js';
6
+ import b4a from 'b4a';
7
+ import AdminControlOperationValidationScenario from '../common/adminControlOperationValidationScenario.js';
8
+ import RequesterAddressValidationScenario from '../common/requesterAddressValidationScenario.js';
9
+ import createRequesterPublicKeyValidationScenario from '../common/requesterPublicKeyValidationScenario.js';
10
+ import AdminEntryMissingScenario from '../common/access-control/adminEntryMissingScenario.js';
11
+ import AdminEntryDecodeFailureScenario from '../common/access-control/adminEntryDecodeFailureScenario.js';
12
+ import AdminOnlyGuardScenario from '../common/access-control/adminOnlyGuardScenario.js';
13
+ import AdminPublicKeyDecodeFailureScenario from '../common/access-control/adminPublicKeyDecodeFailureScenario.js';
14
+ import AdminConsistencyMismatchScenario from '../common/access-control/adminConsistencyMismatchScenario.js';
15
+ import InvalidHashValidationScenario from '../common/payload-structure/invalidHashValidationScenario.js';
16
+ import InvalidAddressValidationScenario from '../common/payload-structure/invalidAddressValidationScenario.js';
17
+ import InvalidSignatureValidationScenario, {
18
+ SignatureMutationStrategy
19
+ } from '../common/payload-structure/invalidSignatureValidationScenario.js';
20
+ import IndexerSequenceStateInvalidScenario from '../common/indexer/indexerSequenceStateInvalidScenario.js';
21
+ import TransactionValidityMismatchScenario from '../common/transactionValidityMismatchScenario.js';
22
+ import OperationAlreadyAppliedScenario from '../common/operationAlreadyAppliedScenario.js';
23
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
24
+ import FeeDecodeFailureScenario from '../common/balances/feeDecodeFailureScenario.js';
25
+ import RequesterNodeEntryMissingScenario from '../common/requester/requesterNodeEntryMissingScenario.js';
26
+ import RequesterBalanceScenarioBase from '../common/balances/base/requesterBalanceScenarioBase.js';
27
+ import {
28
+ setupBanValidatorScenario,
29
+ buildBanValidatorPayload,
30
+ buildBanValidatorPayloadWithTxValidity,
31
+ assertBanValidatorFailureState,
32
+ applyInvalidTargetAddressPayload
33
+ } from './banValidatorScenarioHelpers.js';
34
+ import banValidatorTargetNodeEntryMissingScenario from './banValidatorTargetNodeEntryMissingScenario.js';
35
+ import banValidatorTargetIndexerScenario from './banValidatorTargetIndexerScenario.js';
36
+ import banValidatorTargetRoleUpdateFailureScenario from './banValidatorTargetRoleUpdateFailureScenario.js';
37
+ import banValidatorTargetDecodeFailureScenario from './banValidatorTargetDecodeFailureScenario.js';
38
+ import banValidatorWithdrawFailureScenario from './banValidatorWithdrawFailureScenario.js';
39
+ import { applyWithRequesterEntryCorruption } from '../addWriter/addWriterScenarioHelpers.js';
40
+
41
+ banValidatorHappyPathScenario();
42
+ banValidatorWhitelistedNonWriterScenario();
43
+ banValidatorWhitelistedZeroBalanceScenario();
44
+ banValidatorSequentialBansScenario();
45
+ banValidatorBanAndReWhitelistScenario();
46
+
47
+ // Handler validation order
48
+ new AdminControlOperationValidationScenario({
49
+ title: 'State.apply banValidator rejects payloads when contract schema validation fails',
50
+ setupScenario: setupBanValidatorScenario,
51
+ buildValidPayload: context => buildBanValidatorPayload(context),
52
+ assertStateUnchanged: assertBanValidatorFailureState,
53
+ expectedLogs: ['Contract schema validation failed.']
54
+ }).performScenario();
55
+
56
+ new RequesterAddressValidationScenario({
57
+ title: 'State.apply banValidator rejects payloads when requester address is invalid',
58
+ setupScenario: setupBanValidatorScenario,
59
+ buildValidPayload: context => buildBanValidatorPayload(context),
60
+ assertStateUnchanged: assertBanValidatorFailureState,
61
+ expectedLogs: ['Requester address is invalid.']
62
+ }).performScenario();
63
+
64
+ createRequesterPublicKeyValidationScenario({
65
+ title: 'State.apply banValidator rejects payloads when requester public key is invalid',
66
+ setupScenario: setupBanValidatorScenario,
67
+ buildValidPayload: context => buildBanValidatorPayload(context),
68
+ assertStateUnchanged: assertBanValidatorFailureState,
69
+ expectedLogs: ['Error while decoding requester public key.']
70
+ }).performScenario();
71
+
72
+ new AdminEntryMissingScenario({
73
+ title: 'State.apply banValidator rejects payloads when admin entry is missing',
74
+ setupScenario: setupBanValidatorScenario,
75
+ buildValidPayload: context => buildBanValidatorPayload(context),
76
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
77
+ expectedLogs: ['Invalid admin entry.']
78
+ }).performScenario();
79
+
80
+ new AdminEntryDecodeFailureScenario({
81
+ title: 'State.apply banValidator rejects payloads when admin entry cannot be decoded',
82
+ setupScenario: setupBanValidatorScenario,
83
+ buildValidPayload: context => buildBanValidatorPayload(context),
84
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
85
+ expectedLogs: ['Failed to decode admin node entry.']
86
+ }).performScenario();
87
+
88
+ new AdminPublicKeyDecodeFailureScenario({
89
+ title: 'State.apply banValidator rejects payloads when admin public key cannot be decoded',
90
+ setupScenario: setupBanValidatorScenario,
91
+ buildValidPayload: context => buildBanValidatorPayload(context),
92
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
93
+ expectedLogs: ['Failed to decode admin public key.']
94
+ }).performScenario();
95
+
96
+ new AdminOnlyGuardScenario({
97
+ title: 'State.apply banValidator rejects non-admin nodes',
98
+ setupScenario: setupBanValidatorScenario,
99
+ buildValidPayload: context => buildBanValidatorPayload(context),
100
+ assertStateUnchanged: (t, context) =>
101
+ assertBanValidatorFailureState(t, context, {
102
+ expectedRoles: { isWhitelisted: true, isWriter: false, isIndexer: false },
103
+ allowEntryMutation: true,
104
+ skipSync: true
105
+ }),
106
+ expectedLogs: ['Node is not allowed to perform this operation. (ADMIN ONLY)']
107
+ }).performScenario();
108
+
109
+ new AdminConsistencyMismatchScenario({
110
+ title: 'State.apply banValidator rejects payloads when admin public key mismatches requester',
111
+ setupScenario: setupBanValidatorScenario,
112
+ buildValidPayload: context => buildBanValidatorPayload(context),
113
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
114
+ expectedLogs: ['System admin and node public keys do not match.']
115
+ }).performScenario();
116
+
117
+ new InvalidHashValidationScenario({
118
+ title: 'State.apply banValidator rejects payloads when message hash mismatches tx hash',
119
+ setupScenario: setupBanValidatorScenario,
120
+ buildValidPayload: context => buildBanValidatorPayload(context),
121
+ assertStateUnchanged: assertBanValidatorFailureState,
122
+ expectedLogs: ['Message hash does not match the tx_hash.']
123
+ }).performScenario();
124
+
125
+ new InvalidSignatureValidationScenario({
126
+ title: 'State.apply banValidator rejects payloads when admin signature is invalid (foreign signature)',
127
+ setupScenario: setupBanValidatorScenario,
128
+ buildValidPayload: context => buildBanValidatorPayload(context),
129
+ assertStateUnchanged: assertBanValidatorFailureState,
130
+ expectedLogs: ['Failed to verify message signature.']
131
+ }).performScenario();
132
+
133
+ new InvalidSignatureValidationScenario({
134
+ title: 'State.apply banValidator rejects payloads when admin signature is invalid (zero fill)',
135
+ setupScenario: setupBanValidatorScenario,
136
+ buildValidPayload: context => buildBanValidatorPayload(context),
137
+ assertStateUnchanged: assertBanValidatorFailureState,
138
+ strategy: SignatureMutationStrategy.ZERO_FILL,
139
+ expectedLogs: ['Failed to verify message signature.']
140
+ }).performScenario();
141
+
142
+ new InvalidSignatureValidationScenario({
143
+ title: 'State.apply banValidator rejects payloads when admin signature is invalid (type mismatch)',
144
+ setupScenario: setupBanValidatorScenario,
145
+ buildValidPayload: context => buildBanValidatorPayload(context),
146
+ assertStateUnchanged: assertBanValidatorFailureState,
147
+ strategy: SignatureMutationStrategy.TYPE_MISMATCH,
148
+ expectedLogs: ['Failed to verify message signature.']
149
+ }).performScenario();
150
+
151
+ new IndexerSequenceStateInvalidScenario({
152
+ title: 'State.apply banValidator rejects payloads when indexer sequence state is invalid',
153
+ setupScenario: setupBanValidatorScenario,
154
+ buildValidPayload: context => buildBanValidatorPayload(context),
155
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
156
+ expectedLogs: ['Indexer sequence state is invalid.']
157
+ }).performScenario();
158
+
159
+ new TransactionValidityMismatchScenario({
160
+ title: 'State.apply banValidator rejects payloads when transaction validity mismatches indexer state',
161
+ setupScenario: setupBanValidatorScenario,
162
+ buildValidPayload: context => buildBanValidatorPayload(context),
163
+ assertStateUnchanged: assertBanValidatorFailureState,
164
+ txValidityPath: ['aco', 'txv'],
165
+ rebuildPayloadWithTxValidity: ({ context, mutatedTxValidity }) =>
166
+ buildBanValidatorPayloadWithTxValidity(context, mutatedTxValidity),
167
+ expectedLogs: ['Transaction was not executed.']
168
+ }).performScenario();
169
+
170
+ new OperationAlreadyAppliedScenario({
171
+ title: 'State.apply banValidator rejects payloads when operation was already applied',
172
+ setupScenario: setupBanValidatorScenario,
173
+ buildValidPayload: context => buildBanValidatorPayload(context),
174
+ assertStateUnchanged: (t, context) =>
175
+ assertBanValidatorFailureState(t, context, {
176
+ expectedRoles: { isWhitelisted: false, isWriter: false, isIndexer: false },
177
+ allowEntryMutation: true,
178
+ skipSync: true
179
+ }),
180
+ expectedLogs: ['Operation has already been applied.']
181
+ }).performScenario();
182
+
183
+ new InvalidAddressValidationScenario({
184
+ title: 'State.apply banValidator rejects payloads when target node address is invalid',
185
+ setupScenario: setupBanValidatorScenario,
186
+ buildValidPayload: context => buildBanValidatorPayload(context),
187
+ assertStateUnchanged: assertBanValidatorFailureState,
188
+ addressPath: ['aco', 'ia'],
189
+ expectedLogs: ['Failed to verify target node address.'],
190
+ applyInvalidPayload: applyInvalidTargetAddressPayload
191
+ }).performScenario();
192
+
193
+ banValidatorTargetNodeEntryMissingScenario();
194
+ banValidatorTargetIndexerScenario();
195
+ banValidatorTargetRoleUpdateFailureScenario();
196
+ banValidatorTargetDecodeFailureScenario();
197
+
198
+ new FeeDecodeFailureScenario({
199
+ title: 'State.apply banValidator rejects payloads when fee amount is invalid',
200
+ setupScenario: setupBanValidatorScenario,
201
+ buildValidPayload: context => buildBanValidatorPayload(context),
202
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
203
+ expectedLogs: ['Invalid fee amount.']
204
+ }).performScenario();
205
+
206
+ new RequesterNodeEntryMissingScenario({
207
+ title: 'State.apply banValidator rejects payloads when admin node entry is missing',
208
+ setupScenario: setupBanValidatorScenario,
209
+ buildValidPayload: context => buildBanValidatorPayload(context),
210
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
211
+ selectPeer: context => context.adminBootstrap,
212
+ expectedLogs: ['Invalid admin node entry buffer.']
213
+ }).performScenario();
214
+
215
+ new OperationValidationScenarioBase({
216
+ title: 'State.apply banValidator rejects payloads when admin node entry cannot be decoded',
217
+ setupScenario: setupBanValidatorScenario,
218
+ buildValidPayload: context => buildBanValidatorPayload(context),
219
+ mutatePayload: (_t, payload) => payload,
220
+ applyInvalidPayload: (context, invalidPayload) =>
221
+ applyWithRequesterEntryCorruption(context, invalidPayload, { peer: context.adminBootstrap }),
222
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
223
+ expectedLogs: ['Failed to verify admin node entry.']
224
+ }).performScenario();
225
+
226
+ new RequesterBalanceScenarioBase({
227
+ title: 'State.apply banValidator rejects payloads when admin balance cannot be decoded',
228
+ setupScenario: setupBanValidatorScenario,
229
+ buildValidPayload: context => buildBanValidatorPayload(context),
230
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
231
+ mutateDecodedEntry: decoded => ({ ...decoded, balance: decoded.balance ? decoded.balance.subarray(0, 1) : null }),
232
+ selectPeer: context => context.adminBootstrap,
233
+ expectedLogs: ['Invalid admin balance']
234
+ }).performScenario();
235
+
236
+ new RequesterBalanceScenarioBase({
237
+ title: 'State.apply banValidator rejects payloads when admin balance is insufficient',
238
+ setupScenario: setupBanValidatorScenario,
239
+ buildValidPayload: context => buildBanValidatorPayload(context),
240
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
241
+ mutateDecodedEntry: decoded => ({ ...decoded, balance: b4a.alloc(decoded.balance.length) }),
242
+ selectPeer: context => context.adminBootstrap,
243
+ expectedLogs: ['Insufficient admin balance.']
244
+ }).performScenario();
245
+
246
+ new RequesterBalanceScenarioBase({
247
+ title: 'State.apply banValidator rejects payloads when admin fee cannot be applied',
248
+ setupScenario: setupBanValidatorScenario,
249
+ buildValidPayload: context => buildBanValidatorPayload(context),
250
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
251
+ selectPeer: context => context.adminBootstrap,
252
+ failNextBalanceSub: true,
253
+ expectedLogs: ['Failed to apply fee to admin balance.']
254
+ }).performScenario();
255
+
256
+ new RequesterBalanceScenarioBase({
257
+ title: 'State.apply banValidator rejects payloads when admin balance update fails',
258
+ setupScenario: setupBanValidatorScenario,
259
+ buildValidPayload: context => buildBanValidatorPayload(context),
260
+ assertStateUnchanged: (t, context) => assertBanValidatorFailureState(t, context, { skipSync: true }),
261
+ selectPeer: context => context.adminBootstrap,
262
+ failNextBalanceUpdate: true,
263
+ expectedLogs: ['Failed to update admin node balance.']
264
+ }).performScenario();
265
+
266
+ banValidatorWithdrawFailureScenario();
@@ -0,0 +1,142 @@
1
+ import b4a from 'b4a';
2
+ import { test } from 'brittle';
3
+ import {
4
+ setupBootstrapDeploymentScenario,
5
+ buildBootstrapDeploymentPayload,
6
+ assertBootstrapDeploymentSuccessState
7
+ } from './bootstrapDeploymentScenarioHelpers.js';
8
+ import { initializeBalances, whitelistAddress } from '../common/commonScenarioHelper.js';
9
+ import nodeEntryUtils from '../../../../../src/core/state/utils/nodeEntry.js';
10
+ import deploymentEntryUtils from '../../../../../src/core/state/utils/deploymentEntry.js';
11
+ import { safeDecodeApplyOperation } from '../../../../../src/utils/protobuf/operationHelpers.js';
12
+
13
+ async function setupDuplicateBootstrapScenario(t) {
14
+ const context = await setupBootstrapDeploymentScenario(t, { nodes: 4 });
15
+ const validatorPeer = context.bootstrapDeployment?.validatorPeer ?? context.peers?.[1];
16
+ const primaryDeployer = context.bootstrapDeployment?.deployerPeer ?? context.peers?.[2];
17
+ const secondaryDeployer = context.peers?.[3];
18
+
19
+ if (!validatorPeer || !primaryDeployer || !secondaryDeployer) {
20
+ throw new Error('Duplicate bootstrap scenario requires a validator and two deployers.');
21
+ }
22
+
23
+ const primaryEntry = await validatorPeer.base.view.get(primaryDeployer.wallet.address);
24
+ const primaryDecoded = primaryEntry ? nodeEntryUtils.decode(primaryEntry.value) : null;
25
+ const initialBalance = primaryDecoded?.balance ?? null;
26
+
27
+ if (initialBalance) {
28
+ await initializeBalances(context, [[secondaryDeployer.wallet.address, initialBalance]]);
29
+ }
30
+ await whitelistAddress(context, secondaryDeployer.wallet.address);
31
+ await context.sync();
32
+
33
+ const secondaryEntryBefore = await validatorPeer.base.view.get(secondaryDeployer.wallet.address);
34
+
35
+ context.bootstrapDeployment.secondaryDeployer = secondaryDeployer;
36
+ context.bootstrapDeployment.secondaryDeployerEntryBefore = secondaryEntryBefore
37
+ ? { value: b4a.from(secondaryEntryBefore.value) }
38
+ : null;
39
+
40
+ return context;
41
+ }
42
+
43
+ function buildDuplicatePayload(validPayload, context) {
44
+ const decoded = safeDecodeApplyOperation(validPayload);
45
+ if (!decoded?.bdo) return validPayload;
46
+
47
+ const secondaryDeployer = context.bootstrapDeployment?.secondaryDeployer ?? context.peers?.[3];
48
+ if (!secondaryDeployer) return validPayload;
49
+
50
+ return buildBootstrapDeploymentPayload(context, {
51
+ deployerPeer: secondaryDeployer,
52
+ externalBootstrap: decoded.bdo.bs,
53
+ channel: decoded.bdo.ic,
54
+ txValidity: decoded.bdo.txv
55
+ });
56
+ }
57
+
58
+ async function assertDuplicateBootstrapState(t, context, validPayload, invalidPayload) {
59
+ const validatorPeer = context.bootstrapDeployment?.validatorPeer ?? context.peers?.[1];
60
+ const secondaryDeployer = context.bootstrapDeployment?.secondaryDeployer ?? context.peers?.[3];
61
+ const secondaryBefore = context.bootstrapDeployment?.secondaryDeployerEntryBefore?.value ?? null;
62
+
63
+ const validDecoded = safeDecodeApplyOperation(validPayload);
64
+ const invalidDecoded = safeDecodeApplyOperation(invalidPayload);
65
+ t.ok(validDecoded, 'valid payload decodes for assertions');
66
+ t.ok(invalidDecoded, 'invalid payload decodes for assertions');
67
+ if (!validDecoded?.bdo || !invalidDecoded?.bdo) return;
68
+
69
+ const bootstrapHex = validDecoded.bdo.bs.toString('hex');
70
+ const firstTxHex = validDecoded.bdo.tx.toString('hex');
71
+ const secondTxHex = invalidDecoded.bdo.tx.toString('hex');
72
+
73
+ const deploymentKey = `deployment/${bootstrapHex}`;
74
+ const deploymentEntry = await validatorPeer.base.view.get(deploymentKey);
75
+ t.ok(deploymentEntry, 'deployment entry still stored');
76
+ const decodedDeployment = deploymentEntry ? deploymentEntryUtils.decode(deploymentEntry.value) : null;
77
+ t.ok(decodedDeployment, 'deployment entry decodes');
78
+ if (decodedDeployment?.txHash) {
79
+ t.is(decodedDeployment.txHash.toString('hex'), firstTxHex, 'deployment entry keeps original tx hash');
80
+ } else if (decodedDeployment) {
81
+ t.fail('deployment entry missing tx hash');
82
+ }
83
+
84
+ const firstTxEntry = await validatorPeer.base.view.get(firstTxHex);
85
+ t.ok(firstTxEntry, 'original tx hash remains recorded');
86
+
87
+ const duplicateTxEntry = await validatorPeer.base.view.get(secondTxHex);
88
+ t.is(duplicateTxEntry, null, 'duplicate tx hash not recorded');
89
+
90
+ if (secondaryDeployer) {
91
+ const after = await validatorPeer.base.view.get(secondaryDeployer.wallet.address);
92
+ t.ok(after, 'second deployer entry exists after rejection');
93
+ if (secondaryBefore && after?.value) {
94
+ const beforeDecoded = nodeEntryUtils.decode(secondaryBefore);
95
+ const afterDecoded = nodeEntryUtils.decode(after.value);
96
+ t.ok(beforeDecoded, 'second deployer entry before decodes');
97
+ t.ok(afterDecoded, 'second deployer entry after decodes');
98
+ if (beforeDecoded && afterDecoded) {
99
+ t.ok(b4a.equals(afterDecoded.balance, beforeDecoded.balance), 'second deployer balance unchanged');
100
+ t.ok(
101
+ b4a.equals(afterDecoded.stakedBalance, beforeDecoded.stakedBalance),
102
+ 'second deployer stake unchanged'
103
+ );
104
+ }
105
+ }
106
+ }
107
+ }
108
+
109
+ export default function bootstrapDeploymentDuplicateRegistrationScenario() {
110
+ test('State.apply bootstrapDeployment ignores duplicate bootstrap registrations', async t => {
111
+ const context = await setupDuplicateBootstrapScenario(t);
112
+ const validatorPeer = context.bootstrapDeployment?.validatorPeer ?? context.peers?.[1];
113
+
114
+ const validPayload = await buildBootstrapDeploymentPayload(context);
115
+ const invalidPayload = await buildDuplicatePayload(validPayload, context);
116
+
117
+ const capturedLogs = [];
118
+ const originalConsoleError = console.error;
119
+ console.error = (...args) => {
120
+ capturedLogs.push(args);
121
+ originalConsoleError(...args);
122
+ };
123
+
124
+ try {
125
+ await validatorPeer.base.append(validPayload);
126
+ await validatorPeer.base.update();
127
+ await assertBootstrapDeploymentSuccessState(t, context, { payload: validPayload, skipSync: true });
128
+
129
+ await validatorPeer.base.append(invalidPayload);
130
+ await validatorPeer.base.update();
131
+
132
+ await assertDuplicateBootstrapState(t, context, validPayload, invalidPayload);
133
+
134
+ const foundLog = capturedLogs.some(args =>
135
+ args.some(arg => String(arg).includes('Bootstrap already registered.'))
136
+ );
137
+ t.ok(foundLog, 'expected apply log "Bootstrap already registered." was emitted');
138
+ } finally {
139
+ console.error = originalConsoleError;
140
+ }
141
+ });
142
+ }
@@ -0,0 +1,26 @@
1
+ import { test } from 'brittle';
2
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
3
+ import {
4
+ setupBootstrapDeploymentScenario,
5
+ buildBootstrapDeploymentPayload,
6
+ assertBootstrapDeploymentSuccessState
7
+ } from './bootstrapDeploymentScenarioHelpers.js';
8
+
9
+ export default function bootstrapDeploymentHappyPathScenario() {
10
+ test(
11
+ 'State.apply bootstrapDeployment registers external bootstrap and rewards validator - happy path',
12
+ async t => {
13
+ const context = await setupBootstrapDeploymentScenario(t);
14
+ const validatorPeer =
15
+ context.bootstrapDeployment?.validatorPeer ?? context.adminBootstrap ?? context.peers[0];
16
+
17
+ const payload = await buildBootstrapDeploymentPayload(context);
18
+
19
+ await validatorPeer.base.append(payload);
20
+ await validatorPeer.base.update();
21
+ await eventFlush();
22
+
23
+ await assertBootstrapDeploymentSuccessState(t, context, { payload });
24
+ }
25
+ );
26
+ }