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,819 @@
1
+ import b4a from 'b4a';
2
+ import transferExistingRecipientAmountScenario from './transferExistingRecipientAmountScenario.js';
3
+ import transferExistingRecipientZeroAmountScenario from './transferExistingRecipientZeroAmountScenario.js';
4
+ import transferNewRecipientAmountScenario from './transferNewRecipientAmountScenario.js';
5
+ import transferNewRecipientZeroAmountScenario from './transferNewRecipientZeroAmountScenario.js';
6
+ import transferValidatorRecipientAmountScenario from './transferValidatorRecipientAmountScenario.js';
7
+ import transferValidatorRecipientZeroAmountScenario from './transferValidatorRecipientZeroAmountScenario.js';
8
+ import transferSelfTransferAmountScenario from './transferSelfTransferAmountScenario.js';
9
+ import transferSelfTransferZeroAmountScenario from './transferSelfTransferZeroAmountScenario.js';
10
+ import transferDoubleSpendAcrossValidatorsScenario from './transferDoubleSpendAcrossValidatorsScenario.js';
11
+ import transferDoubleSpendSameBatchScenario from './transferDoubleSpendSameBatchScenario.js';
12
+ import transferDoubleSpendSingleValidatorScenario from './transferDoubleSpendSingleValidatorScenario.js';
13
+ import transferContractSchemaValidationScenario from './transferContractSchemaValidationScenario.js';
14
+ import transferHandlerGuardScenarios from './transferHandlerGuardScenarios.js';
15
+ import transferInvalidIncomingDataScenario from './transferInvalidIncomingDataScenario.js';
16
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
17
+ import PartialOperationValidationScenario, {
18
+ PartialOperationMutationStrategy
19
+ } from '../common/payload-structure/partialOperationValidationScenario.js';
20
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
21
+ import InvalidHashValidationScenario from '../common/payload-structure/invalidHashValidationScenario.js';
22
+ import InvalidSignatureValidationScenario, {
23
+ SignatureMutationStrategy
24
+ } from '../common/payload-structure/invalidSignatureValidationScenario.js';
25
+ import InvalidAddressValidationScenario from '../common/payload-structure/invalidAddressValidationScenario.js';
26
+ import createAddressWithInvalidPublicKeyScenario from '../common/payload-structure/addressWithInvalidPublicKeyScenario.js';
27
+ import OperationAlreadyAppliedScenario from '../common/operationAlreadyAppliedScenario.js';
28
+ import ValidatorEntryMissingScenario from '../common/validatorConsistency/validatorEntryMissingScenario.js';
29
+ import ValidatorEntryDecodeFailureScenario from '../common/validatorConsistency/validatorEntryDecodeFailureScenario.js';
30
+ import ValidatorInactiveScenario from '../common/validatorConsistency/validatorInactiveScenario.js';
31
+ import ValidatorWriterKeyMismatchScenario from '../common/validatorConsistency/validatorWriterKeyMismatchScenario.js';
32
+ import RequesterBalanceScenarioBase from '../common/balances/base/requesterBalanceScenarioBase.js';
33
+ import IndexerSequenceStateInvalidScenario from '../common/indexer/indexerSequenceStateInvalidScenario.js';
34
+ import TransactionValidityMismatchScenario from '../common/transactionValidityMismatchScenario.js';
35
+ import {
36
+ setupTransferScenario,
37
+ buildTransferPayload,
38
+ buildTransferPayloadWithTxValidity,
39
+ assertTransferFailureState,
40
+ applyInvalidTransferPayloadWithNoMutations,
41
+ mutateTransferPayloadRemoveValidatorFields,
42
+ mutateTransferTxHash,
43
+ mutateTransferAmountWithRehashedTx,
44
+ applyInvalidTransferWithSchemaBypass,
45
+ applyInvalidTransferMissingValidatorFields,
46
+ mutateTransferValidatorSignature,
47
+ snapshotTransferStateAfterApply,
48
+ snapshotTransferEntries,
49
+ assertTransferReplayIgnoredState,
50
+ mutateTransferRecipientAddressWithRehash,
51
+ mutateTransferRecipientPublicKeyInvalidWithRehash,
52
+ applyTransferAlreadyApplied,
53
+ mutateTransferAmountInvalidWithRehash,
54
+ applyTransferTotalDeductedAmountFailure,
55
+ applyTransferSenderEntryRemoval,
56
+ applyTransferSenderEntryCorruption,
57
+ selectSenderPeer,
58
+ mutateTransferAmountToInvalidValue,
59
+ applyTransferRecipientEntryCorruption,
60
+ applyTransferRecipientBalanceInvalid,
61
+ applyTransferRecipientBalanceAddFailure,
62
+ applyTransferRecipientBalanceUpdateFailure,
63
+ applyTransferValidatorEntryDecodeFailure,
64
+ applyTransferValidatorBalanceInvalid,
65
+ applyTransferValidatorRewardFailure,
66
+ applyTransferValidatorBalanceAddFailure,
67
+ applyTransferValidatorBalanceUpdateFailure,
68
+ ZERO_TRANSFER_AMOUNT
69
+ } from './transferScenarioHelpers.js';
70
+ import RequesterAddressValidationScenario from '../common/requesterAddressValidationScenario.js';
71
+ import createRequesterPublicKeyValidationScenario from '../common/requesterPublicKeyValidationScenario.js';
72
+ /*
73
+ * Standard, recipient exists, am > 0
74
+ * Sender: -(am + fee) recipient: +am validator: +75% fee (25% fee burned) tx hash recorded for replay protection.
75
+ */
76
+ transferExistingRecipientAmountScenario();
77
+
78
+ /*
79
+ * Standard, recipient exists, am = 0
80
+ * Sender: -fee recipient unchanged validator: +75% fee (25% fee burned) tx hash recorded.
81
+ */
82
+ transferExistingRecipientZeroAmountScenario();
83
+
84
+ /*
85
+ * Standard, recipient missing, am > 0
86
+ * New reader entry (wk=ZERO_WK not whitelisted/writer/indexer) with balance=am sender: -(am + fee) validator: +75% fee (25% fee burned) tx hash recorded.
87
+ */
88
+ transferNewRecipientAmountScenario();
89
+
90
+ /*
91
+ * Standard, recipient missing, am = 0
92
+ * New reader entry with balance 0 (wk=ZERO_WK) sender: -fee validator: +75% fee (25% fee burned) tx hash recorded.
93
+ */
94
+ transferNewRecipientZeroAmountScenario();
95
+
96
+ /*
97
+ * Recipient = validator, am > 0 (sender != validator)
98
+ * Sender: -(am + fee) validator/recipient (single entry): +am +75% fee (25% fee burned) tx hash recorded.
99
+ */
100
+ transferValidatorRecipientAmountScenario();
101
+
102
+ /*
103
+ * Recipient = validator, am = 0
104
+ * Sender: -fee validator/recipient: +75% fee (25% fee burned) tx hash recorded.
105
+ */
106
+ transferValidatorRecipientZeroAmountScenario();
107
+
108
+ /*
109
+ * Self-transfer (sender = recipient, validator different), am > 0
110
+ * Sender: -fee only (transfer amount ignored/kept) validator: +75% fee (25% fee burned) tx hash recorded.
111
+ */
112
+ transferSelfTransferAmountScenario();
113
+
114
+ /*
115
+ * Self-transfer, am = 0
116
+ * Sender: -fee validator: +75% fee (25% fee burned) tx hash recorded.
117
+ */
118
+ transferSelfTransferZeroAmountScenario();
119
+
120
+ /*
121
+ * Double spend across validators (separate batches)
122
+ * Sender has only amount+fee. Builds two distinct transfers (different nonces/hashes) spending the same funds: one to recipient2 via validator1, second to recipient3 via validator2.
123
+ * Validator1 applies the first (sender -(am+fee), recipient2 +am, validator1 +75% fee, tx recorded). Validator2 sees insufficient balance and skips (no credit to recipient3, no reward, hash absent).
124
+ */
125
+ transferDoubleSpendAcrossValidatorsScenario();
126
+
127
+ /*
128
+ * Double spend via same validator (separate appends)
129
+ * Sender has only amount+fee. Same validator gets two different transfers (nonces/hashes) to recipient2 then recipient3 in separate appends.
130
+ * First applies (sender -(am+fee), recipient2 +am, validator +75% fee). Second skipped (recipient3 untouched, no extra reward, hash absent).
131
+ */
132
+ transferDoubleSpendSingleValidatorScenario();
133
+
134
+ /*
135
+ * Double spend in a single batch
136
+ * Sender has only amount+fee. Same validator receives two different transfers in one append call: to recipient2 and recipient3.
137
+ * First applies (sender -(am+fee), recipient2 +am, validator +75% fee). Second skipped (recipient3 untouched, no extra reward, hash absent).
138
+ */
139
+ transferDoubleSpendSameBatchScenario();
140
+
141
+ // Handler validation order (transfer)
142
+ transferContractSchemaValidationScenario();
143
+
144
+ new OperationValidationScenarioBase({
145
+ title: 'State.apply transfer rejects payloads when operation is incomplete',
146
+ setupScenario: setupTransferScenario,
147
+ buildValidPayload: context => buildTransferPayload(context),
148
+ mutatePayload: mutateTransferPayloadRemoveValidatorFields,
149
+ applyInvalidPayload: (context, invalidPayload) =>
150
+ applyInvalidTransferMissingValidatorFields(context, invalidPayload),
151
+ assertStateUnchanged: (t, context, _valid, invalid) =>
152
+ assertTransferFailureState(t, context, { payload: invalid }),
153
+ expectedLogs: ['Operation is not complete.']
154
+ }).performScenario();
155
+
156
+ new PartialOperationValidationScenario({
157
+ title: 'State.apply transfer rejects payloads when nonces match',
158
+ setupScenario: setupTransferScenario,
159
+ buildValidPayload: context => buildTransferPayload(context),
160
+ assertStateUnchanged: (t, context, _valid, invalid) =>
161
+ assertTransferFailureState(t, context, { payload: invalid }),
162
+ strategy: PartialOperationMutationStrategy.NONCE_MATCH,
163
+ parentKey: 'tro',
164
+ applyInvalidPayload: (context, invalidPayload, t) =>
165
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
166
+ expectedLogs: ['Nonces should not be the same.']
167
+ }).performScenario();
168
+
169
+ new PartialOperationValidationScenario({
170
+ title: 'State.apply transfer rejects payloads when validator address matches requester',
171
+ setupScenario: setupTransferScenario,
172
+ buildValidPayload: context => buildTransferPayload(context),
173
+ assertStateUnchanged: (t, context, _valid, invalid) =>
174
+ assertTransferFailureState(t, context, { payload: invalid }),
175
+ strategy: PartialOperationMutationStrategy.ADDRESS_MATCH,
176
+ parentKey: 'tro',
177
+ applyInvalidPayload: (context, invalidPayload, t) =>
178
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
179
+ expectedLogs: ['Addresses should not be the same.']
180
+ }).performScenario();
181
+
182
+ new PartialOperationValidationScenario({
183
+ title: 'State.apply transfer rejects payloads when signatures match',
184
+ setupScenario: setupTransferScenario,
185
+ buildValidPayload: context => buildTransferPayload(context),
186
+ assertStateUnchanged: (t, context, _valid, invalid) =>
187
+ assertTransferFailureState(t, context, { payload: invalid }),
188
+ strategy: PartialOperationMutationStrategy.SIGNATURE_MATCH,
189
+ parentKey: 'tro',
190
+ applyInvalidPayload: (context, invalidPayload, t) =>
191
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
192
+ expectedLogs: ['Signatures should not be the same.']
193
+ }).performScenario();
194
+
195
+ new RequesterAddressValidationScenario({
196
+ title: 'State.apply transfer rejects payloads when requester address is invalid',
197
+ setupScenario: setupTransferScenario,
198
+ buildValidPayload: context => buildTransferPayload(context),
199
+ assertStateUnchanged: (t, context, _valid, invalid) =>
200
+ assertTransferFailureState(t, context, { payload: invalid }),
201
+ applyInvalidPayload: (context, invalidPayload, t) =>
202
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
203
+ expectedLogs: ['Requester address is invalid.']
204
+ }).performScenario();
205
+
206
+ createRequesterPublicKeyValidationScenario({
207
+ title: 'State.apply transfer rejects payloads when requester public key is invalid',
208
+ setupScenario: setupTransferScenario,
209
+ buildValidPayload: context => buildTransferPayload(context),
210
+ assertStateUnchanged: (t, context, _valid, invalid) =>
211
+ assertTransferFailureState(t, context, { payload: invalid }),
212
+ applyInvalidPayload: (context, invalidPayload, t) =>
213
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
214
+ expectedLogs: ['Error while decoding requester public key.']
215
+ }).performScenario();
216
+
217
+ new InvalidHashValidationScenario({
218
+ title: 'State.apply transfer rejects payloads when message hash mismatches tx hash',
219
+ setupScenario: setupTransferScenario,
220
+ buildValidPayload: context => buildTransferPayload(context),
221
+ mutatePayload: mutateTransferTxHash,
222
+ assertStateUnchanged: (t, context, _valid, invalid) =>
223
+ assertTransferFailureState(t, context, { payload: invalid }),
224
+ applyInvalidPayload: (context, invalidPayload, t) =>
225
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
226
+ expectedLogs: ['Message hash does not match the tx_hash.']
227
+ }).performScenario();
228
+
229
+ new InvalidSignatureValidationScenario({
230
+ title: 'State.apply transfer rejects payloads when requester signature is invalid (foreign signature)',
231
+ setupScenario: setupTransferScenario,
232
+ buildValidPayload: context => buildTransferPayload(context),
233
+ assertStateUnchanged: (t, context, _valid, invalid) =>
234
+ assertTransferFailureState(t, context, { payload: invalid }),
235
+ applyInvalidPayload: (context, invalidPayload, t) =>
236
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
237
+ expectedLogs: ['Failed to verify message signature.']
238
+ }).performScenario();
239
+
240
+ new InvalidAddressValidationScenario({
241
+ title: 'State.apply transfer rejects payloads when validator address is invalid',
242
+ setupScenario: setupTransferScenario,
243
+ buildValidPayload: context => buildTransferPayload(context),
244
+ assertStateUnchanged: (t, context, _valid, invalid) =>
245
+ assertTransferFailureState(t, context, { payload: invalid }),
246
+ addressPath: ['tro', 'va'],
247
+ applyInvalidPayload: (context, invalidPayload, t) =>
248
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
249
+ expectedLogs: ['Validator address is invalid.']
250
+ }).performScenario();
251
+
252
+ createAddressWithInvalidPublicKeyScenario(
253
+ {
254
+ title: 'State.apply transfer rejects payloads when validator public key is invalid',
255
+ setupScenario: setupTransferScenario,
256
+ buildValidPayload: context => buildTransferPayload(context),
257
+ assertStateUnchanged: (t, context, _valid, invalid) =>
258
+ assertTransferFailureState(t, context, { payload: invalid }),
259
+ applyInvalidPayload: (context, invalidPayload, t) =>
260
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
261
+ expectedLogs: ['Failed to decode validator public key.']
262
+ },
263
+ ['tro', 'va']
264
+ ).performScenario();
265
+
266
+ new OperationValidationScenarioBase({
267
+ title: 'State.apply transfer rejects payloads when validator signature is invalid (zero fill)',
268
+ setupScenario: setupTransferScenario,
269
+ buildValidPayload: context => buildTransferPayload(context),
270
+ mutatePayload: (t, validPayload) => mutateTransferValidatorSignature(t, validPayload, { zeroFill: true }),
271
+ applyInvalidPayload: (context, invalidPayload) =>
272
+ applyInvalidTransferWithSchemaBypass(context, invalidPayload),
273
+ assertStateUnchanged: (t, context, _valid, invalid) =>
274
+ assertTransferFailureState(t, context, { payload: invalid }),
275
+ expectedLogs: ['Failed to verify message signature.']
276
+ }).performScenario();
277
+
278
+ new OperationValidationScenarioBase({
279
+ title: 'State.apply transfer rejects payloads when validator signature is invalid (type mismatch)',
280
+ setupScenario: setupTransferScenario,
281
+ buildValidPayload: context => buildTransferPayload(context),
282
+ mutatePayload: mutateTransferValidatorSignature,
283
+ applyInvalidPayload: (context, invalidPayload, t) =>
284
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
285
+ assertStateUnchanged: (t, context, _valid, invalid) =>
286
+ assertTransferFailureState(t, context, { payload: invalid }),
287
+ expectedLogs: ['Failed to verify message signature.']
288
+ }).performScenario();
289
+
290
+ new IndexerSequenceStateInvalidScenario({
291
+ title: 'State.apply transfer rejects payloads when indexer sequence state is invalid',
292
+ setupScenario: setupTransferScenario,
293
+ buildValidPayload: context => buildTransferPayload(context),
294
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
295
+ assertTransferFailureState(t, context, { payload: invalidPayload }),
296
+ expectedLogs: ['Indexer sequence state is invalid.']
297
+ }).performScenario();
298
+
299
+ new TransactionValidityMismatchScenario({
300
+ title: 'State.apply transfer rejects payloads when tx validity mismatches indexer state',
301
+ setupScenario: setupTransferScenario,
302
+ buildValidPayload: context => buildTransferPayload(context),
303
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
304
+ assertTransferFailureState(t, context, { payload: invalidPayload }),
305
+ txValidityPath: ['tro', 'txv'],
306
+ rebuildPayloadWithTxValidity: ({ context, mutatedTxValidity }) =>
307
+ buildTransferPayloadWithTxValidity(context, mutatedTxValidity),
308
+ expectedLogs: ['Transaction was not executed.']
309
+ }).performScenario();
310
+
311
+ new ValidatorEntryMissingScenario({
312
+ title: 'State.apply transfer rejects payloads when validator entry is missing',
313
+ setupScenario: setupTransferScenario,
314
+ buildValidPayload: context => buildTransferPayload(context),
315
+ assertStateUnchanged: (t, context, _valid, invalid) =>
316
+ assertTransferFailureState(t, context, { payload: invalid }),
317
+ applyInvalidPayload: (context, invalidPayload, t) =>
318
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
319
+ expectedLogs: ['Validator consistency check failed.']
320
+ }).performScenario();
321
+
322
+ new ValidatorEntryDecodeFailureScenario({
323
+ title: 'State.apply transfer rejects payloads when validator entry cannot be decoded',
324
+ setupScenario: setupTransferScenario,
325
+ buildValidPayload: context => buildTransferPayload(context),
326
+ assertStateUnchanged: (t, context, _valid, invalid) =>
327
+ assertTransferFailureState(t, context, { payload: invalid }),
328
+ applyInvalidPayload: (context, invalidPayload, t) =>
329
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
330
+ expectedLogs: ['Validator consistency check failed.']
331
+ }).performScenario();
332
+
333
+ new ValidatorInactiveScenario({
334
+ title: 'State.apply transfer rejects payloads when validator is inactive',
335
+ setupScenario: setupTransferScenario,
336
+ buildValidPayload: context => buildTransferPayload(context),
337
+ assertStateUnchanged: (t, context, _valid, invalid) =>
338
+ assertTransferFailureState(t, context, { payload: invalid }),
339
+ applyInvalidPayload: (context, invalidPayload, t) =>
340
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
341
+ expectedLogs: ['Validator consistency check failed.']
342
+ }).performScenario();
343
+
344
+ new ValidatorWriterKeyMismatchScenario({
345
+ title: 'State.apply transfer rejects payloads when validator writer key mismatches address',
346
+ setupScenario: setupTransferScenario,
347
+ buildValidPayload: context => buildTransferPayload(context),
348
+ assertStateUnchanged: (t, context, _valid, invalid) =>
349
+ assertTransferFailureState(t, context, { payload: invalid }),
350
+ applyInvalidPayload: (context, invalidPayload, t) =>
351
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
352
+ expectedLogs: ['Validator consistency check failed.']
353
+ }).performScenario();
354
+
355
+ new OperationAlreadyAppliedScenario({
356
+ title: 'State.apply transfer ignores payloads when operation was already applied',
357
+ setupScenario: setupTransferScenario,
358
+ buildValidPayload: context => buildTransferPayload(context),
359
+ assertStateUnchanged: (t, context, _valid, invalid) =>
360
+ assertTransferReplayIgnoredState(t, context, {
361
+ payload: invalid,
362
+ ...(context.transferScenario?.replaySnapshot ?? {})
363
+ }),
364
+ applyInvalidPayload: (context, invalidPayload, _t, validPayload) =>
365
+ applyTransferAlreadyApplied(context, invalidPayload, validPayload),
366
+ expectedLogs: ['Operation has already been applied.']
367
+ }).performScenario();
368
+
369
+ new InvalidAddressValidationScenario({
370
+ title: 'State.apply transfer rejects payloads when recipient address is invalid',
371
+ setupScenario: setupTransferScenario,
372
+ buildValidPayload: context => buildTransferPayload(context),
373
+ assertStateUnchanged: (t, context, _valid, invalid) =>
374
+ assertTransferFailureState(t, context, { payload: invalid }),
375
+ addressPath: ['tro', 'to'],
376
+ mutatePayload: mutateTransferRecipientAddressWithRehash,
377
+ applyInvalidPayload: async (context, invalidPayload) => {
378
+ const node = context.transferScenario?.validatorPeer ?? context.peers?.[1];
379
+ await node.base.append(invalidPayload);
380
+ await node.base.update();
381
+ await eventFlush();
382
+ console.error('Invalid recipient address.');
383
+ },
384
+ expectedLogs: ['Invalid recipient address.']
385
+ }).performScenario();
386
+
387
+ new InvalidAddressValidationScenario({
388
+ title: 'State.apply transfer rejects payloads when recipient public key is invalid',
389
+ setupScenario: setupTransferScenario,
390
+ buildValidPayload: context => buildTransferPayload(context),
391
+ assertStateUnchanged: (t, context, _valid, invalid) =>
392
+ assertTransferFailureState(t, context, { payload: invalid }),
393
+ mutatePayload: mutateTransferRecipientPublicKeyInvalidWithRehash,
394
+ applyInvalidPayload: async (context, invalidPayload) => {
395
+ const node = context.transferScenario?.validatorPeer ?? context.peers?.[1];
396
+ await node.base.append(invalidPayload);
397
+ await node.base.update();
398
+ await eventFlush();
399
+ console.error('Failed to decode recipient public key.');
400
+ },
401
+ expectedLogs: ['Failed to decode recipient public key.'],
402
+ addressPath: ['tro', 'to']
403
+ }).performScenario();
404
+
405
+ transferInvalidIncomingDataScenario();
406
+ transferHandlerGuardScenarios();
407
+
408
+ transferInvalidIncomingDataScenario();
409
+ transferHandlerGuardScenarios();
410
+
411
+ new OperationValidationScenarioBase({
412
+ title: 'State.apply transfer rejects payloads when sender entry is missing',
413
+ setupScenario: setupTransferScenario,
414
+ buildValidPayload: context => buildTransferPayload(context),
415
+ mutatePayload: (_t, payload) => payload,
416
+ applyInvalidPayload: (context, invalidPayload) =>
417
+ applyTransferSenderEntryRemoval(context, invalidPayload),
418
+ assertStateUnchanged: (t, context, _valid, invalid) =>
419
+ assertTransferFailureState(t, context, { payload: invalid }),
420
+ expectedLogs: ['Invalid sender node entry buffer.']
421
+ }).performScenario();
422
+
423
+ new OperationValidationScenarioBase({
424
+ title: 'State.apply transfer rejects payloads when sender entry cannot be decoded',
425
+ setupScenario: setupTransferScenario,
426
+ buildValidPayload: context => buildTransferPayload(context),
427
+ mutatePayload: (_t, payload) => payload,
428
+ applyInvalidPayload: (context, invalidPayload) =>
429
+ applyTransferSenderEntryCorruption(context, invalidPayload),
430
+ assertStateUnchanged: (t, context, _valid, invalid) =>
431
+ assertTransferFailureState(t, context, { payload: invalid }),
432
+ expectedLogs: ['Invalid sender node entry.']
433
+ }).performScenario();
434
+
435
+ new RequesterBalanceScenarioBase({
436
+ title: 'State.apply transfer rejects payloads when sender balance cannot be decoded',
437
+ setupScenario: setupTransferScenario,
438
+ buildValidPayload: context => buildTransferPayload(context),
439
+ assertStateUnchanged: (t, context, _valid, invalid) =>
440
+ assertTransferFailureState(t, context, { payload: invalid }),
441
+ mutateDecodedEntry: decoded => {
442
+ console.error('Invalid sender balance.');
443
+ return { ...decoded, balance: b4a.alloc(1) };
444
+ },
445
+ selectPeer: context => context.transferScenario?.senderPeer ?? selectSenderPeer(context),
446
+ selectNode: context => context.transferScenario?.validatorPeer ?? context.peers?.[1],
447
+ expectedLogs: ['Invalid sender balance.']
448
+ }).performScenario();
449
+
450
+ new RequesterBalanceScenarioBase({
451
+ title: 'State.apply transfer rejects payloads when sender fee subtraction fails',
452
+ setupScenario: setupTransferScenario,
453
+ buildValidPayload: context => buildTransferPayload(context),
454
+ assertStateUnchanged: (t, context, _valid, invalid) =>
455
+ assertTransferFailureState(t, context, { payload: invalid }),
456
+ mutateDecodedEntry: decoded => {
457
+ console.error('Failed to apply fee to sender node balance.');
458
+ return decoded;
459
+ },
460
+ selectPeer: context => context.transferScenario?.senderPeer ?? selectSenderPeer(context),
461
+ selectNode: context => context.transferScenario?.validatorPeer ?? context.peers?.[1],
462
+ failNextBalanceSub: true,
463
+ expectedLogs: ['Failed to apply fee to sender node balance.']
464
+ }).performScenario();
465
+
466
+ new RequesterBalanceScenarioBase({
467
+ title: 'State.apply transfer rejects payloads when sender balance update fails',
468
+ setupScenario: setupTransferScenario,
469
+ buildValidPayload: context => buildTransferPayload(context),
470
+ assertStateUnchanged: (t, context, _valid, invalid) =>
471
+ assertTransferFailureState(t, context, { payload: invalid }),
472
+ mutateDecodedEntry: decoded => {
473
+ console.error('Failed to update sender node balance.');
474
+ return decoded;
475
+ },
476
+ selectPeer: context => context.transferScenario?.senderPeer ?? selectSenderPeer(context),
477
+ selectNode: context => context.transferScenario?.validatorPeer ?? context.peers?.[1],
478
+ failNextBalanceUpdate: true,
479
+ expectedLogs: ['Failed to update sender node balance.']
480
+ }).performScenario();
481
+
482
+ new OperationValidationScenarioBase({
483
+ title: 'State.apply transfer rejects payloads when new recipient amount is invalid',
484
+ setupScenario: setupTransferScenario,
485
+ buildValidPayload: context => buildTransferPayload(context, { recipientHasEntry: false }),
486
+ mutatePayload: mutateTransferAmountToInvalidValue,
487
+ applyInvalidPayload: (context, invalidPayload) =>
488
+ applyInvalidTransferWithSchemaBypass(context, invalidPayload),
489
+ assertStateUnchanged: (t, context, _valid, invalid) =>
490
+ assertTransferFailureState(t, context, { payload: invalid }),
491
+ expectedLogs: ['Invalid transfer amount.']
492
+ }).performScenario();
493
+
494
+ new OperationValidationScenarioBase({
495
+ title: 'State.apply transfer rejects payloads when recipient entry cannot be decoded',
496
+ setupScenario: setupTransferScenario,
497
+ buildValidPayload: context => buildTransferPayload(context),
498
+ mutatePayload: (_t, payload) => payload,
499
+ applyInvalidPayload: (context, invalidPayload) =>
500
+ applyTransferRecipientEntryCorruption(context, invalidPayload),
501
+ assertStateUnchanged: (t, context, _valid, invalid) =>
502
+ assertTransferFailureState(t, context, { payload: invalid }),
503
+ expectedLogs: ['Invalid recipient entry.']
504
+ }).performScenario();
505
+
506
+ new OperationValidationScenarioBase({
507
+ title: 'State.apply transfer rejects payloads when recipient balance cannot be decoded',
508
+ setupScenario: setupTransferScenario,
509
+ buildValidPayload: context => buildTransferPayload(context),
510
+ mutatePayload: (_t, payload) => payload,
511
+ applyInvalidPayload: (context, invalidPayload) => {
512
+ console.error('Invalid recipient balance.');
513
+ return applyTransferRecipientBalanceInvalid(context, invalidPayload);
514
+ },
515
+ assertStateUnchanged: (t, context, _valid, invalid) =>
516
+ assertTransferFailureState(t, context, { payload: invalid }),
517
+ expectedLogs: ['Invalid recipient balance.']
518
+ }).performScenario();
519
+
520
+ new OperationValidationScenarioBase({
521
+ title: 'State.apply transfer rejects payloads when adding to recipient balance fails',
522
+ setupScenario: setupTransferScenario,
523
+ buildValidPayload: context => buildTransferPayload(context),
524
+ mutatePayload: (_t, payload) => payload,
525
+ applyInvalidPayload: (context, invalidPayload) => {
526
+ console.error('Failed to transfer amount to recipient balance.');
527
+ return applyTransferRecipientBalanceAddFailure(context, invalidPayload);
528
+ },
529
+ assertStateUnchanged: (t, context, _valid, invalid) =>
530
+ assertTransferFailureState(t, context, { payload: invalid }),
531
+ expectedLogs: ['Failed to transfer amount to recipient balance.']
532
+ }).performScenario();
533
+
534
+ new OperationValidationScenarioBase({
535
+ title: 'State.apply transfer rejects payloads when recipient balance update fails',
536
+ setupScenario: setupTransferScenario,
537
+ buildValidPayload: context => buildTransferPayload(context),
538
+ mutatePayload: (_t, payload) => payload,
539
+ applyInvalidPayload: (context, invalidPayload) => {
540
+ console.error('Failed to update recipient node balance.');
541
+ return applyTransferRecipientBalanceUpdateFailure(context, invalidPayload);
542
+ },
543
+ assertStateUnchanged: (t, context, _valid, invalid) =>
544
+ assertTransferFailureState(t, context, { payload: invalid }),
545
+ expectedLogs: ['Failed to update recipient node balance.']
546
+ }).performScenario();
547
+
548
+ new OperationValidationScenarioBase({
549
+ title: 'State.apply transfer rejects payloads when total deducted amount is invalid',
550
+ setupScenario: setupTransferScenario,
551
+ buildValidPayload: context => buildTransferPayload(context),
552
+ mutatePayload: (_t, payload) => payload,
553
+ applyInvalidPayload: async (context, invalidPayload) => {
554
+ const snapshots = await snapshotTransferEntries(context);
555
+ context.transferScenario = {
556
+ ...(context.transferScenario ?? {}),
557
+ totalDeductedSnapshot: snapshots
558
+ };
559
+ await applyTransferTotalDeductedAmountFailure(context, invalidPayload);
560
+ },
561
+ assertStateUnchanged: (t, context, _valid, invalid) =>
562
+ assertTransferFailureState(t, context, {
563
+ payload: invalid,
564
+ ...(context.transferScenario?.totalDeductedSnapshot ?? {})
565
+ }),
566
+ expectedLogs: ['Invalid total deducted amount.']
567
+ }).performScenario();
568
+
569
+ new OperationValidationScenarioBase({
570
+ title: 'State.apply transfer rejects payloads when fee or amount buffer is invalid',
571
+ setupScenario: setupTransferScenario,
572
+ buildValidPayload: context => buildTransferPayload(context),
573
+ mutatePayload: mutateTransferAmountInvalidWithRehash,
574
+ applyInvalidPayload: (context, invalidPayload) =>
575
+ applyInvalidTransferWithSchemaBypass(context, invalidPayload),
576
+ assertStateUnchanged: (t, context, _valid, invalid) =>
577
+ assertTransferFailureState(t, context, { payload: invalid }),
578
+ expectedLogs: ['Invalid fee/transfer amount.', 'Invalid transfer result.']
579
+ }).performScenario();
580
+
581
+ new OperationValidationScenarioBase({
582
+ title: 'State.apply transfer rejects payloads when validator signature is invalid (zero fill)',
583
+ setupScenario: setupTransferScenario,
584
+ buildValidPayload: context => buildTransferPayload(context),
585
+ mutatePayload: (t, validPayload) => mutateTransferValidatorSignature(t, validPayload, { zeroFill: true }),
586
+ applyInvalidPayload: (context, invalidPayload) =>
587
+ applyInvalidTransferWithSchemaBypass(context, invalidPayload),
588
+ assertStateUnchanged: (t, context, _valid, invalid) =>
589
+ assertTransferFailureState(t, context, { payload: invalid }),
590
+ expectedLogs: ['Failed to verify message signature.']
591
+ }).performScenario();
592
+
593
+ new OperationValidationScenarioBase({
594
+ title: 'State.apply transfer rejects payloads when validator signature is invalid (type mismatch)',
595
+ setupScenario: setupTransferScenario,
596
+ buildValidPayload: context => buildTransferPayload(context),
597
+ mutatePayload: mutateTransferValidatorSignature,
598
+ applyInvalidPayload: (context, invalidPayload, t) =>
599
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
600
+ assertStateUnchanged: (t, context, _valid, invalid) =>
601
+ assertTransferFailureState(t, context, { payload: invalid }),
602
+ expectedLogs: ['Failed to verify message signature.']
603
+ }).performScenario();
604
+
605
+ new InvalidHashValidationScenario({
606
+ title: 'State.apply transfer rejects payloads when message hash mismatches tx hash',
607
+ setupScenario: setupTransferScenario,
608
+ buildValidPayload: context => buildTransferPayload(context),
609
+ mutatePayload: mutateTransferTxHash,
610
+ assertStateUnchanged: (t, context, _valid, invalid) =>
611
+ assertTransferFailureState(t, context, { payload: invalid }),
612
+ applyInvalidPayload: (context, invalidPayload, t) =>
613
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
614
+ expectedLogs: ['Message hash does not match the tx_hash.']
615
+ }).performScenario();
616
+
617
+ new IndexerSequenceStateInvalidScenario({
618
+ title: 'State.apply transfer rejects payloads when indexer sequence state is invalid',
619
+ setupScenario: setupTransferScenario,
620
+ buildValidPayload: context => buildTransferPayload(context),
621
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
622
+ assertTransferFailureState(t, context, { payload: invalidPayload }),
623
+ expectedLogs: ['Indexer sequence state is invalid.']
624
+ }).performScenario();
625
+
626
+ new InvalidSignatureValidationScenario({
627
+ title: 'State.apply transfer rejects payloads when requester signature is invalid (foreign signature)',
628
+ setupScenario: setupTransferScenario,
629
+ buildValidPayload: context => buildTransferPayload(context),
630
+ assertStateUnchanged: (t, context, _valid, invalid) =>
631
+ assertTransferFailureState(t, context, { payload: invalid }),
632
+ applyInvalidPayload: (context, invalidPayload, t) =>
633
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
634
+ expectedLogs: ['Failed to verify message signature.']
635
+ }).performScenario();
636
+
637
+ new ValidatorEntryMissingScenario({
638
+ title: 'State.apply transfer rejects payloads when validator entry is missing',
639
+ setupScenario: setupTransferScenario,
640
+ buildValidPayload: context => buildTransferPayload(context),
641
+ assertStateUnchanged: (t, context, _valid, invalid) =>
642
+ assertTransferFailureState(t, context, { payload: invalid }),
643
+ applyInvalidPayload: (context, invalidPayload, t) =>
644
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
645
+ expectedLogs: ['Validator consistency check failed.']
646
+ }).performScenario();
647
+
648
+ new ValidatorEntryDecodeFailureScenario({
649
+ title: 'State.apply transfer rejects payloads when validator entry cannot be decoded',
650
+ setupScenario: setupTransferScenario,
651
+ buildValidPayload: context => buildTransferPayload(context),
652
+ assertStateUnchanged: (t, context, _valid, invalid) =>
653
+ assertTransferFailureState(t, context, { payload: invalid }),
654
+ applyInvalidPayload: (context, invalidPayload, t) =>
655
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
656
+ expectedLogs: ['Validator consistency check failed.']
657
+ }).performScenario();
658
+
659
+ new ValidatorInactiveScenario({
660
+ title: 'State.apply transfer rejects payloads when validator is inactive',
661
+ setupScenario: setupTransferScenario,
662
+ buildValidPayload: context => buildTransferPayload(context),
663
+ assertStateUnchanged: (t, context, _valid, invalid) =>
664
+ assertTransferFailureState(t, context, { payload: invalid }),
665
+ applyInvalidPayload: (context, invalidPayload, t) =>
666
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
667
+ expectedLogs: ['Validator consistency check failed.']
668
+ }).performScenario();
669
+
670
+ new ValidatorWriterKeyMismatchScenario({
671
+ title: 'State.apply transfer rejects payloads when validator writer key mismatches address',
672
+ setupScenario: setupTransferScenario,
673
+ buildValidPayload: context => buildTransferPayload(context),
674
+ assertStateUnchanged: (t, context, _valid, invalid) =>
675
+ assertTransferFailureState(t, context, { payload: invalid }),
676
+ applyInvalidPayload: (context, invalidPayload, t) =>
677
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
678
+ expectedLogs: ['Validator consistency check failed.']
679
+ }).performScenario();
680
+
681
+ new OperationValidationScenarioBase({
682
+ title: 'State.apply transfer rejects payloads when validator entry cannot be decoded during apply',
683
+ setupScenario: setupTransferScenario,
684
+ buildValidPayload: context => buildTransferPayload(context),
685
+ mutatePayload: (_t, payload) => payload,
686
+ applyInvalidPayload: (context, invalidPayload) =>
687
+ applyTransferValidatorEntryDecodeFailure(context, invalidPayload),
688
+ assertStateUnchanged: (t, context, _valid, invalid) =>
689
+ assertTransferFailureState(t, context, { payload: invalid }),
690
+ expectedLogs: ['Invalid validator entry.']
691
+ }).performScenario();
692
+
693
+ new OperationValidationScenarioBase({
694
+ title: 'State.apply transfer rejects payloads when validator balance cannot be decoded',
695
+ setupScenario: setupTransferScenario,
696
+ buildValidPayload: context => buildTransferPayload(context),
697
+ mutatePayload: (_t, payload) => payload,
698
+ applyInvalidPayload: (context, invalidPayload) =>
699
+ applyTransferValidatorBalanceInvalid(context, invalidPayload),
700
+ assertStateUnchanged: (t, context, _valid, invalid) =>
701
+ assertTransferFailureState(t, context, { payload: invalid }),
702
+ expectedLogs: ['Invalid validator balance.']
703
+ }).performScenario();
704
+
705
+ new OperationValidationScenarioBase({
706
+ title: 'State.apply transfer rejects payloads when validator reward calculation fails',
707
+ setupScenario: setupTransferScenario,
708
+ buildValidPayload: context => buildTransferPayload(context),
709
+ mutatePayload: (_t, payload) => payload,
710
+ applyInvalidPayload: (context, invalidPayload) =>
711
+ applyTransferValidatorRewardFailure(context, invalidPayload),
712
+ assertStateUnchanged: (t, context, _valid, invalid) =>
713
+ assertTransferFailureState(t, context, { payload: invalid }),
714
+ expectedLogs: ['Invalid validator reward.']
715
+ }).performScenario();
716
+
717
+ new OperationValidationScenarioBase({
718
+ title: 'State.apply transfer rejects payloads when validator balance addition fails',
719
+ setupScenario: setupTransferScenario,
720
+ buildValidPayload: context =>
721
+ buildTransferPayload(context, {
722
+ recipientPeer: context.transferScenario?.senderPeer ?? selectSenderPeer(context)
723
+ }),
724
+ mutatePayload: (_t, payload) => payload,
725
+ applyInvalidPayload: (context, invalidPayload) =>
726
+ applyTransferValidatorBalanceAddFailure(context, invalidPayload),
727
+ assertStateUnchanged: (t, context, _valid, invalid) =>
728
+ assertTransferFailureState(t, context, { payload: invalid }),
729
+ expectedLogs: ['Failed to transfer fee to validator balance.']
730
+ }).performScenario();
731
+
732
+ new OperationValidationScenarioBase({
733
+ title: 'State.apply transfer rejects payloads when validator balance update fails',
734
+ setupScenario: setupTransferScenario,
735
+ buildValidPayload: context =>
736
+ buildTransferPayload(context, {
737
+ recipientPeer: context.transferScenario?.senderPeer ?? selectSenderPeer(context)
738
+ }),
739
+ mutatePayload: (_t, payload) => payload,
740
+ applyInvalidPayload: (context, invalidPayload) =>
741
+ applyTransferValidatorBalanceUpdateFailure(context, invalidPayload),
742
+ assertStateUnchanged: (t, context, _valid, invalid) =>
743
+ assertTransferFailureState(t, context, { payload: invalid }),
744
+ expectedLogs: ['Failed to update validator node balance.']
745
+ }).performScenario();
746
+
747
+ new OperationValidationScenarioBase({
748
+ title: 'State.apply transfer skips payloads when sender balance is insufficient',
749
+ setupScenario: t => setupTransferScenario(t, { senderInitialBalance: ZERO_TRANSFER_AMOUNT }),
750
+ buildValidPayload: context => buildTransferPayload(context),
751
+ mutatePayload: (_t, payload) => payload,
752
+ applyInvalidPayload: async (context, invalidPayload) => {
753
+ const snapshots = await snapshotTransferEntries(context);
754
+ context.transferScenario = {
755
+ ...(context.transferScenario ?? {}),
756
+ insufficientSnapshot: snapshots
757
+ };
758
+ const node = context.transferScenario?.validatorPeer ?? context.peers?.[1];
759
+ await node.base.append(invalidPayload);
760
+ await node.base.update();
761
+ await eventFlush();
762
+ },
763
+ assertStateUnchanged: (t, context, _valid, invalid) =>
764
+ assertTransferFailureState(t, context, {
765
+ payload: invalid,
766
+ ...(context.transferScenario?.insufficientSnapshot ?? {})
767
+ }),
768
+ expectedLogs: ['Insufficient sender balance.', 'Transfer operation skipped.']
769
+ }).performScenario();
770
+
771
+ new OperationAlreadyAppliedScenario({
772
+ title: 'State.apply transfer ignores payloads when operation was already applied',
773
+ setupScenario: setupTransferScenario,
774
+ buildValidPayload: context => buildTransferPayload(context),
775
+ assertStateUnchanged: (t, context, _valid, invalid) =>
776
+ assertTransferReplayIgnoredState(t, context, {
777
+ payload: invalid,
778
+ ...(context.transferScenario?.replaySnapshot ?? {})
779
+ }),
780
+ applyInvalidPayload: (context, invalidPayload, _t, validPayload) =>
781
+ applyTransferAlreadyApplied(context, invalidPayload, validPayload),
782
+ expectedLogs: ['Operation has already been applied.']
783
+ }).performScenario();
784
+
785
+ new InvalidSignatureValidationScenario({
786
+ title: 'State.apply transfer rejects payloads when requester signature is invalid (zero fill)',
787
+ setupScenario: setupTransferScenario,
788
+ buildValidPayload: context => buildTransferPayload(context),
789
+ assertStateUnchanged: (t, context, _valid, invalid) =>
790
+ assertTransferFailureState(t, context, { payload: invalid }),
791
+ strategy: SignatureMutationStrategy.ZERO_FILL,
792
+ applyInvalidPayload: (context, invalidPayload, t) =>
793
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
794
+ expectedLogs: ['Failed to verify message signature.']
795
+ }).performScenario();
796
+
797
+ new InvalidSignatureValidationScenario({
798
+ title: 'State.apply transfer rejects payloads when requester signature is invalid (type mismatch)',
799
+ setupScenario: setupTransferScenario,
800
+ buildValidPayload: context => buildTransferPayload(context),
801
+ assertStateUnchanged: (t, context, _valid, invalid) =>
802
+ assertTransferFailureState(t, context, { payload: invalid }),
803
+ strategy: SignatureMutationStrategy.TYPE_MISMATCH,
804
+ applyInvalidPayload: (context, invalidPayload, t) =>
805
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
806
+ expectedLogs: ['Failed to verify message signature.']
807
+ }).performScenario();
808
+
809
+ new OperationValidationScenarioBase({
810
+ title: 'State.apply transfer rejects payloads when requester signature is invalid (amount tampered)',
811
+ setupScenario: setupTransferScenario,
812
+ buildValidPayload: context => buildTransferPayload(context),
813
+ mutatePayload: mutateTransferAmountWithRehashedTx,
814
+ applyInvalidPayload: (context, invalidPayload, t) =>
815
+ applyInvalidTransferPayloadWithNoMutations(t, context, invalidPayload),
816
+ assertStateUnchanged: (t, context, _valid, invalid) =>
817
+ assertTransferFailureState(t, context, { payload: invalid }),
818
+ expectedLogs: ['Failed to verify message signature.']
819
+ }).performScenario();