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,429 @@
1
+ import bootstrapDeploymentHappyPathScenario from './bootstrapDeploymentHappyPathScenario.js';
2
+ import bootstrapDeploymentMultipleBootstrapScenario from './bootstrapDeploymentMultipleBootstrapScenario.js';
3
+ import bootstrapDeploymentIncompleteOperationScenario from './bootstrapDeploymentIncompleteOperationScenario.js';
4
+ import b4a from 'b4a';
5
+ import PartialOperationValidationScenario, {
6
+ PartialOperationMutationStrategy
7
+ } from '../common/payload-structure/partialOperationValidationScenario.js';
8
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
9
+ import RequesterAddressValidationScenario from '../common/requesterAddressValidationScenario.js';
10
+ import createRequesterPublicKeyValidationScenario from '../common/requesterPublicKeyValidationScenario.js';
11
+ import InvalidHashValidationScenario from '../common/payload-structure/invalidHashValidationScenario.js';
12
+ import InvalidPayloadValidationScenario from '../common/payload-structure/invalidPayloadValidationScenario.js';
13
+ import InvalidAddressValidationScenario from '../common/payload-structure/invalidAddressValidationScenario.js';
14
+ import createAddressWithInvalidPublicKeyScenario from '../common/payload-structure/addressWithInvalidPublicKeyScenario.js';
15
+ import TransactionValidityMismatchScenario from '../common/transactionValidityMismatchScenario.js';
16
+ import IndexerSequenceStateInvalidScenario from '../common/indexer/indexerSequenceStateInvalidScenario.js';
17
+ import InvalidSignatureValidationScenario, {
18
+ SignatureMutationStrategy
19
+ } from '../common/payload-structure/invalidSignatureValidationScenario.js';
20
+ import ValidatorEntryDecodeFailureScenario from '../common/validatorConsistency/validatorEntryDecodeFailureScenario.js';
21
+ import ValidatorInactiveScenario from '../common/validatorConsistency/validatorInactiveScenario.js';
22
+ import ValidatorWriterKeyMismatchScenario from '../common/validatorConsistency/validatorWriterKeyMismatchScenario.js';
23
+ import ValidatorConsistencyScenarioBase, {
24
+ ValidatorEntryMutation
25
+ } from '../common/validatorConsistency/base/validatorConsistencyScenarioBase.js';
26
+ import OperationAlreadyAppliedScenario from '../common/operationAlreadyAppliedScenario.js';
27
+ import bootstrapDeploymentDuplicateRegistrationScenario from './bootstrapDeploymentDuplicateRegistrationScenario.js';
28
+ import bootstrapDeploymentInvalidDeploymentEntryScenario from './bootstrapDeploymentInvalidDeploymentEntryScenario.js';
29
+ import FeeDecodeFailureScenario from '../common/balances/feeDecodeFailureScenario.js';
30
+ import RequesterBalanceInsufficientScenario from '../common/balances/requesterBalanceInsufficientScenario.js';
31
+ import RequesterBalanceUpdateFailureScenario from '../common/balances/requesterBalanceUpdateFailureScenario.js';
32
+ import RequesterNodeEntryBufferMissingScenario from '../common/requester/requesterNodeEntryBufferMissingScenario.js';
33
+ import RequesterNodeEntryDecodeFailureScenario from '../common/requester/requesterNodeEntryDecodeFailureScenario.js';
34
+ import ValidatorEntryInvalidBalanceScenario from '../common/validatorEntryValidation/validatorEntryInvalidBalanceScenario.js';
35
+ import ValidatorEntryRewardFailureScenario from '../common/balances/validatorEntryRewardFailureScenario.js';
36
+ import ValidatorEntryUpdateFailureScenario from '../common/balances/validatorEntryUpdateFailureScenario.js';
37
+ import RequesterBalanceDecodeFailureScenario from '../common/balances/requesterBalanceDecodeFailureScenario.js';
38
+ import RequesterBalanceFeeApplicationFailureScenario from '../common/balances/requesterBalanceFeeApplicationFailureScenario.js';
39
+ import ValidatorNodeEntryDecodeFailureScenario from '../common/balances/validatorNodeEntryDecodeFailureScenario.js';
40
+ import {
41
+ setupBootstrapDeploymentScenario,
42
+ buildBootstrapDeploymentPayload,
43
+ buildBootstrapDeploymentPayloadWithTxValidity,
44
+ assertBootstrapDeploymentFailureState,
45
+ assertBootstrapDeploymentSuccessState,
46
+ mutateToNetworkBootstrap,
47
+ appendInvalidPayload
48
+ } from './bootstrapDeploymentScenarioHelpers.js';
49
+ import bootstrapDeploymentInvalidValidatorNodeEntryScenario from './invalidValidatorNodeEntryScenario.js';
50
+ import { safeDecodeApplyOperation, safeEncodeApplyOperation } from '../../../../../src/utils/protobuf/operationHelpers.js';
51
+
52
+ bootstrapDeploymentHappyPathScenario();
53
+ bootstrapDeploymentMultipleBootstrapScenario();
54
+
55
+ new InvalidPayloadValidationScenario({
56
+ title: 'State.apply bootstrapDeployment rejects payloads that fail contract validation',
57
+ setupScenario: setupBootstrapDeploymentScenario,
58
+ buildValidPayload: buildBootstrapDeploymentPayload,
59
+ mutatePayload: (t, validPayload) => {
60
+ const decoded = safeDecodeApplyOperation(validPayload);
61
+ t.ok(decoded, 'valid payload decodes before mutation');
62
+ if (!decoded) return validPayload;
63
+ const invalid = b4a.alloc(1);
64
+ return safeEncodeApplyOperation({ ...decoded, address: invalid });
65
+ },
66
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
67
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalidPayload }),
68
+ expectedLogs: ['Contract schema validation failed.']
69
+ }).performScenario();
70
+
71
+ bootstrapDeploymentIncompleteOperationScenario();
72
+
73
+ new OperationValidationScenarioBase({
74
+ title: 'State.apply bootstrapDeployment rejects attempts to deploy existing MSB bootstrap',
75
+ setupScenario: setupBootstrapDeploymentScenario,
76
+ buildValidPayload: buildBootstrapDeploymentPayload,
77
+ mutatePayload: (t, payload, context) => mutateToNetworkBootstrap(t, payload, context),
78
+ applyInvalidPayload: appendInvalidPayload,
79
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
80
+ assertBootstrapDeploymentFailureState(t, context, {
81
+ payload: invalidPayload,
82
+ bootstrapBufferOverride: context.bootstrap?.base?.local?.key ?? null
83
+ }),
84
+ expectedLogs: ['Cannot deploy bootstrap on existing same bootstrap.']
85
+ }).performScenario();
86
+
87
+ new PartialOperationValidationScenario({
88
+ title: 'State.apply bootstrapDeployment rejects matching requester and validator nonces',
89
+ setupScenario: setupBootstrapDeploymentScenario,
90
+ buildValidPayload: buildBootstrapDeploymentPayload,
91
+ assertStateUnchanged: (t, context, _valid, invalid) =>
92
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalid }),
93
+ strategy: PartialOperationMutationStrategy.NONCE_MATCH,
94
+ parentKey: 'bdo',
95
+ expectedLogs: ['Nonces should not be the same.']
96
+ }).performScenario();
97
+
98
+ new PartialOperationValidationScenario({
99
+ title: 'State.apply bootstrapDeployment rejects matching requester and validator addresses',
100
+ setupScenario: setupBootstrapDeploymentScenario,
101
+ buildValidPayload: buildBootstrapDeploymentPayload,
102
+ assertStateUnchanged: (t, context, _valid, invalid) =>
103
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalid }),
104
+ strategy: PartialOperationMutationStrategy.ADDRESS_MATCH,
105
+ parentKey: 'bdo',
106
+ expectedLogs: ['Addresses should be different.']
107
+ }).performScenario();
108
+
109
+ new PartialOperationValidationScenario({
110
+ title: 'State.apply bootstrapDeployment rejects matching requester and validator signatures',
111
+ setupScenario: setupBootstrapDeploymentScenario,
112
+ buildValidPayload: buildBootstrapDeploymentPayload,
113
+ assertStateUnchanged: (t, context, _valid, invalid) =>
114
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalid }),
115
+ strategy: PartialOperationMutationStrategy.SIGNATURE_MATCH,
116
+ parentKey: 'bdo',
117
+ expectedLogs: ['Signatures should be different.']
118
+ }).performScenario();
119
+
120
+ new RequesterAddressValidationScenario({
121
+ title: 'State.apply bootstrapDeployment rejects invalid requester address',
122
+ setupScenario: setupBootstrapDeploymentScenario,
123
+ buildValidPayload: buildBootstrapDeploymentPayload,
124
+ assertStateUnchanged: (t, context, _valid, invalid) =>
125
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalid }),
126
+ expectedLogs: ['Requester address is invalid.']
127
+ }).performScenario();
128
+
129
+ createRequesterPublicKeyValidationScenario({
130
+ title: 'State.apply bootstrapDeployment rejects requester public key decode failures',
131
+ setupScenario: setupBootstrapDeploymentScenario,
132
+ buildValidPayload: buildBootstrapDeploymentPayload,
133
+ assertStateUnchanged: (t, context, _valid, invalid) =>
134
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalid }),
135
+ expectedLogs: ['Failed to decode requester public key.']
136
+ }).performScenario();
137
+
138
+ new InvalidHashValidationScenario({
139
+ title: 'State.apply bootstrapDeployment rejects payloads when tx hash does not match message',
140
+ setupScenario: setupBootstrapDeploymentScenario,
141
+ buildValidPayload: buildBootstrapDeploymentPayload,
142
+ assertStateUnchanged: (t, context, _valid, invalid) =>
143
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalid }),
144
+ expectedLogs: ['Message hash does not match the tx_hash.']
145
+ }).performScenario();
146
+
147
+ new InvalidSignatureValidationScenario({
148
+ title: 'State.apply bootstrapDeployment requester signature is invalid (foreign signature)',
149
+ setupScenario: setupBootstrapDeploymentScenario,
150
+ buildValidPayload: buildBootstrapDeploymentPayload,
151
+ assertStateUnchanged: (t, context, _valid, invalid) =>
152
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalid }),
153
+ expectedLogs: ['Failed to verify message signature.']
154
+ }).performScenario();
155
+
156
+ new InvalidSignatureValidationScenario({
157
+ title: 'State.apply bootstrapDeployment requester signature is invalid (zero fill)',
158
+ setupScenario: setupBootstrapDeploymentScenario,
159
+ buildValidPayload: buildBootstrapDeploymentPayload,
160
+ assertStateUnchanged: (t, context, _valid, invalid) =>
161
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalid }),
162
+ strategy: SignatureMutationStrategy.ZERO_FILL,
163
+ expectedLogs: ['Failed to verify message signature.']
164
+ }).performScenario();
165
+
166
+ new InvalidSignatureValidationScenario({
167
+ title: 'State.apply bootstrapDeployment requester signature is invalid (type mismatch)',
168
+ setupScenario: setupBootstrapDeploymentScenario,
169
+ buildValidPayload: buildBootstrapDeploymentPayload,
170
+ assertStateUnchanged: (t, context, _valid, invalid) =>
171
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalid }),
172
+ strategy: SignatureMutationStrategy.TYPE_MISMATCH,
173
+ expectedLogs: ['Failed to verify message signature.']
174
+ }).performScenario();
175
+
176
+ new InvalidAddressValidationScenario({
177
+ title: 'State.apply bootstrapDeployment validator address is invalid',
178
+ setupScenario: setupBootstrapDeploymentScenario,
179
+ buildValidPayload: buildBootstrapDeploymentPayload,
180
+ assertStateUnchanged: (t, context, _valid, invalid) =>
181
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalid }),
182
+ addressPath: ['bdo', 'va'],
183
+ expectedLogs: ['Invalid validator address.']
184
+ }).performScenario();
185
+
186
+ createAddressWithInvalidPublicKeyScenario({
187
+ title: 'State.apply bootstrapDeployment validator public key is invalid',
188
+ setupScenario: setupBootstrapDeploymentScenario,
189
+ buildValidPayload: buildBootstrapDeploymentPayload,
190
+ assertStateUnchanged: (t, context, _valid, invalid) =>
191
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalid }),
192
+ addressPath: ['bdo', 'va'],
193
+ expectedLogs: ['Failed to decode validator public key.']
194
+ }).performScenario();
195
+
196
+ function mutateValidatorSignature(t, validPayload) {
197
+ const decoded = safeDecodeApplyOperation(validPayload);
198
+ t.ok(decoded, 'fixtures decode');
199
+ const parent = decoded?.bdo;
200
+ if (!parent?.vs) return validPayload;
201
+ const mutated = b4a.from(parent.vs);
202
+ mutated[mutated.length - 1] ^= 0xff;
203
+ parent.vs = mutated;
204
+ return safeEncodeApplyOperation(decoded);
205
+ }
206
+
207
+ new OperationValidationScenarioBase({
208
+ title: 'State.apply bootstrapDeployment validator signature is invalid',
209
+ setupScenario: setupBootstrapDeploymentScenario,
210
+ buildValidPayload: buildBootstrapDeploymentPayload,
211
+ mutatePayload: mutateValidatorSignature,
212
+ applyInvalidPayload: appendInvalidPayload,
213
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
214
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalidPayload }),
215
+ expectedLogs: ['Failed to verify validator message signature.']
216
+ }).performScenario();
217
+
218
+ //TODO: Add to another tests
219
+ new IndexerSequenceStateInvalidScenario({
220
+ title: 'State.apply bootstrapDeployment rejects payloads when indexer sequence state is invalid',
221
+ setupScenario: setupBootstrapDeploymentScenario,
222
+ buildValidPayload: buildBootstrapDeploymentPayload,
223
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
224
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalidPayload }),
225
+ expectedLogs: ['Indexer sequence state is invalid.']
226
+ }).performScenario();
227
+
228
+ new TransactionValidityMismatchScenario({
229
+ title: 'State.apply bootstrapDeployment rejects payloads when tx validity mismatches indexer state',
230
+ setupScenario: setupBootstrapDeploymentScenario,
231
+ buildValidPayload: buildBootstrapDeploymentPayload,
232
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
233
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalidPayload }),
234
+ txValidityPath: ['bdo', 'txv'],
235
+ rebuildPayloadWithTxValidity: ({ context, mutatedTxValidity }) =>
236
+ buildBootstrapDeploymentPayloadWithTxValidity(context, mutatedTxValidity),
237
+ expectedLogs: ['Transaction was not executed.']
238
+ }).performScenario();
239
+
240
+ new ValidatorConsistencyScenarioBase({
241
+ title: 'State.apply bootstrapDeployment rejects payloads when validator entry is missing',
242
+ setupScenario: setupBootstrapDeploymentScenario,
243
+ buildValidPayload: buildBootstrapDeploymentPayload,
244
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
245
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalidPayload }),
246
+ mutateEntry: () => ValidatorEntryMutation.DELETE,
247
+ validatorAddressPath: ['bdo', 'va'],
248
+ expectedLogs: ['Incoming validator entry is null.']
249
+ }).performScenario();
250
+
251
+ new ValidatorEntryDecodeFailureScenario({
252
+ title: 'State.apply bootstrapDeployment rejects payloads when validator entry cannot be decoded',
253
+ setupScenario: setupBootstrapDeploymentScenario,
254
+ buildValidPayload: buildBootstrapDeploymentPayload,
255
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
256
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalidPayload, skipSync: true }),
257
+ validatorAddressPath: ['bdo', 'va'],
258
+ expectedLogs: ['Failed to decode validator entry.']
259
+ }).performScenario();
260
+
261
+ new ValidatorInactiveScenario({
262
+ title: 'State.apply bootstrapDeployment rejects payloads when validator is not an active writer',
263
+ setupScenario: setupBootstrapDeploymentScenario,
264
+ buildValidPayload: buildBootstrapDeploymentPayload,
265
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
266
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalidPayload, skipSync: true }),
267
+ validatorAddressPath: ['bdo', 'va'],
268
+ expectedLogs: ['Operation validator is not active']
269
+ }).performScenario();
270
+
271
+ new ValidatorWriterKeyMismatchScenario({
272
+ title: 'State.apply bootstrapDeployment rejects payloads when validator writer key mismatches requester',
273
+ setupScenario: setupBootstrapDeploymentScenario,
274
+ buildValidPayload: buildBootstrapDeploymentPayload,
275
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
276
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalidPayload, skipSync: true }),
277
+ validatorAddressPath: ['bdo', 'va'],
278
+ expectedLogs: ['Validator cannot be the same as requester.']
279
+ }).performScenario();
280
+
281
+ new OperationAlreadyAppliedScenario({
282
+ title: 'State.apply bootstrapDeployment rejects payloads that were already applied',
283
+ setupScenario: setupBootstrapDeploymentScenario,
284
+ buildValidPayload: buildBootstrapDeploymentPayload,
285
+ assertStateUnchanged: (t, context, validPayload) =>
286
+ assertBootstrapDeploymentSuccessState(t, context, { payload: validPayload }),
287
+ selectNode: context => context.bootstrapDeployment?.validatorPeer ?? context.bootstrap ?? context.peers?.[0],
288
+ expectedLogs: ['Operation has already been applied.']
289
+ }).performScenario();
290
+
291
+ bootstrapDeploymentDuplicateRegistrationScenario();
292
+ bootstrapDeploymentInvalidDeploymentEntryScenario();
293
+
294
+ new FeeDecodeFailureScenario({
295
+ title: 'State.apply bootstrapDeployment rejects payloads when fee cannot be decoded',
296
+ setupScenario: setupBootstrapDeploymentScenario,
297
+ buildValidPayload: buildBootstrapDeploymentPayload,
298
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
299
+ assertBootstrapDeploymentFailureState(t, context, {
300
+ payload: invalidPayload,
301
+ skipValidatorEquality: true
302
+ }),
303
+ selectNode: context => context.bootstrapDeployment?.validatorPeer ?? context.bootstrap ?? context.peers?.[0]
304
+ }).performScenario();
305
+
306
+ new RequesterNodeEntryBufferMissingScenario({
307
+ title: 'State.apply bootstrapDeployment rejects payloads when requester node entry buffer is missing',
308
+ setupScenario: setupBootstrapDeploymentScenario,
309
+ buildValidPayload: buildBootstrapDeploymentPayload,
310
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
311
+ assertBootstrapDeploymentFailureState(t, context, {
312
+ payload: invalidPayload,
313
+ skipSync: true,
314
+ skipValidatorEquality: true
315
+ }),
316
+ selectNode: context => context.bootstrapDeployment?.validatorPeer ?? context.bootstrap ?? context.peers?.[0]
317
+ }).performScenario();
318
+
319
+ new RequesterNodeEntryDecodeFailureScenario({
320
+ title: 'State.apply bootstrapDeployment rejects payloads when requester node entry cannot be decoded',
321
+ setupScenario: setupBootstrapDeploymentScenario,
322
+ buildValidPayload: buildBootstrapDeploymentPayload,
323
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
324
+ assertBootstrapDeploymentFailureState(t, context, {
325
+ payload: invalidPayload,
326
+ skipSync: true,
327
+ skipValidatorEquality: true
328
+ }),
329
+ selectPeer: context => context.bootstrapDeployment?.deployerPeer ?? context.peers?.[2],
330
+ selectNode: context => context.bootstrapDeployment?.validatorPeer ?? context.bootstrap ?? context.peers?.[0]
331
+ }).performScenario();
332
+
333
+ new RequesterBalanceDecodeFailureScenario({
334
+ title: 'State.apply bootstrapDeployment rejects payloads when requester balance cannot be decoded',
335
+ setupScenario: setupBootstrapDeploymentScenario,
336
+ buildValidPayload: buildBootstrapDeploymentPayload,
337
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
338
+ assertBootstrapDeploymentFailureState(t, context, {
339
+ payload: invalidPayload,
340
+ skipSync: true,
341
+ skipValidatorEquality: true
342
+ }),
343
+ selectPeer: context => context.bootstrapDeployment?.deployerPeer ?? context.peers?.[2],
344
+ selectNode: context => context.bootstrapDeployment?.validatorPeer ?? context.bootstrap ?? context.peers?.[0]
345
+ }).performScenario();
346
+
347
+ new RequesterBalanceInsufficientScenario({
348
+ title: 'State.apply bootstrapDeployment rejects payloads when requester balance is insufficient',
349
+ setupScenario: setupBootstrapDeploymentScenario,
350
+ buildValidPayload: buildBootstrapDeploymentPayload,
351
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
352
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalidPayload, skipSync: true }),
353
+ selectPeer: context => context.bootstrapDeployment?.deployerPeer ?? context.peers?.[2],
354
+ selectNode: context => context.bootstrapDeployment?.validatorPeer ?? context.bootstrap ?? context.peers?.[0]
355
+ }).performScenario();
356
+
357
+ new RequesterBalanceUpdateFailureScenario({
358
+ title: 'State.apply bootstrapDeployment rejects payloads when requester balance update fails',
359
+ setupScenario: setupBootstrapDeploymentScenario,
360
+ buildValidPayload: buildBootstrapDeploymentPayload,
361
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
362
+ assertBootstrapDeploymentFailureState(t, context, {
363
+ payload: invalidPayload,
364
+ skipSync: true,
365
+ skipValidatorEquality: true
366
+ }),
367
+ selectPeer: context => context.bootstrapDeployment?.deployerPeer ?? context.peers?.[2],
368
+ selectNode: context => context.bootstrapDeployment?.validatorPeer ?? context.bootstrap ?? context.peers?.[0],
369
+ expectedLogs: ['Failed to update requester node balance.']
370
+ }).performScenario();
371
+
372
+ new RequesterBalanceFeeApplicationFailureScenario({
373
+ title: 'State.apply bootstrapDeployment rejects payloads when requester fee cannot be applied',
374
+ setupScenario: setupBootstrapDeploymentScenario,
375
+ buildValidPayload: buildBootstrapDeploymentPayload,
376
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
377
+ assertBootstrapDeploymentFailureState(t, context, {
378
+ payload: invalidPayload,
379
+ skipSync: true,
380
+ skipValidatorEquality: true
381
+ }),
382
+ selectPeer: context => context.bootstrapDeployment?.deployerPeer ?? context.peers?.[2],
383
+ selectNode: context => context.bootstrapDeployment?.validatorPeer ?? context.bootstrap ?? context.peers?.[0],
384
+ expectedLogs: ['Failed to apply fee to requester.']
385
+ }).performScenario();
386
+
387
+ bootstrapDeploymentInvalidValidatorNodeEntryScenario();
388
+
389
+ new ValidatorEntryInvalidBalanceScenario({
390
+ title: 'State.apply bootstrapDeployment rejects payloads when validator balance is invalid',
391
+ setupScenario: setupBootstrapDeploymentScenario,
392
+ buildValidPayload: buildBootstrapDeploymentPayload,
393
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
394
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalidPayload }),
395
+ validatorAddressPath: ['bdo', 'va'],
396
+ selectNode: context => context.bootstrapDeployment?.validatorPeer ?? context.bootstrap ?? context.peers?.[0]
397
+ }).performScenario();
398
+
399
+ new ValidatorNodeEntryDecodeFailureScenario({
400
+ title: 'State.apply bootstrapDeployment rejects payloads when validator node entry cannot be decoded',
401
+ setupScenario: setupBootstrapDeploymentScenario,
402
+ buildValidPayload: buildBootstrapDeploymentPayload,
403
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
404
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalidPayload, skipValidatorEquality: true }),
405
+ selectPeer: context => context.bootstrapDeployment?.validatorPeer ?? context.peers?.[1],
406
+ selectNode: context => context.bootstrapDeployment?.validatorPeer ?? context.bootstrap ?? context.peers?.[0],
407
+ validatorPeer: context => context.bootstrapDeployment?.validatorPeer ?? context.peers?.[1]
408
+ }).performScenario();
409
+
410
+ new ValidatorEntryRewardFailureScenario({
411
+ title: 'State.apply bootstrapDeployment rejects payloads when validator fee transfer fails',
412
+ setupScenario: setupBootstrapDeploymentScenario,
413
+ buildValidPayload: buildBootstrapDeploymentPayload,
414
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
415
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalidPayload }),
416
+ validatorAddressPath: ['bdo', 'va'],
417
+ selectNode: context => context.bootstrapDeployment?.validatorPeer ?? context.bootstrap ?? context.peers?.[0]
418
+ }).performScenario();
419
+
420
+ new ValidatorEntryUpdateFailureScenario({
421
+ title: 'State.apply bootstrapDeployment rejects payloads when validator node balance update fails',
422
+ setupScenario: setupBootstrapDeploymentScenario,
423
+ buildValidPayload: buildBootstrapDeploymentPayload,
424
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
425
+ assertBootstrapDeploymentFailureState(t, context, { payload: invalidPayload }),
426
+ validatorAddressPath: ['bdo', 'va'],
427
+ selectNode: context => context.bootstrapDeployment?.validatorPeer ?? context.bootstrap ?? context.peers?.[0],
428
+ expectedLogs: ['Failed to update validator node balance.']
429
+ }).performScenario();
@@ -0,0 +1,119 @@
1
+ import b4a from 'b4a';
2
+ import OperationValidationScenarioBase from '../base/OperationValidationScenarioBase.js';
3
+ import { eventFlush } from '../../../../../helpers/autobaseTestHelpers.js';
4
+ import { EntryType, TRAC_ADDRESS_SIZE } from '../../../../../../src/utils/constants.js';
5
+ import addressUtils from '../../../../../../src/core/state/utils/address.js';
6
+
7
+ export default class AdminConsistencyMismatchScenario extends OperationValidationScenarioBase {
8
+ constructor({
9
+ title,
10
+ setupScenario,
11
+ buildValidPayload,
12
+ assertStateUnchanged,
13
+ expectedLogs
14
+ }) {
15
+ super({
16
+ title,
17
+ setupScenario,
18
+ buildValidPayload,
19
+ mutatePayload: passThroughPayload,
20
+ applyInvalidPayload: applyWithMutatedAdminEntry,
21
+ assertStateUnchanged,
22
+ expectedLogs
23
+ });
24
+ }
25
+ }
26
+
27
+ function passThroughPayload(_t, payload) {
28
+ return payload;
29
+ }
30
+
31
+ async function applyWithMutatedAdminEntry(context, payload) {
32
+ const adminNode = context.adminBootstrap;
33
+ if (!adminNode?.base) {
34
+ throw new Error('Admin consistency scenario requires an admin bootstrap node.');
35
+ }
36
+
37
+ const reader = context.peers?.[1];
38
+ if (!reader?.wallet?.address) {
39
+ throw new Error('Admin consistency scenario requires a reader peer with a wallet.');
40
+ }
41
+
42
+ const alternateAddressBuffer = addressUtils.addressToBuffer(reader.wallet.address);
43
+ if (!alternateAddressBuffer || alternateAddressBuffer.length !== TRAC_ADDRESS_SIZE) {
44
+ throw new Error('Failed to derive alternate admin address buffer.');
45
+ }
46
+
47
+ const cleanup = patchAdminEntryMismatch(adminNode.base, alternateAddressBuffer);
48
+
49
+ try {
50
+ await adminNode.base.append(payload);
51
+ await adminNode.base.update();
52
+ await eventFlush();
53
+ } finally {
54
+ await cleanup?.();
55
+ }
56
+ }
57
+
58
+ function patchAdminEntryMismatch(base, alternateAddressBuffer) {
59
+ const originalApply = base._handlers.apply;
60
+ let shouldPatchNextApply = true;
61
+
62
+ base._handlers.apply = async (nodes, view, baseCtx) => {
63
+ if (!shouldPatchNextApply) {
64
+ return originalApply(nodes, view, baseCtx);
65
+ }
66
+
67
+ shouldPatchNextApply = false;
68
+ const previousBatch = view.batch;
69
+ const boundBatch = previousBatch.bind(view);
70
+
71
+ view.batch = function patchedBatch(...args) {
72
+ const batch = boundBatch(...args);
73
+ const originalGet = batch.get?.bind(batch);
74
+ if (typeof originalGet === 'function') {
75
+ let mutatedOnce = false;
76
+ batch.get = async key => {
77
+ if (!isAdminEntryKey(key) || mutatedOnce) {
78
+ return originalGet(key);
79
+ }
80
+
81
+ const adminEntry = await originalGet(key);
82
+ if (!adminEntry?.value) {
83
+ return adminEntry;
84
+ }
85
+
86
+ mutatedOnce = true;
87
+ const mutated = b4a.from(adminEntry.value);
88
+ alternateAddressBuffer.copy(mutated, 0, 0, TRAC_ADDRESS_SIZE);
89
+
90
+ return {
91
+ ...adminEntry,
92
+ value: mutated
93
+ };
94
+ };
95
+ }
96
+ return batch;
97
+ };
98
+
99
+ try {
100
+ return await originalApply(nodes, view, baseCtx);
101
+ } finally {
102
+ view.batch = previousBatch;
103
+ }
104
+ };
105
+
106
+ return () => {
107
+ base._handlers.apply = originalApply;
108
+ };
109
+ }
110
+
111
+ function isAdminEntryKey(key) {
112
+ if (typeof key === 'string') {
113
+ return key === EntryType.ADMIN;
114
+ }
115
+ if (b4a.isBuffer(key)) {
116
+ return b4a.equals(key, b4a.from(EntryType.ADMIN));
117
+ }
118
+ return false;
119
+ }
@@ -0,0 +1,130 @@
1
+ import b4a from 'b4a';
2
+ import { eventFlush } from '../../../../../helpers/autobaseTestHelpers.js';
3
+ import OperationValidationScenarioBase from '../base/OperationValidationScenarioBase.js';
4
+ import { EntryType } from '../../../../../../src/utils/constants.js';
5
+
6
+ const ADMIN_KEY_BUFFER = b4a.from(EntryType.ADMIN);
7
+
8
+ export default class AdminEntryDecodeFailureScenario extends OperationValidationScenarioBase {
9
+ constructor({
10
+ title,
11
+ setupScenario,
12
+ buildValidPayload,
13
+ assertStateUnchanged,
14
+ selectNode = defaultSelectNode,
15
+ expectedLogs
16
+ }) {
17
+ super({
18
+ title,
19
+ setupScenario,
20
+ buildValidPayload,
21
+ mutatePayload: passThroughPayload,
22
+ applyInvalidPayload: createApplyInvalidPayload(selectNode),
23
+ assertStateUnchanged,
24
+ expectedLogs
25
+ });
26
+ }
27
+ }
28
+
29
+ function passThroughPayload(_t, payload) {
30
+ return payload;
31
+ }
32
+
33
+ function defaultSelectNode(context) {
34
+ return context.adminBootstrap ?? context.bootstrap ?? context.peers?.[0] ?? null;
35
+ }
36
+
37
+ function createApplyInvalidPayload(selectNode) {
38
+ return async (context, payload) => {
39
+ const node = selectNode(context);
40
+ if (!node?.base) {
41
+ throw new Error('Admin entry decode failure scenario requires a writable node.');
42
+ }
43
+
44
+ const cleanup = patchAdminEntryDecode(node.base);
45
+ try {
46
+ await applyPayload(node, payload);
47
+ } finally {
48
+ await cleanup();
49
+ }
50
+ };
51
+ }
52
+
53
+ function patchAdminEntryDecode(base) {
54
+ const originalApply = base._handlers.apply;
55
+ let shouldPatchNextApply = true;
56
+
57
+ base._handlers.apply = async (nodes, view, baseCtx) => {
58
+ if (!shouldPatchNextApply) {
59
+ return originalApply(nodes, view, baseCtx);
60
+ }
61
+
62
+ shouldPatchNextApply = false;
63
+ const previousBatch = view.batch;
64
+ const boundBatch = previousBatch.bind(view);
65
+
66
+ view.batch = function patchedBatch(...args) {
67
+ const batch = boundBatch(...args);
68
+ const originalGet = batch.get?.bind(batch);
69
+ if (typeof originalGet === 'function') {
70
+ let shouldCorruptEntry = true;
71
+ batch.get = async key => {
72
+ if (!isAdminEntryKey(key)) {
73
+ return originalGet(key);
74
+ }
75
+
76
+ const adminEntry = await originalGet(key);
77
+ if (!adminEntry?.value) {
78
+ return adminEntry;
79
+ }
80
+
81
+ if (shouldCorruptEntry) {
82
+ shouldCorruptEntry = false;
83
+ return {
84
+ ...adminEntry,
85
+ value: corruptEntry(adminEntry.value)
86
+ };
87
+ }
88
+
89
+ return adminEntry;
90
+ };
91
+ }
92
+ return batch;
93
+ };
94
+
95
+ try {
96
+ return await originalApply(nodes, view, baseCtx);
97
+ } finally {
98
+ view.batch = previousBatch;
99
+ }
100
+ };
101
+
102
+ return () => {
103
+ base._handlers.apply = originalApply;
104
+ };
105
+ }
106
+
107
+ function isAdminEntryKey(key) {
108
+ if (typeof key === 'string') {
109
+ return key === EntryType.ADMIN;
110
+ }
111
+ if (b4a.isBuffer(key)) {
112
+ return b4a.equals(key, ADMIN_KEY_BUFFER);
113
+ }
114
+ return false;
115
+ }
116
+
117
+ function corruptEntry(value) {
118
+ if (!b4a.isBuffer(value) || value.length === 0) {
119
+ return b4a.alloc(1);
120
+ }
121
+ const corrupted = b4a.alloc(1);
122
+ corrupted[0] = value[0] ^ 0xff;
123
+ return corrupted;
124
+ }
125
+
126
+ async function applyPayload(node, payload) {
127
+ await node.base.append(payload);
128
+ await node.base.update();
129
+ await eventFlush();
130
+ }