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,38 @@
1
+ import { test } from 'brittle';
2
+ import {
3
+ setupTransferScenario,
4
+ buildTransferPayload,
5
+ assertTransferSuccessState,
6
+ snapshotTransferEntries,
7
+ DEFAULT_TRANSFER_AMOUNT
8
+ } from './transferScenarioHelpers.js';
9
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
10
+
11
+ export default function transferSelfTransferAmountScenario() {
12
+ test('State.apply transfer charges fee on self-transfer (amount > 0)', async t => {
13
+ /* Sender transfers to self, amount is ignored, only fee is deducted and validator gets 75% fee. */
14
+ const context = await setupTransferScenario(t, { recipientHasEntry: true });
15
+ const { senderPeer, validatorPeer } = context.transferScenario;
16
+
17
+ const snapshots = await snapshotTransferEntries(context, {
18
+ senderPeer,
19
+ recipientPeer: senderPeer,
20
+ validatorPeer
21
+ });
22
+ const payload = await buildTransferPayload(context, {
23
+ recipientPeer: senderPeer,
24
+ amount: DEFAULT_TRANSFER_AMOUNT
25
+ });
26
+
27
+ await validatorPeer.base.append(payload);
28
+ await validatorPeer.base.update();
29
+ await eventFlush();
30
+
31
+ await assertTransferSuccessState(t, context, {
32
+ payload,
33
+ senderEntryBefore: snapshots.senderEntry,
34
+ recipientEntryBefore: snapshots.recipientEntry,
35
+ validatorEntryBefore: snapshots.validatorEntry
36
+ });
37
+ });
38
+ }
@@ -0,0 +1,38 @@
1
+ import { test } from 'brittle';
2
+ import {
3
+ setupTransferScenario,
4
+ buildTransferPayload,
5
+ assertTransferSuccessState,
6
+ snapshotTransferEntries,
7
+ ZERO_TRANSFER_AMOUNT
8
+ } from './transferScenarioHelpers.js';
9
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
10
+
11
+ export default function transferSelfTransferZeroAmountScenario() {
12
+ test('State.apply transfer charges fee on self-transfer (amount = 0)', async t => {
13
+ /* Sender transfers zero to self, fee is deducted, validator earns 75% fee. */
14
+ const context = await setupTransferScenario(t, { recipientHasEntry: true });
15
+ const { senderPeer, validatorPeer } = context.transferScenario;
16
+
17
+ const snapshots = await snapshotTransferEntries(context, {
18
+ senderPeer,
19
+ recipientPeer: senderPeer,
20
+ validatorPeer
21
+ });
22
+ const payload = await buildTransferPayload(context, {
23
+ recipientPeer: senderPeer,
24
+ amount: ZERO_TRANSFER_AMOUNT
25
+ });
26
+
27
+ await validatorPeer.base.append(payload);
28
+ await validatorPeer.base.update();
29
+ await eventFlush();
30
+
31
+ await assertTransferSuccessState(t, context, {
32
+ payload,
33
+ senderEntryBefore: snapshots.senderEntry,
34
+ recipientEntryBefore: snapshots.recipientEntry,
35
+ validatorEntryBefore: snapshots.validatorEntry
36
+ });
37
+ });
38
+ }
@@ -0,0 +1,38 @@
1
+ import { test } from 'brittle';
2
+ import {
3
+ setupTransferScenario,
4
+ buildTransferPayload,
5
+ assertTransferSuccessState,
6
+ snapshotTransferEntries,
7
+ DEFAULT_TRANSFER_AMOUNT
8
+ } from './transferScenarioHelpers.js';
9
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
10
+
11
+ export default function transferValidatorRecipientAmountScenario() {
12
+ test('State.apply transfer credits validator recipient with amount and fee (amount > 0)', async t => {
13
+ /* Recipient is the validator, sender pays amount+fee, validator receives amount plus 75% fee. */
14
+ const context = await setupTransferScenario(t, { recipientHasEntry: true });
15
+ const { senderPeer, validatorPeer } = context.transferScenario;
16
+
17
+ const snapshots = await snapshotTransferEntries(context, {
18
+ senderPeer,
19
+ recipientPeer: validatorPeer,
20
+ validatorPeer
21
+ });
22
+ const payload = await buildTransferPayload(context, {
23
+ recipientPeer: validatorPeer,
24
+ amount: DEFAULT_TRANSFER_AMOUNT
25
+ });
26
+
27
+ await validatorPeer.base.append(payload);
28
+ await validatorPeer.base.update();
29
+ await eventFlush();
30
+
31
+ await assertTransferSuccessState(t, context, {
32
+ payload,
33
+ senderEntryBefore: snapshots.senderEntry,
34
+ recipientEntryBefore: snapshots.recipientEntry,
35
+ validatorEntryBefore: snapshots.validatorEntry
36
+ });
37
+ });
38
+ }
@@ -0,0 +1,38 @@
1
+ import { test } from 'brittle';
2
+ import {
3
+ setupTransferScenario,
4
+ buildTransferPayload,
5
+ assertTransferSuccessState,
6
+ snapshotTransferEntries,
7
+ ZERO_TRANSFER_AMOUNT
8
+ } from './transferScenarioHelpers.js';
9
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
10
+
11
+ export default function transferValidatorRecipientZeroAmountScenario() {
12
+ test('State.apply transfer rewards validator recipient with fee only (amount = 0)', async t => {
13
+ /* Recipient is validator, sender pays fee, validator/recipient receives 75% fee. */
14
+ const context = await setupTransferScenario(t, { recipientHasEntry: true });
15
+ const { senderPeer, validatorPeer } = context.transferScenario;
16
+
17
+ const snapshots = await snapshotTransferEntries(context, {
18
+ senderPeer,
19
+ recipientPeer: validatorPeer,
20
+ validatorPeer
21
+ });
22
+ const payload = await buildTransferPayload(context, {
23
+ recipientPeer: validatorPeer,
24
+ amount: ZERO_TRANSFER_AMOUNT
25
+ });
26
+
27
+ await validatorPeer.base.append(payload);
28
+ await validatorPeer.base.update();
29
+ await eventFlush();
30
+
31
+ await assertTransferSuccessState(t, context, {
32
+ payload,
33
+ senderEntryBefore: snapshots.senderEntry,
34
+ recipientEntryBefore: snapshots.recipientEntry,
35
+ validatorEntryBefore: snapshots.validatorEntry
36
+ });
37
+ });
38
+ }
@@ -0,0 +1,318 @@
1
+ import b4a from 'b4a';
2
+ import txOperationStandardHappyPathScenario from './txOperationStandardHappyPathScenario.js';
3
+ import txOperationValidatorCreatorHappyPathScenario from './txOperationValidatorCreatorHappyPathScenario.js';
4
+ import txOperationRequesterCreatorHappyPathScenario from './txOperationRequesterCreatorHappyPathScenario.js';
5
+ import txOperationDifferentValidatorCreatorHappyPathScenario from './txOperationDifferentValidatorCreatorHappyPathScenario.js';
6
+ import InvalidPayloadValidationScenario from '../common/payload-structure/invalidPayloadValidationScenario.js';
7
+ import InvalidHashValidationScenario from '../common/payload-structure/invalidHashValidationScenario.js';
8
+ import PartialOperationValidationScenario, {
9
+ PartialOperationMutationStrategy
10
+ } from '../common/payload-structure/partialOperationValidationScenario.js';
11
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
12
+ import InvalidSignatureValidationScenario, {
13
+ SignatureMutationStrategy
14
+ } from '../common/payload-structure/invalidSignatureValidationScenario.js';
15
+ import RequesterAddressValidationScenario from '../common/requesterAddressValidationScenario.js';
16
+ import createRequesterPublicKeyValidationScenario from '../common/requesterPublicKeyValidationScenario.js';
17
+ import InvalidAddressValidationScenario from '../common/payload-structure/invalidAddressValidationScenario.js';
18
+ import createAddressWithInvalidPublicKeyScenario from '../common/payload-structure/addressWithInvalidPublicKeyScenario.js';
19
+ import IndexerSequenceStateInvalidScenario from '../common/indexer/indexerSequenceStateInvalidScenario.js';
20
+ import TransactionValidityMismatchScenario from '../common/transactionValidityMismatchScenario.js';
21
+ import OperationAlreadyAppliedScenario from '../common/operationAlreadyAppliedScenario.js';
22
+ import ValidatorConsistencyScenarioBase, {
23
+ ValidatorEntryMutation
24
+ } from '../common/validatorConsistency/base/validatorConsistencyScenarioBase.js';
25
+ import ValidatorEntryDecodeFailureScenario from '../common/validatorConsistency/validatorEntryDecodeFailureScenario.js';
26
+ import ValidatorInactiveScenario from '../common/validatorConsistency/validatorInactiveScenario.js';
27
+ import ValidatorWriterKeyMismatchScenario from '../common/validatorConsistency/validatorWriterKeyMismatchScenario.js';
28
+ import { safeDecodeApplyOperation, safeEncodeApplyOperation } from '../../../../../src/utils/protobuf/operationHelpers.js';
29
+ import {
30
+ setupTxOperationScenario,
31
+ buildTxOperationPayload,
32
+ buildTxOperationPayloadWithTxValidity,
33
+ assertTxOperationFailureState,
34
+ assertTxOperationSuccessState,
35
+ mutateBootstrapEqualMbs,
36
+ mutateMbsMismatch,
37
+ appendInvalidTxPayload,
38
+ mutateValidatorSignature
39
+ } from './txOperationScenarioHelpers.js';
40
+ import txOperationBootstrapNotRegisteredScenario from './txOperationBootstrapNotRegisteredScenario.js';
41
+ import txOperationInvalidDeploymentEntryScenario from './txOperationInvalidDeploymentEntryScenario.js';
42
+ import txOperationInvalidSubnetCreatorAddressScenario from './txOperationInvalidSubnetCreatorAddressScenario.js';
43
+ import txOperationInvalidFeeAmountScenario from './txOperationInvalidFeeAmountScenario.js';
44
+ import txOperationTransferFeeInvalidRequesterEntryScenario from './txOperationTransferFeeInvalidRequesterEntryScenario.js';
45
+ import txOperationTransferFeeDecodeRequesterEntryScenario from './txOperationTransferFeeDecodeRequesterEntryScenario.js';
46
+ import txOperationTransferFeeInvalidRequesterBalanceScenario from './txOperationTransferFeeInvalidRequesterBalanceScenario.js';
47
+ import txOperationTransferFeeInsufficientRequesterBalanceScenario from './txOperationTransferFeeInsufficientRequesterBalanceScenario.js';
48
+ import txOperationTransferFeeSubtractFailureScenario from './txOperationTransferFeeSubtractFailureScenario.js';
49
+ import txOperationTransferFeeUpdateFailureScenario from './txOperationTransferFeeUpdateFailureScenario.js';
50
+ import txOperationTransferFeeDecodeValidatorEntryScenario from './txOperationTransferFeeDecodeValidatorEntryScenario.js';
51
+ import txOperationTransferFeeInvalidValidatorBalanceScenario from './txOperationTransferFeeInvalidValidatorBalanceScenario.js';
52
+ import txOperationTransferFeeAddValidatorBalanceFailureScenario from './txOperationTransferFeeAddValidatorBalanceFailureScenario.js';
53
+ import txOperationTransferFeeUpdateValidatorBalanceFailureScenario from './txOperationTransferFeeUpdateValidatorBalanceFailureScenario.js';
54
+ import txOperationTransferFeeAddValidatorBonusFailureScenario from './txOperationTransferFeeAddValidatorBonusFailureScenario.js';
55
+ import txOperationTransferFeeUpdateValidatorBonusFailureScenario from './txOperationTransferFeeUpdateValidatorBonusFailureScenario.js';
56
+ import txOperationTransferFeeMissingCreatorEntryScenario from './txOperationTransferFeeMissingCreatorEntryScenario.js';
57
+ import txOperationTransferFeeDecodeCreatorEntryScenario from './txOperationTransferFeeDecodeCreatorEntryScenario.js';
58
+ import txOperationTransferFeeInvalidCreatorBalanceScenario from './txOperationTransferFeeInvalidCreatorBalanceScenario.js';
59
+ import txOperationTransferFeeAddCreatorBalanceFailureScenario from './txOperationTransferFeeAddCreatorBalanceFailureScenario.js';
60
+ import txOperationTransferFeeUpdateCreatorBalanceFailureScenario from './txOperationTransferFeeUpdateCreatorBalanceFailureScenario.js';
61
+ import {
62
+ txOperationTransferFeeResultNullScenario,
63
+ txOperationTransferFeeResultIgnoredScenario,
64
+ txOperationTransferFeeRequesterEntryMissingScenario,
65
+ txOperationTransferFeeValidatorEntryMissingScenario
66
+ } from './txOperationTransferFeeGuardBypassScenario.js';
67
+
68
+ txOperationStandardHappyPathScenario();
69
+ txOperationValidatorCreatorHappyPathScenario();
70
+ txOperationRequesterCreatorHappyPathScenario();
71
+ txOperationDifferentValidatorCreatorHappyPathScenario();
72
+
73
+ new InvalidPayloadValidationScenario({
74
+ title: 'State.apply txOperation rejects payloads that fail contract validation',
75
+ setupScenario: setupTxOperationScenario,
76
+ buildValidPayload: buildTxOperationPayload,
77
+ mutatePayload: (t, validPayload) => {
78
+ const decoded = safeDecodeApplyOperation(validPayload);
79
+ t.ok(decoded, 'valid payload decodes before mutation');
80
+ if (!decoded) return validPayload;
81
+ return safeEncodeApplyOperation({ ...decoded, address: b4a.alloc(1) });
82
+ },
83
+ applyInvalidPayload: appendInvalidTxPayload,
84
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
85
+ assertTxOperationFailureState(t, context, { payload: invalidPayload }),
86
+ expectedLogs: ['Contract schema validation failed.']
87
+ }).performScenario();
88
+
89
+ // validator cannot sign own tx
90
+ new PartialOperationValidationScenario({
91
+ title: 'State.apply txOperation rejects payloads when validator signs own transaction',
92
+ setupScenario: setupTxOperationScenario,
93
+ buildValidPayload: buildTxOperationPayload,
94
+ assertStateUnchanged: (t, context, _valid, invalid) =>
95
+ assertTxOperationFailureState(t, context, { payload: invalid }),
96
+ strategy: PartialOperationMutationStrategy.ADDRESS_MATCH,
97
+ parentKey: 'txo',
98
+ expectedLogs: ['Validator cannot sign its own transaction.']
99
+ }).performScenario();
100
+
101
+ // nonces and signatures equality checks
102
+ new PartialOperationValidationScenario({
103
+ title: 'State.apply txOperation rejects payloads when nonces match',
104
+ setupScenario: setupTxOperationScenario,
105
+ buildValidPayload: buildTxOperationPayload,
106
+ assertStateUnchanged: (t, context, _valid, invalid) =>
107
+ assertTxOperationFailureState(t, context, { payload: invalid }),
108
+ strategy: PartialOperationMutationStrategy.NONCE_MATCH,
109
+ parentKey: 'txo',
110
+ expectedLogs: ['Nonces should not be the same.']
111
+ }).performScenario();
112
+
113
+ new PartialOperationValidationScenario({
114
+ title: 'State.apply txOperation rejects payloads when signatures match',
115
+ setupScenario: setupTxOperationScenario,
116
+ buildValidPayload: buildTxOperationPayload,
117
+ assertStateUnchanged: (t, context, _valid, invalid) =>
118
+ assertTxOperationFailureState(t, context, { payload: invalid }),
119
+ strategy: PartialOperationMutationStrategy.SIGNATURE_MATCH,
120
+ parentKey: 'txo',
121
+ expectedLogs: ['Signatures should not be the same.']
122
+ }).performScenario();
123
+
124
+ // bootstrap consistency
125
+ new OperationValidationScenarioBase({
126
+ title: 'State.apply txOperation rejects payloads when external bootstrap matches MSB bootstrap',
127
+ setupScenario: setupTxOperationScenario,
128
+ buildValidPayload: buildTxOperationPayload,
129
+ mutatePayload: (t, validPayload) => mutateBootstrapEqualMbs(t, validPayload),
130
+ applyInvalidPayload: appendInvalidTxPayload,
131
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
132
+ assertTxOperationFailureState(t, context, { payload: invalidPayload }),
133
+ expectedLogs: ['Network and external bootstrap cannot be the same.']
134
+ }).performScenario();
135
+
136
+ new OperationValidationScenarioBase({
137
+ title: 'State.apply txOperation rejects payloads when declared MSB bootstrap mismatches network',
138
+ setupScenario: setupTxOperationScenario,
139
+ buildValidPayload: buildTxOperationPayload,
140
+ mutatePayload: (t, validPayload) => mutateMbsMismatch(t, validPayload),
141
+ applyInvalidPayload: appendInvalidTxPayload,
142
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
143
+ assertTxOperationFailureState(t, context, { payload: invalidPayload }),
144
+ expectedLogs: ['Declared MSB bootstrap is different than real MSB bootstrap.']
145
+ }).performScenario();
146
+
147
+ // requester identity
148
+ new RequesterAddressValidationScenario({
149
+ title: 'State.apply txOperation rejects invalid requester address',
150
+ setupScenario: setupTxOperationScenario,
151
+ buildValidPayload: buildTxOperationPayload,
152
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
153
+ assertTxOperationFailureState(t, context, { payload: invalidPayload }),
154
+ expectedLogs: ['Invalid requester address.']
155
+ }).performScenario();
156
+
157
+ createRequesterPublicKeyValidationScenario({
158
+ title: 'State.apply txOperation rejects requester public key decode failures',
159
+ setupScenario: setupTxOperationScenario,
160
+ buildValidPayload: buildTxOperationPayload,
161
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
162
+ assertTxOperationFailureState(t, context, { payload: invalidPayload }),
163
+ expectedLogs: ['Failed to decode requester public key.']
164
+ }).performScenario();
165
+
166
+ new InvalidHashValidationScenario({
167
+ title: 'State.apply txOperation rejects payloads when requester hash mismatches tx_hash',
168
+ setupScenario: setupTxOperationScenario,
169
+ buildValidPayload: buildTxOperationPayload,
170
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
171
+ assertTxOperationFailureState(t, context, { payload: invalidPayload }),
172
+ applyInvalidPayload: appendInvalidTxPayload,
173
+ expectedLogs: ['Message hash does not match the tx_hash.']
174
+ }).performScenario();
175
+
176
+ new InvalidSignatureValidationScenario({
177
+ title: 'State.apply txOperation rejects payloads when requester signature verification fails',
178
+ setupScenario: setupTxOperationScenario,
179
+ buildValidPayload: buildTxOperationPayload,
180
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
181
+ assertTxOperationFailureState(t, context, { payload: invalidPayload }),
182
+ applyInvalidPayload: appendInvalidTxPayload,
183
+ expectedLogs: ['Failed to verify message signature.']
184
+ }).performScenario();
185
+
186
+ // validator identity
187
+ new InvalidAddressValidationScenario({
188
+ title: 'State.apply txOperation rejects invalid validator address',
189
+ setupScenario: setupTxOperationScenario,
190
+ buildValidPayload: buildTxOperationPayload,
191
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
192
+ assertTxOperationFailureState(t, context, { payload: invalidPayload }),
193
+ addressPath: ['txo', 'va'],
194
+ expectedLogs: ['Invalid validator address.']
195
+ }).performScenario();
196
+
197
+ createAddressWithInvalidPublicKeyScenario({
198
+ title: 'State.apply txOperation rejects validator public key decode failures',
199
+ setupScenario: setupTxOperationScenario,
200
+ buildValidPayload: buildTxOperationPayload,
201
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
202
+ assertTxOperationFailureState(t, context, { payload: invalidPayload }),
203
+ addressPath: ['txo', 'va'],
204
+ expectedLogs: ['Failed to decode validator public key.']
205
+ }).performScenario();
206
+
207
+ new OperationValidationScenarioBase({
208
+ title: 'State.apply txOperation rejects validator message signature verification failures',
209
+ setupScenario: setupTxOperationScenario,
210
+ buildValidPayload: buildTxOperationPayload,
211
+ mutatePayload: (t, validPayload) => mutateValidatorSignature(t, validPayload),
212
+ applyInvalidPayload: appendInvalidTxPayload,
213
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
214
+ assertTxOperationFailureState(t, context, { payload: invalidPayload }),
215
+ expectedLogs: ['Failed to verify validator message signature.']
216
+ }).performScenario();
217
+
218
+ // indexer sequence + tx validity
219
+ new IndexerSequenceStateInvalidScenario({
220
+ title: 'State.apply txOperation rejects payloads when indexer sequence state is invalid',
221
+ setupScenario: setupTxOperationScenario,
222
+ buildValidPayload: buildTxOperationPayload,
223
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
224
+ assertTxOperationFailureState(t, context, { payload: invalidPayload }),
225
+ expectedLogs: ['Indexer sequence state is invalid.']
226
+ }).performScenario();
227
+
228
+ new TransactionValidityMismatchScenario({
229
+ title: 'State.apply txOperation rejects payloads when tx validity mismatches indexer state',
230
+ setupScenario: setupTxOperationScenario,
231
+ buildValidPayload: buildTxOperationPayload,
232
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
233
+ assertTxOperationFailureState(t, context, { payload: invalidPayload }),
234
+ txValidityPath: ['txo', 'txv'],
235
+ rebuildPayloadWithTxValidity: ({ context, mutatedTxValidity }) =>
236
+ buildTxOperationPayloadWithTxValidity(context, mutatedTxValidity),
237
+ expectedLogs: ['Transaction was not executed.']
238
+ }).performScenario();
239
+
240
+ new ValidatorConsistencyScenarioBase({
241
+ title: 'State.apply txOperation rejects payloads when validator entry is missing',
242
+ setupScenario: setupTxOperationScenario,
243
+ buildValidPayload: buildTxOperationPayload,
244
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
245
+ assertTxOperationFailureState(t, context, { payload: invalidPayload }),
246
+ mutateEntry: () => ValidatorEntryMutation.DELETE,
247
+ validatorAddressPath: ['txo', 'va'],
248
+ expectedLogs: ['Incoming validator entry is null.']
249
+ }).performScenario();
250
+
251
+ new ValidatorEntryDecodeFailureScenario({
252
+ title: 'State.apply txOperation rejects payloads when validator entry cannot be decoded',
253
+ setupScenario: setupTxOperationScenario,
254
+ buildValidPayload: buildTxOperationPayload,
255
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
256
+ assertTxOperationFailureState(t, context, { payload: invalidPayload, skipSync: true }),
257
+ validatorAddressPath: ['txo', 'va'],
258
+ expectedLogs: ['Failed to decode validator entry.']
259
+ }).performScenario();
260
+
261
+ new ValidatorInactiveScenario({
262
+ title: 'State.apply txOperation rejects payloads when validator is not an active writer',
263
+ setupScenario: setupTxOperationScenario,
264
+ buildValidPayload: buildTxOperationPayload,
265
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
266
+ assertTxOperationFailureState(t, context, { payload: invalidPayload, skipSync: true }),
267
+ validatorAddressPath: ['txo', 'va'],
268
+ expectedLogs: ['Operation validator is not active']
269
+ }).performScenario();
270
+
271
+ new ValidatorWriterKeyMismatchScenario({
272
+ title: 'State.apply txOperation rejects payloads when validator writer key mismatches requester',
273
+ setupScenario: setupTxOperationScenario,
274
+ buildValidPayload: buildTxOperationPayload,
275
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
276
+ assertTxOperationFailureState(t, context, { payload: invalidPayload, skipSync: true }),
277
+ validatorAddressPath: ['txo', 'va'],
278
+ expectedLogs: ['Validator cannot be the same as requester.']
279
+ }).performScenario();
280
+
281
+ new OperationAlreadyAppliedScenario({
282
+ title: 'State.apply txOperation rejects duplicate operations',
283
+ setupScenario: setupTxOperationScenario,
284
+ buildValidPayload: buildTxOperationPayload,
285
+ selectNode: context => context.txOperation?.validatorPeer ?? context.peers?.[1],
286
+ assertStateUnchanged: (t, context, validPayload) =>
287
+ assertTxOperationSuccessState(t, context, { payload: validPayload, skipSync: true }),
288
+ expectedLogs: ['Operation has already been applied.']
289
+ }).performScenario();
290
+
291
+ txOperationBootstrapNotRegisteredScenario();
292
+ txOperationInvalidDeploymentEntryScenario();
293
+ txOperationInvalidSubnetCreatorAddressScenario();
294
+ txOperationInvalidFeeAmountScenario();
295
+ // transferFeeTxOperation
296
+ txOperationTransferFeeInvalidRequesterEntryScenario();
297
+ txOperationTransferFeeDecodeRequesterEntryScenario();
298
+ txOperationTransferFeeInvalidRequesterBalanceScenario();
299
+ txOperationTransferFeeInsufficientRequesterBalanceScenario();
300
+ txOperationTransferFeeSubtractFailureScenario();
301
+ txOperationTransferFeeUpdateFailureScenario();
302
+ txOperationTransferFeeDecodeValidatorEntryScenario();
303
+ txOperationTransferFeeInvalidValidatorBalanceScenario();
304
+ txOperationTransferFeeAddValidatorBalanceFailureScenario();
305
+ txOperationTransferFeeUpdateValidatorBalanceFailureScenario();
306
+ txOperationTransferFeeAddValidatorBonusFailureScenario();
307
+ txOperationTransferFeeUpdateValidatorBonusFailureScenario();
308
+ txOperationTransferFeeMissingCreatorEntryScenario();
309
+ txOperationTransferFeeDecodeCreatorEntryScenario();
310
+ txOperationTransferFeeInvalidCreatorBalanceScenario();
311
+ txOperationTransferFeeAddCreatorBalanceFailureScenario();
312
+ txOperationTransferFeeUpdateCreatorBalanceFailureScenario();
313
+ txOperationTransferFeeResultNullScenario().performScenario();
314
+ txOperationTransferFeeResultIgnoredScenario().performScenario();
315
+ txOperationTransferFeeRequesterEntryMissingScenario().performScenario();
316
+ txOperationTransferFeeValidatorEntryMissingScenario().performScenario();
317
+ // Post-transferFee guards for null requester/validator entries are unreachable with current transferFeeTxOperation
318
+ // (it returns null/IGNORE on earlier failures), so no dedicated scenarios exist yet.
@@ -0,0 +1,70 @@
1
+ import b4a from 'b4a';
2
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
3
+ import { safeDecodeApplyOperation } from '../../../../../src/utils/protobuf/operationHelpers.js';
4
+ import { EntryType } from '../../../../../src/utils/constants.js';
5
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
6
+ import {
7
+ setupTxOperationScenario,
8
+ buildTxOperationPayload,
9
+ assertTxOperationFailureState
10
+ } from './txOperationScenarioHelpers.js';
11
+
12
+ export default function txOperationBootstrapNotRegisteredScenario() {
13
+ new OperationValidationScenarioBase({
14
+ title: 'State.apply txOperation rejects payloads when bootstrap is not registered',
15
+ setupScenario: setupTxOperationScenario,
16
+ buildValidPayload: buildTxOperationPayload,
17
+ mutatePayload: async (t, _validPayload, context) => {
18
+ const unregisteredBootstrap = b4a.alloc(32, 0x24);
19
+ t.ok(context.txOperation?.msbBootstrap, 'msb bootstrap available');
20
+ t.ok(!b4a.equals(unregisteredBootstrap, context.txOperation?.msbBootstrap), 'unregistered bootstrap differs from MSB');
21
+ return buildTxOperationPayload(context, { externalBootstrap: unregisteredBootstrap });
22
+ },
23
+ applyInvalidPayload: async (context, invalidPayload, t, validPayload) => {
24
+ const node = context.txOperation?.validatorPeer ?? context.peers?.[1];
25
+ const payload = validPayload ?? invalidPayload;
26
+ const decoded = safeDecodeApplyOperation(payload);
27
+ t.ok(decoded?.txo?.bs, 'payload exposes bootstrap key');
28
+ const bootstrapKey = decoded?.txo?.bs?.toString('hex');
29
+ if (!bootstrapKey) return;
30
+
31
+ const deploymentKey = `${EntryType.DEPLOYMENT}${bootstrapKey}`;
32
+ const originalApply = node.base._handlers.apply;
33
+
34
+ node.base._handlers.apply = async function patchedApply(nodes, view, baseCtx) {
35
+ const originalBatch = view.batch;
36
+ view.batch = function patchedBatch(...args) {
37
+ const batch = originalBatch.apply(this, args);
38
+ if (!batch || typeof batch.get !== 'function') return batch;
39
+ const originalGet = batch.get.bind(batch);
40
+ batch.get = async key => {
41
+ if (key === deploymentKey) return null;
42
+ if (b4a.isBuffer(key) && b4a.equals(key, decoded.txo.bs)) return null;
43
+ return originalGet(key);
44
+ };
45
+ return batch;
46
+ };
47
+
48
+ try {
49
+ return await originalApply.call(this, nodes, view, baseCtx);
50
+ } finally {
51
+ view.batch = originalBatch;
52
+ }
53
+ };
54
+
55
+ try {
56
+ await node.base.append(payload);
57
+ await node.base.update();
58
+ await eventFlush();
59
+ } finally {
60
+ node.base._handlers.apply = originalApply;
61
+ }
62
+ },
63
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
64
+ assertTxOperationFailureState(t, context, {
65
+ payload: invalidPayload,
66
+ validatorEntryBefore: null
67
+ }),
68
+ expectedLogs: ['Bootstrap has not been registered.']
69
+ }).performScenario();
70
+ }
@@ -0,0 +1,23 @@
1
+ import { test } from 'brittle';
2
+ import {
3
+ setupTxOperationScenario,
4
+ buildTxOperationPayload,
5
+ assertTxOperationSuccessState
6
+ } from './txOperationScenarioHelpers.js';
7
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
8
+
9
+ // Validator signs tx; bootstrap creator is another peer -> validator gets 50%, creator 25%, 25% burned.
10
+
11
+ export default function txOperationDifferentValidatorCreatorHappyPathScenario() {
12
+ test('State.apply txOperation rewards different validator and creator (50/25 split)', async t => {
13
+ const context = await setupTxOperationScenario(t, { creatorPeerKind: 'deployer' });
14
+ const payload = await buildTxOperationPayload(context);
15
+ const validatorPeer = context.txOperation?.validatorPeer ?? context.adminBootstrap;
16
+
17
+ await validatorPeer.base.append(payload);
18
+ await validatorPeer.base.update();
19
+ await eventFlush();
20
+
21
+ await assertTxOperationSuccessState(t, context, { payload, distribution: 'standard' });
22
+ });
23
+ }
@@ -0,0 +1,48 @@
1
+ import b4a from 'b4a';
2
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
3
+ import { safeDecodeApplyOperation } from '../../../../../src/utils/protobuf/operationHelpers.js';
4
+ import { EntryType } from '../../../../../src/utils/constants.js';
5
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
6
+ import deploymentEntryUtils from '../../../../../src/core/state/utils/deploymentEntry.js';
7
+ import {
8
+ setupTxOperationScenario,
9
+ buildTxOperationPayload,
10
+ assertTxOperationFailureState
11
+ } from './txOperationScenarioHelpers.js';
12
+
13
+ export default function txOperationInvalidDeploymentEntryScenario() {
14
+ new OperationValidationScenarioBase({
15
+ title: 'State.apply txOperation rejects payloads when deployment entry cannot be decoded',
16
+ setupScenario: setupTxOperationScenario,
17
+ buildValidPayload: buildTxOperationPayload,
18
+ mutatePayload: async (_t, validPayload) => validPayload,
19
+ applyInvalidPayload: async (context, invalidPayload, t, validPayload) => {
20
+ const node = context.txOperation?.validatorPeer ?? context.peers?.[1];
21
+ const payload = validPayload ?? invalidPayload;
22
+ const decoded = safeDecodeApplyOperation(payload);
23
+ t.ok(decoded?.txo?.bs, 'payload exposes bootstrap key');
24
+ const bootstrapKey = decoded?.txo?.bs?.toString('hex');
25
+ if (!bootstrapKey) return;
26
+
27
+ const deploymentKey = `${EntryType.DEPLOYMENT}${bootstrapKey}`;
28
+ const originalApply = node.base._handlers.apply;
29
+ const originalDecode = deploymentEntryUtils.decode;
30
+ deploymentEntryUtils.decode = () => null;
31
+
32
+ try {
33
+ await node.base.append(payload);
34
+ await node.base.update();
35
+ await eventFlush();
36
+ } finally {
37
+ node.base._handlers.apply = originalApply;
38
+ deploymentEntryUtils.decode = originalDecode;
39
+ }
40
+ },
41
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
42
+ assertTxOperationFailureState(t, context, {
43
+ payload: invalidPayload,
44
+ validatorEntryBefore: null
45
+ }),
46
+ expectedLogs: ['Invalid deployment entry.']
47
+ }).performScenario();
48
+ }
@@ -0,0 +1,39 @@
1
+ import b4a from 'b4a';
2
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
3
+ import transactionUtils from '../../../../../src/core/state/utils/transaction.js';
4
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
5
+ import {
6
+ setupTxOperationScenario,
7
+ buildTxOperationPayload,
8
+ assertTxOperationFailureState
9
+ } from './txOperationScenarioHelpers.js';
10
+
11
+ export default function txOperationInvalidFeeAmountScenario() {
12
+ new OperationValidationScenarioBase({
13
+ title: 'State.apply txOperation rejects payloads when fee amount is invalid',
14
+ setupScenario: setupTxOperationScenario,
15
+ buildValidPayload: buildTxOperationPayload,
16
+ mutatePayload: async (_t, validPayload) => validPayload,
17
+ applyInvalidPayload: async (context, invalidPayload) => {
18
+ const node = context.txOperation?.validatorPeer ?? context.peers?.[1];
19
+ const originalFee = transactionUtils.FEE;
20
+ transactionUtils.FEE = b4a.alloc(1, 0x00);
21
+
22
+ try {
23
+ await node.base.append(invalidPayload);
24
+ await node.base.update();
25
+ await eventFlush();
26
+ } finally {
27
+ transactionUtils.FEE = originalFee;
28
+ }
29
+ },
30
+ assertStateUnchanged: (t, context, _valid, invalidPayload) =>
31
+ assertTxOperationFailureState(t, context, {
32
+ payload: invalidPayload,
33
+ validatorEntryBefore: null,
34
+ deployerEntryBefore: null,
35
+ requesterEntryBefore: null
36
+ }),
37
+ expectedLogs: ['Invalid fee amount.']
38
+ }).performScenario();
39
+ }