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,197 @@
1
+ import b4a from 'b4a';
2
+ import OperationValidationScenarioBase from '../../base/OperationValidationScenarioBase.js';
3
+ import nodeEntryUtils from '../../../../../../../src/core/state/utils/nodeEntry.js';
4
+ import addressUtils from '../../../../../../../src/core/state/utils/address.js';
5
+ import { BALANCE_ZERO } from '../../../../../../../src/core/state/utils/balance.js';
6
+ import { eventFlush } from '../../../../../../helpers/autobaseTestHelpers.js';
7
+
8
+ export default class RequesterBalanceScenarioBase extends OperationValidationScenarioBase {
9
+ constructor({
10
+ title,
11
+ setupScenario,
12
+ buildValidPayload,
13
+ assertStateUnchanged,
14
+ mutateDecodedEntry = decoded => decoded,
15
+ selectPeer,
16
+ selectNode = defaultSelectNode,
17
+ expectedLogs,
18
+ mutatePayload,
19
+ applyInvalidPayload,
20
+ failNextBalanceSub = false,
21
+ failNextBalanceUpdate = false
22
+ }) {
23
+ if (typeof mutateDecodedEntry !== 'function') {
24
+ throw new Error('Requester balance scenario requires a mutateDecodedEntry function.');
25
+ }
26
+
27
+ if (typeof selectPeer !== 'function') {
28
+ throw new Error('Requester balance scenario requires a selectPeer function.');
29
+ }
30
+
31
+ super({
32
+ title,
33
+ setupScenario,
34
+ buildValidPayload,
35
+ mutatePayload: mutatePayload ?? passThroughPayload,
36
+ applyInvalidPayload:
37
+ typeof applyInvalidPayload === 'function'
38
+ ? applyInvalidPayload
39
+ : createApplyInvalidPayload({
40
+ selectNode,
41
+ selectPeer,
42
+ mutateDecodedEntry,
43
+ failNextBalanceSub,
44
+ failNextBalanceUpdate
45
+ }),
46
+ assertStateUnchanged,
47
+ expectedLogs
48
+ });
49
+ }
50
+ }
51
+
52
+ function passThroughPayload(_t, payload) {
53
+ return payload;
54
+ }
55
+
56
+ function defaultSelectNode(context) {
57
+ return context.adminBootstrap ?? context.bootstrap ?? context.peers?.[0] ?? null;
58
+ }
59
+
60
+ function createApplyInvalidPayload({
61
+ selectNode,
62
+ selectPeer,
63
+ mutateDecodedEntry,
64
+ failNextBalanceSub,
65
+ failNextBalanceUpdate
66
+ }) {
67
+ return async (context, payload) => {
68
+ const node = selectNode(context);
69
+ if (!node?.base) {
70
+ throw new Error('Requester balance scenario requires a writable node.');
71
+ }
72
+
73
+ const peer = selectPeer(context);
74
+ if (!peer?.wallet?.address) {
75
+ throw new Error('Requester balance scenario requires a peer with an address.');
76
+ }
77
+
78
+ const targetAddressString = peer.wallet.address;
79
+ const targetAddressBuffer = addressUtils.addressToBuffer(targetAddressString);
80
+
81
+ const originalDecode = nodeEntryUtils.decode;
82
+ let shouldFailNextSub = false;
83
+ let shouldFailNextUpdate = false;
84
+ let shouldMutateNextDecode = false;
85
+
86
+ nodeEntryUtils.decode = function patchedDecode(buffer) {
87
+ const decoded = originalDecode(buffer);
88
+ if (shouldMutateNextDecode) {
89
+ shouldMutateNextDecode = false;
90
+ shouldFailNextSub = failNextBalanceSub;
91
+ shouldFailNextUpdate = failNextBalanceUpdate;
92
+ const mutated = mutateDecodedEntry(decoded, { context, peer });
93
+ if (mutated && typeof mutated === 'object') {
94
+ return mutated;
95
+ }
96
+ }
97
+ return decoded;
98
+ };
99
+
100
+ let originalSub = null;
101
+ if (failNextBalanceSub) {
102
+ const balancePrototype = Object.getPrototypeOf(BALANCE_ZERO);
103
+ originalSub = balancePrototype.sub;
104
+ balancePrototype.sub = function patchedSub(...args) {
105
+ if (shouldFailNextSub) {
106
+ shouldFailNextSub = false;
107
+ return null;
108
+ }
109
+ return originalSub.call(this, ...args);
110
+ };
111
+ }
112
+
113
+ let originalUpdate = null;
114
+ let balancePrototype = null;
115
+ if (failNextBalanceUpdate) {
116
+ balancePrototype = Object.getPrototypeOf(BALANCE_ZERO);
117
+ originalUpdate = balancePrototype.update;
118
+ balancePrototype.update = function patchedUpdate(...args) {
119
+ if (shouldFailNextUpdate) {
120
+ shouldFailNextUpdate = false;
121
+ return null;
122
+ }
123
+ return originalUpdate.call(this, ...args);
124
+ };
125
+ }
126
+
127
+ const cleanup = patchNodeEntryAccess({
128
+ base: node.base,
129
+ targetAddressString,
130
+ targetAddressBuffer,
131
+ onTargetHit: () => {
132
+ shouldMutateNextDecode = true;
133
+ }
134
+ });
135
+
136
+ try {
137
+ await node.base.append(payload);
138
+ await node.base.update();
139
+ await eventFlush();
140
+ } finally {
141
+ nodeEntryUtils.decode = originalDecode;
142
+ if (failNextBalanceSub && originalSub) {
143
+ const balancePrototype = Object.getPrototypeOf(BALANCE_ZERO);
144
+ balancePrototype.sub = originalSub;
145
+ }
146
+ if (failNextBalanceUpdate && balancePrototype && originalUpdate) {
147
+ balancePrototype.update = originalUpdate;
148
+ }
149
+ cleanup();
150
+ }
151
+ };
152
+ }
153
+
154
+ function patchNodeEntryAccess({ base, targetAddressString, targetAddressBuffer, onTargetHit }) {
155
+ const originalApply = base._handlers.apply;
156
+
157
+ base._handlers.apply = async function patchedApply(nodes, view, baseCtx) {
158
+ const originalBatch = view.batch;
159
+ view.batch = function patchedBatch(...args) {
160
+ const batch = originalBatch.apply(this, args);
161
+ if (!batch?.get) {
162
+ return batch;
163
+ }
164
+
165
+ const originalGet = batch.get.bind(batch);
166
+ batch.get = async key => {
167
+ const entry = await originalGet(key);
168
+ if (isTargetKey(key, targetAddressString, targetAddressBuffer)) {
169
+ onTargetHit?.();
170
+ }
171
+ return entry;
172
+ };
173
+
174
+ return batch;
175
+ };
176
+
177
+ try {
178
+ return await originalApply.call(this, nodes, view, baseCtx);
179
+ } finally {
180
+ view.batch = originalBatch;
181
+ }
182
+ };
183
+
184
+ return () => {
185
+ base._handlers.apply = originalApply;
186
+ };
187
+ }
188
+
189
+ function isTargetKey(key, targetAddressString, targetAddressBuffer) {
190
+ if (typeof key === 'string') {
191
+ return key === targetAddressString;
192
+ }
193
+ if (b4a.isBuffer(key) && targetAddressBuffer) {
194
+ return b4a.equals(key, targetAddressBuffer);
195
+ }
196
+ return false;
197
+ }
@@ -0,0 +1,52 @@
1
+ import b4a from 'b4a';
2
+ import OperationValidationScenarioBase from '../base/OperationValidationScenarioBase.js';
3
+ import transactionUtils from '../../../../../../src/core/state/utils/transaction.js';
4
+ import { eventFlush } from '../../../../../helpers/autobaseTestHelpers.js';
5
+
6
+ const DEFAULT_LOG = 'Invalid fee amount.';
7
+
8
+ export default class FeeDecodeFailureScenario extends OperationValidationScenarioBase {
9
+ constructor({
10
+ title,
11
+ setupScenario,
12
+ buildValidPayload,
13
+ assertStateUnchanged,
14
+ expectedLogs = [DEFAULT_LOG],
15
+ selectNode = defaultSelectNode,
16
+ mutatePayload
17
+ }) {
18
+ super({
19
+ title,
20
+ setupScenario,
21
+ buildValidPayload,
22
+ mutatePayload: mutatePayload ?? passThroughPayload,
23
+ applyInvalidPayload: async (context, payload) => {
24
+ const node = selectNode(context);
25
+ if (!node?.base) {
26
+ throw new Error('Fee decode failure scenario requires a writable node.');
27
+ }
28
+
29
+ const originalFee = transactionUtils.FEE;
30
+ transactionUtils.FEE = b4a.alloc(1); // invalid buffer to break toBalance()
31
+
32
+ try {
33
+ await node.base.append(payload);
34
+ await node.base.update();
35
+ await eventFlush();
36
+ } finally {
37
+ transactionUtils.FEE = originalFee;
38
+ }
39
+ },
40
+ assertStateUnchanged,
41
+ expectedLogs
42
+ });
43
+ }
44
+ }
45
+
46
+ function passThroughPayload(_t, payload) {
47
+ return payload;
48
+ }
49
+
50
+ function defaultSelectNode(context) {
51
+ return context.adminBootstrap ?? context.bootstrap ?? context.peers?.[0] ?? null;
52
+ }
@@ -0,0 +1,15 @@
1
+ import b4a from 'b4a';
2
+ import RequesterBalanceScenarioBase from './base/requesterBalanceScenarioBase.js';
3
+
4
+ export default class RequesterBalanceDecodeFailureScenario extends RequesterBalanceScenarioBase {
5
+ constructor(options) {
6
+ super({
7
+ ...options,
8
+ mutateDecodedEntry: decoded => ({
9
+ ...decoded,
10
+ balance: b4a.alloc(1)
11
+ }),
12
+ expectedLogs: ['Invalid requester balance.']
13
+ });
14
+ }
15
+ }
@@ -0,0 +1,11 @@
1
+ import RequesterBalanceScenarioBase from './base/requesterBalanceScenarioBase.js';
2
+
3
+ export default class RequesterBalanceFeeApplicationFailureScenario extends RequesterBalanceScenarioBase {
4
+ constructor(options) {
5
+ super({
6
+ ...options,
7
+ failNextBalanceSub: true,
8
+ expectedLogs: options?.expectedLogs ?? ['Failed to apply fee to requester balance.']
9
+ });
10
+ }
11
+ }
@@ -0,0 +1,15 @@
1
+ import b4a from 'b4a';
2
+ import RequesterBalanceScenarioBase from './base/requesterBalanceScenarioBase.js';
3
+
4
+ export default class RequesterBalanceInsufficientScenario extends RequesterBalanceScenarioBase {
5
+ constructor(options) {
6
+ super({
7
+ ...options,
8
+ mutateDecodedEntry: decoded => ({
9
+ ...decoded,
10
+ balance: b4a.alloc(decoded.balance.length)
11
+ }),
12
+ expectedLogs: ['Insufficient requester balance.']
13
+ });
14
+ }
15
+ }
@@ -0,0 +1,11 @@
1
+ import RequesterBalanceScenarioBase from './base/requesterBalanceScenarioBase.js';
2
+
3
+ export default class RequesterBalanceUpdateFailureScenario extends RequesterBalanceScenarioBase {
4
+ constructor(options) {
5
+ super({
6
+ ...options,
7
+ failNextBalanceUpdate: true,
8
+ expectedLogs: options?.expectedLogs ?? ['Failed to update node balance.']
9
+ });
10
+ }
11
+ }
@@ -0,0 +1,11 @@
1
+ import ValidatorEntryValidationScenarioBase from '../validatorEntryValidation/base/validatorEntryValidationScenarioBase.js';
2
+
3
+ export default class ValidatorEntryRewardFailureScenario extends ValidatorEntryValidationScenarioBase {
4
+ constructor(options) {
5
+ super({
6
+ ...options,
7
+ failNextBalanceAdd: true,
8
+ expectedLogs: options?.expectedLogs ?? ['Failed to transfer fee to validator.']
9
+ });
10
+ }
11
+ }
@@ -0,0 +1,11 @@
1
+ import ValidatorEntryValidationScenarioBase from '../validatorEntryValidation/base/validatorEntryValidationScenarioBase.js';
2
+
3
+ export default class ValidatorEntryUpdateFailureScenario extends ValidatorEntryValidationScenarioBase {
4
+ constructor(options) {
5
+ super({
6
+ ...options,
7
+ failNextBalanceUpdate: true,
8
+ expectedLogs: options?.expectedLogs ?? ['Failed to update validator entry.']
9
+ });
10
+ }
11
+ }
@@ -0,0 +1,40 @@
1
+ import nodeEntryUtils from '../../../../../../src/core/state/utils/nodeEntry.js';
2
+ import RequesterBalanceScenarioBase from './base/requesterBalanceScenarioBase.js';
3
+
4
+ export default class ValidatorNodeEntryDecodeFailureScenario extends RequesterBalanceScenarioBase {
5
+ constructor({
6
+ title = 'State.apply operation rejects payloads when validator node entry cannot be decoded',
7
+ expectedLogs = ['Failed to decode validator entry.'],
8
+ ...options
9
+ }) {
10
+ super({
11
+ ...options,
12
+ title,
13
+ expectedLogs,
14
+ selectPeer: options.selectPeer ?? (() => options.validatorPeer ?? null),
15
+ mutateDecodedEntry: decoded => decoded,
16
+ applyInvalidPayload: createApplyInvalidPayload(options)
17
+ });
18
+ }
19
+ }
20
+
21
+ function createApplyInvalidPayload({ selectNode, selectPeer }) {
22
+ return async (context, payload) => {
23
+ const node = selectNode?.(context) ?? context.bootstrap ?? context.adminBootstrap ?? context.peers?.[0];
24
+ const peer = selectPeer?.(context) ?? context.bootstrapDeployment?.validatorPeer ?? context.peers?.[1];
25
+ if (!node?.base || !peer?.wallet?.address) {
26
+ throw new Error('Validator node entry decode failure scenario requires node and validator peer.');
27
+ }
28
+
29
+ const originalDecode = nodeEntryUtils.decode;
30
+
31
+ nodeEntryUtils.decode = () => null;
32
+
33
+ try {
34
+ await node.base.append(payload);
35
+ await node.base.update();
36
+ } finally {
37
+ nodeEntryUtils.decode = originalDecode;
38
+ }
39
+ };
40
+ }
@@ -0,0 +1,114 @@
1
+ import { test } from 'brittle';
2
+
3
+ export default class OperationValidationScenarioBase {
4
+ constructor({
5
+ title,
6
+ setupScenario,
7
+ buildValidPayload,
8
+ mutatePayload,
9
+ applyInvalidPayload,
10
+ assertStateUnchanged,
11
+ expectedLogs = []
12
+ }) {
13
+ Object.entries({
14
+ title,
15
+ setupScenario,
16
+ buildValidPayload,
17
+ mutatePayload,
18
+ applyInvalidPayload,
19
+ assertStateUnchanged
20
+ }).forEach(([key, value]) => {
21
+ if (!value) {
22
+ throw new Error(`Operation validation scenario missing required ${key}`);
23
+ }
24
+ });
25
+
26
+ if (typeof title !== 'string') {
27
+ throw new Error('Operation validation scenario requires a title string.');
28
+ }
29
+
30
+ this.title = title;
31
+ this.setupScenario = setupScenario;
32
+ this.buildValidPayload = buildValidPayload;
33
+ this.mutatePayload = mutatePayload;
34
+ this.applyInvalidPayload = applyInvalidPayload;
35
+ this.assertStateUnchanged = assertStateUnchanged;
36
+ this.expectedLogs = this.#normalizeExpectedLogs(expectedLogs);
37
+ }
38
+
39
+ async execute(t) {
40
+ const shouldCaptureLogs = this.expectedLogs.length > 0;
41
+ const capturedLogs = [];
42
+ const originalConsoleError = console.error;
43
+ if (shouldCaptureLogs) {
44
+ console.error = (...args) => {
45
+ capturedLogs.push(args);
46
+ originalConsoleError(...args);
47
+ };
48
+ }
49
+
50
+ const context = await this.setupScenario(t);
51
+ const validPayload = await this.buildValidPayload(context, t);
52
+ t.ok(validPayload?.length > 0, 'valid payload encodes');
53
+
54
+ const invalidPayload = await this.mutatePayload(t, validPayload, context);
55
+ t.ok(invalidPayload.length > 0, 'invalid payload still encodes');
56
+
57
+ try {
58
+ await this.applyInvalidPayload(context, invalidPayload, t, validPayload);
59
+ await this.assertStateUnchanged(t, context, validPayload, invalidPayload);
60
+ } finally {
61
+ if (shouldCaptureLogs) {
62
+ console.error = originalConsoleError;
63
+ this.#assertExpectedLogs(t, capturedLogs);
64
+ }
65
+ }
66
+ }
67
+
68
+ performScenario() {
69
+ test(this.title, t => this.execute(t));
70
+ }
71
+
72
+ #assertExpectedLogs(t, capturedLogs) {
73
+ for (const matcher of this.expectedLogs) {
74
+ const found = matcher.values.some(expected =>
75
+ capturedLogs.some(args =>
76
+ args.some(arg => this.#toComparableString(arg).includes(expected))
77
+ )
78
+ );
79
+ t.ok(found, `expected apply log ${matcher.label} was emitted`);
80
+ }
81
+ }
82
+
83
+ #normalizeExpectedLogs(expectedLogs) {
84
+ if (!Array.isArray(expectedLogs) || expectedLogs.length === 0) return [];
85
+ return expectedLogs.map(entry => {
86
+ if (typeof entry === 'string') {
87
+ return { values: [entry], label: `"${entry}"` };
88
+ }
89
+ if (Array.isArray(entry) && entry.length > 0) {
90
+ return {
91
+ values: entry.map(value => (typeof value === 'string' ? value : String(value))),
92
+ label: entry.map(value => `"${value}"`).join(' or ')
93
+ };
94
+ }
95
+ if (entry && Array.isArray(entry.anyOf) && entry.anyOf.length > 0) {
96
+ return {
97
+ values: entry.anyOf.map(value => (typeof value === 'string' ? value : String(value))),
98
+ label: entry.anyOf.map(value => `"${value}"`).join(' or ')
99
+ };
100
+ }
101
+ throw new Error('expectedLogs entries must be strings, arrays of strings, or { anyOf: string[] }');
102
+ });
103
+ }
104
+
105
+ #toComparableString(arg) {
106
+ if (typeof arg === 'string') return arg;
107
+ if (arg instanceof Error) return arg.message || arg.toString();
108
+ try {
109
+ return JSON.stringify(arg);
110
+ } catch {
111
+ return String(arg);
112
+ }
113
+ }
114
+ }
@@ -0,0 +1,103 @@
1
+ import { setupStateNetwork } from '../../../../helpers/StateNetworkFactory.js';
2
+ import {
3
+ seedBootstrapIndexer,
4
+ defaultOpenHyperbeeView,
5
+ deriveIndexerSequenceState,
6
+ eventFlush
7
+ } from '../../../../helpers/autobaseTestHelpers.js';
8
+ import CompleteStateMessageOperations from '../../../../../src/messages/completeStateMessages/CompleteStateMessageOperations.js';
9
+ import { AUTOBASE_VALUE_ENCODING } from '../../../../../src/utils/constants.js';
10
+ import { buildAddAdminRequesterPayload } from '../addAdmin/addAdminScenarioHelpers.js';
11
+
12
+ /**
13
+ * Boots a network with an initialized admin node and returns the shared context.
14
+ */
15
+ export async function setupAdminNetwork(t, { nodes = 2 } = {}) {
16
+ const context = await setupStateNetwork({
17
+ nodes: Math.max(nodes, 2),
18
+ valueEncoding: AUTOBASE_VALUE_ENCODING,
19
+ open: defaultOpenHyperbeeView
20
+ });
21
+
22
+ seedBootstrapIndexer(context);
23
+
24
+ t.teardown(async () => {
25
+ await context.teardown();
26
+ });
27
+
28
+ await bootstrapAdmin(context);
29
+ return context;
30
+ }
31
+
32
+ /**
33
+ * Ensures the first reader peer receives funds and becomes whitelisted.
34
+ */
35
+ export async function setupAdminAndWhitelistedReaderNetwork(
36
+ t,
37
+ { nodes = 2, readerInitialBalance = null } = {}
38
+ ) {
39
+ const context = await setupAdminNetwork(t, { nodes });
40
+ const readerPeers = context.peers.slice(1);
41
+ const firstReader = readerPeers[0];
42
+ if (!firstReader) return context;
43
+
44
+ if (readerInitialBalance) {
45
+ await initializeBalances(context, [[firstReader.wallet.address, readerInitialBalance]]);
46
+ }
47
+
48
+ await whitelistAddress(context, firstReader.wallet.address);
49
+ return context;
50
+ }
51
+
52
+ /**
53
+ * Appends balance initialization payloads for provided pairs.
54
+ * @param {Array<[string, Buffer]>} recipients - address/balance tuples.
55
+ */
56
+ export async function initializeBalances(context, recipients) {
57
+ const adminNode = context.adminBootstrap;
58
+ if (!adminNode || !Array.isArray(recipients) || recipients.length === 0) return;
59
+
60
+ const txValidity = await deriveIndexerSequenceState(adminNode.base);
61
+ const payloads = await CompleteStateMessageOperations.assembleBalanceInitializationMessages(
62
+ adminNode.wallet,
63
+ txValidity,
64
+ recipients
65
+ );
66
+
67
+ for (const payload of payloads) {
68
+ await adminNode.base.append(payload);
69
+ await adminNode.base.update();
70
+ await eventFlush();
71
+ }
72
+ }
73
+
74
+ export async function whitelistAddress(context, address) {
75
+ const adminNode = context.adminBootstrap;
76
+ if (!adminNode || !address) return;
77
+
78
+ const txValidity = await deriveIndexerSequenceState(adminNode.base);
79
+ const payload = await CompleteStateMessageOperations.assembleAppendWhitelistMessages(
80
+ adminNode.wallet,
81
+ txValidity,
82
+ address
83
+ );
84
+
85
+ await adminNode.base.append(payload);
86
+ await adminNode.base.update();
87
+ await eventFlush();
88
+ }
89
+
90
+ async function bootstrapAdmin(context) {
91
+ const adminNode = context.adminBootstrap;
92
+ const payload = await buildAddAdminRequesterPayload(context);
93
+ await adminNode.base.append(payload);
94
+ await adminNode.base.update();
95
+ await eventFlush();
96
+ }
97
+
98
+ export default {
99
+ setupAdminNetwork,
100
+ setupAdminAndWhitelistedReaderNetwork,
101
+ initializeBalances,
102
+ whitelistAddress
103
+ };
@@ -0,0 +1,36 @@
1
+ import OperationValidationScenarioBase from '../base/OperationValidationScenarioBase.js';
2
+ import { applyWithRequesterEntryCorruption } from '../../addWriter/addWriterScenarioHelpers.js';
3
+
4
+ const passThroughPayload = (_t, payload) => payload;
5
+
6
+ export default class IndexerNodeEntryDecodeFailureScenario extends OperationValidationScenarioBase {
7
+ constructor({
8
+ title,
9
+ setupScenario,
10
+ buildValidPayload,
11
+ assertStateUnchanged,
12
+ expectedLogs,
13
+ selectPeer,
14
+ mutatePayload = passThroughPayload
15
+ }) {
16
+ const peerSelector = typeof selectPeer === 'function' ? selectPeer : null;
17
+
18
+ super({
19
+ title,
20
+ setupScenario,
21
+ buildValidPayload,
22
+ mutatePayload,
23
+ applyInvalidPayload: (context, invalidPayload, t, validPayload) => {
24
+ const peer =
25
+ peerSelector?.(context, {
26
+ invalidPayload,
27
+ validPayload,
28
+ t
29
+ }) ?? null;
30
+ return applyWithRequesterEntryCorruption(context, invalidPayload, { peer });
31
+ },
32
+ assertStateUnchanged,
33
+ expectedLogs
34
+ });
35
+ }
36
+ }
@@ -0,0 +1,36 @@
1
+ import OperationValidationScenarioBase from '../base/OperationValidationScenarioBase.js';
2
+ import { applyWithRequesterEntryRemoval } from '../../addWriter/addWriterScenarioHelpers.js';
3
+
4
+ const passThroughPayload = (_t, payload) => payload;
5
+
6
+ export default class IndexerNodeEntryMissingScenario extends OperationValidationScenarioBase {
7
+ constructor({
8
+ title,
9
+ setupScenario,
10
+ buildValidPayload,
11
+ assertStateUnchanged,
12
+ expectedLogs,
13
+ selectPeer,
14
+ mutatePayload = passThroughPayload
15
+ }) {
16
+ const peerSelector = typeof selectPeer === 'function' ? selectPeer : null;
17
+
18
+ super({
19
+ title,
20
+ setupScenario,
21
+ buildValidPayload,
22
+ mutatePayload,
23
+ applyInvalidPayload: (context, invalidPayload, t, validPayload) => {
24
+ const peer =
25
+ peerSelector?.(context, {
26
+ invalidPayload,
27
+ validPayload,
28
+ t
29
+ }) ?? null;
30
+ return applyWithRequesterEntryRemoval(context, invalidPayload, { peer });
31
+ },
32
+ assertStateUnchanged,
33
+ expectedLogs
34
+ });
35
+ }
36
+ }
@@ -0,0 +1,29 @@
1
+ import OperationValidationScenarioBase from '../base/OperationValidationScenarioBase.js';
2
+
3
+ const passThroughPayload = (_t, payload) => payload;
4
+
5
+ export default class IndexerRoleUpdateFailureScenario extends OperationValidationScenarioBase {
6
+ constructor({
7
+ title,
8
+ setupScenario,
9
+ buildValidPayload,
10
+ assertStateUnchanged,
11
+ expectedLogs,
12
+ applyRoleMutationFailure,
13
+ mutatePayload = passThroughPayload
14
+ }) {
15
+ if (typeof applyRoleMutationFailure !== 'function') {
16
+ throw new Error('Indexer role update failure scenario requires an applyRoleMutationFailure function.');
17
+ }
18
+
19
+ super({
20
+ title,
21
+ setupScenario,
22
+ buildValidPayload,
23
+ mutatePayload,
24
+ applyInvalidPayload: applyRoleMutationFailure,
25
+ assertStateUnchanged,
26
+ expectedLogs
27
+ });
28
+ }
29
+ }