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,55 @@
1
+ import { test } from 'brittle';
2
+ import b4a from 'b4a';
3
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
4
+ import nodeEntryUtils from '../../../../../src/core/state/utils/nodeEntry.js';
5
+ import { safeDecodeApplyOperation } from '../../../../../src/utils/protobuf/operationHelpers.js';
6
+ import setupAppendWhitelistScenario, {
7
+ buildAppendWhitelistPayload,
8
+ selectReaderPeer,
9
+ assertReaderWhitelisted
10
+ } from './appendWhitelistScenarioHelpers.js';
11
+
12
+ export default function appendWhitelistHappyPathScenario() {
13
+ test('State.apply appendWhitelist registers reader nodes and assigns licenses - happy path', async t => {
14
+ const context = await setupAppendWhitelistScenario(t);
15
+ const adminNode = context.adminBootstrap;
16
+ const readerPeer = selectReaderPeer(context);
17
+
18
+ const adminNodeEntryBefore = await adminNode.base.view.get(adminNode.wallet.address);
19
+ t.ok(adminNodeEntryBefore, 'admin node entry exists');
20
+ const decodedAdminBefore = nodeEntryUtils.decode(adminNodeEntryBefore.value);
21
+ t.ok(decodedAdminBefore, 'admin node entry decodes');
22
+ const adminBalanceSnapshot = decodedAdminBefore.balance && b4a.from(decodedAdminBefore.balance);
23
+
24
+ const payload = await buildAppendWhitelistPayload(context, readerPeer.wallet.address);
25
+ const decodedPayload = safeDecodeApplyOperation(payload);
26
+ t.ok(decodedPayload, 'payload decodes');
27
+ const whitelistTxHash = decodedPayload?.aco?.tx?.toString('hex');
28
+ t.ok(whitelistTxHash, 'whitelist tx hash extracted');
29
+
30
+ await adminNode.base.append(payload);
31
+ await adminNode.base.update();
32
+ await eventFlush();
33
+
34
+ await assertReaderWhitelisted(t, adminNode.base, readerPeer.wallet.address, {
35
+ expectedLicenseCount: 2
36
+ });
37
+
38
+ const adminNodeEntryAfter = await adminNode.base.view.get(adminNode.wallet.address);
39
+ const decodedAdminAfter = nodeEntryUtils.decode(adminNodeEntryAfter.value);
40
+ t.ok(decodedAdminAfter, 'admin node entry decodes after whitelist append');
41
+ t.ok(
42
+ b4a.equals(decodedAdminAfter.balance, adminBalanceSnapshot),
43
+ 'admin balance remains unchanged while initialization enabled'
44
+ );
45
+
46
+ const txEntry = await adminNode.base.view.get(whitelistTxHash);
47
+ t.ok(txEntry, 'whitelist transaction recorded to prevent replays');
48
+
49
+ await context.sync();
50
+
51
+ await assertReaderWhitelisted(t, readerPeer.base, readerPeer.wallet.address, {
52
+ expectedLicenseCount: 2
53
+ });
54
+ });
55
+ }
@@ -0,0 +1,103 @@
1
+ import b4a from 'b4a';
2
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
3
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
4
+ import nodeEntryUtils, { ZERO_BALANCE } from '../../../../../src/core/state/utils/nodeEntry.js';
5
+ import setupAppendWhitelistScenario, {
6
+ buildAppendWhitelistPayload,
7
+ selectReaderPeer,
8
+ assertAppendWhitelistFailureState
9
+ } from './appendWhitelistScenarioHelpers.js';
10
+ import { buildDisableInitializationPayload } from '../disableInitialization/disableInitializationScenarioHelpers.js';
11
+
12
+ export default function appendWhitelistInsufficientAdminBalanceScenario() {
13
+ new OperationValidationScenarioBase({
14
+ title: 'State.apply appendWhitelist aborts when admin balance cannot cover the fee',
15
+ setupScenario: setupAppendWhitelistScenario,
16
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
17
+ mutatePayload: passThroughPayload,
18
+ applyInvalidPayload: createApplyWithDepletedAdmin(),
19
+ assertStateUnchanged: (t, context) =>
20
+ assertAppendWhitelistFailureState(t, context, {
21
+ readerAddress: selectReaderPeer(context).wallet.address,
22
+ expectedLicenseCount: 1,
23
+ skipSync: true
24
+ }),
25
+ expectedLogs: ['Insufficient admin balance.']
26
+ }).performScenario();
27
+ }
28
+
29
+ function passThroughPayload(_t, payload) {
30
+ return payload;
31
+ }
32
+
33
+ function createApplyWithDepletedAdmin() {
34
+ return async (context, payload) => {
35
+ const adminNode = context.adminBootstrap;
36
+ const adminAddressKey = adminNode.wallet.address;
37
+
38
+ const disablePayload = await buildDisableInitializationPayload(context);
39
+ await adminNode.base.append(disablePayload);
40
+ await adminNode.base.update();
41
+ await eventFlush();
42
+
43
+ const cleanup = patchAdminBalanceToZero(adminNode.base, adminAddressKey);
44
+ try {
45
+ await adminNode.base.append(payload);
46
+ await adminNode.base.update();
47
+ await eventFlush();
48
+ } finally {
49
+ await cleanup();
50
+ }
51
+ };
52
+ }
53
+
54
+ function patchAdminBalanceToZero(base, adminAddressKey) {
55
+ const originalApply = base._handlers.apply;
56
+ let shouldPatchNextApply = true;
57
+
58
+ base._handlers.apply = async (nodes, view, baseCtx) => {
59
+ if (!shouldPatchNextApply) {
60
+ return originalApply(nodes, view, baseCtx);
61
+ }
62
+
63
+ shouldPatchNextApply = false;
64
+ const previousBatch = view.batch;
65
+ const boundBatch = previousBatch.bind(view);
66
+
67
+ view.batch = function patchedBatch(...args) {
68
+ const batch = boundBatch(...args);
69
+ const originalGet = batch.get?.bind(batch);
70
+ if (typeof originalGet === 'function') {
71
+ let mutated = false;
72
+ batch.get = async key => {
73
+ if (mutated || key !== adminAddressKey) {
74
+ return originalGet(key);
75
+ }
76
+
77
+ const adminEntry = await originalGet(key);
78
+ if (!adminEntry?.value) return adminEntry;
79
+
80
+ const depletedEntryBuffer = nodeEntryUtils.setBalance(
81
+ b4a.from(adminEntry.value),
82
+ ZERO_BALANCE
83
+ );
84
+
85
+ if (!depletedEntryBuffer) return adminEntry;
86
+ mutated = true;
87
+ return { ...adminEntry, value: depletedEntryBuffer };
88
+ };
89
+ }
90
+ return batch;
91
+ };
92
+
93
+ try {
94
+ return await originalApply(nodes, view, baseCtx);
95
+ } finally {
96
+ view.batch = previousBatch;
97
+ }
98
+ };
99
+
100
+ return async () => {
101
+ base._handlers.apply = originalApply;
102
+ };
103
+ }
@@ -0,0 +1,60 @@
1
+ import { test } from 'brittle';
2
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
3
+ import setupAppendWhitelistScenario, {
4
+ buildAppendWhitelistPayload,
5
+ selectReaderPeer,
6
+ assertReaderWhitelisted
7
+ } from './appendWhitelistScenarioHelpers.js';
8
+
9
+ export default function appendWhitelistNodeAlreadyWhitelistedScenario() {
10
+ test('State.apply appendWhitelist rejects nodes that are already whitelisted', async t => {
11
+ const context = await setupAppendWhitelistScenario(t);
12
+ const adminNode = context.adminBootstrap;
13
+ const readerPeer = selectReaderPeer(context);
14
+
15
+ const firstPayload = await buildAppendWhitelistPayload(
16
+ context,
17
+ readerPeer.wallet.address
18
+ );
19
+ await adminNode.base.append(firstPayload);
20
+ await adminNode.base.update();
21
+ await eventFlush();
22
+
23
+ await assertReaderWhitelisted(t, adminNode.base, readerPeer.wallet.address, {
24
+ expectedLicenseCount: 2
25
+ });
26
+
27
+ const duplicatePayload = await buildAppendWhitelistPayload(
28
+ context,
29
+ readerPeer.wallet.address
30
+ );
31
+
32
+ const capturedLogs = [];
33
+ const originalConsoleError = console.error;
34
+ console.error = (...args) => {
35
+ capturedLogs.push(args);
36
+ originalConsoleError(...args);
37
+ };
38
+
39
+ try {
40
+ await adminNode.base.append(duplicatePayload);
41
+ await adminNode.base.update();
42
+ await eventFlush();
43
+ } finally {
44
+ console.error = originalConsoleError;
45
+ }
46
+
47
+ await assertReaderWhitelisted(t, adminNode.base, readerPeer.wallet.address, {
48
+ expectedLicenseCount: 2
49
+ });
50
+ await context.sync();
51
+ await assertReaderWhitelisted(t, readerPeer.base, readerPeer.wallet.address, {
52
+ expectedLicenseCount: 2
53
+ });
54
+
55
+ const foundLog = capturedLogs.some(args =>
56
+ args.some(arg => typeof arg === 'string' && arg.includes('Node already whitelisted.'))
57
+ );
58
+ t.ok(foundLog, 'expected apply log "Node already whitelisted." was emitted');
59
+ });
60
+ }
@@ -0,0 +1,191 @@
1
+ import b4a from 'b4a';
2
+ import { setupStateNetwork } from '../../../../helpers/StateNetworkFactory.js';
3
+ import {
4
+ seedBootstrapIndexer,
5
+ defaultOpenHyperbeeView,
6
+ deriveIndexerSequenceState,
7
+ eventFlush
8
+ } from '../../../../helpers/autobaseTestHelpers.js';
9
+ import CompleteStateMessageOperations from '../../../../../src/messages/completeStateMessages/CompleteStateMessageOperations.js';
10
+ import { AUTOBASE_VALUE_ENCODING, EntryType } from '../../../../../src/utils/constants.js';
11
+ import { safeDecodeApplyOperation, safeEncodeApplyOperation } from '../../../../../src/utils/protobuf/operationHelpers.js';
12
+ import nodeEntryUtils, { ZERO_LICENSE } from '../../../../../src/core/state/utils/nodeEntry.js';
13
+ import lengthEntryUtils from '../../../../../src/core/state/utils/lengthEntry.js';
14
+ import addressUtils from '../../../../../src/core/state/utils/address.js';
15
+ import { buildAddAdminRequesterPayload } from '../addAdmin/addAdminScenarioHelpers.js';
16
+
17
+ export async function setupAppendWhitelistScenario(t, { nodes = 2 } = {}) {
18
+ const context = await setupStateNetwork({
19
+ nodes: Math.max(nodes, 2),
20
+ valueEncoding: AUTOBASE_VALUE_ENCODING,
21
+ open: defaultOpenHyperbeeView
22
+ });
23
+
24
+ seedBootstrapIndexer(context);
25
+
26
+ t.teardown(async () => {
27
+ await context.teardown();
28
+ });
29
+
30
+ await bootstrapAdmin(context);
31
+ return context;
32
+ }
33
+
34
+ export function selectReaderPeer(context, offset = 0) {
35
+ const readerPeers = context.peers.slice(1);
36
+ if (readerPeers.length === 0) {
37
+ throw new Error('Append whitelist scenarios require at least one reader node.');
38
+ }
39
+ return readerPeers[Math.min(offset, readerPeers.length - 1)];
40
+ }
41
+
42
+
43
+ export async function buildAppendWhitelistPayload(context, readerAddress = null) {
44
+ const adminNode = context.adminBootstrap;
45
+ const targetAddress = readerAddress ?? selectReaderPeer(context).wallet.address;
46
+ const txValidity = await deriveIndexerSequenceState(adminNode.base);
47
+ return CompleteStateMessageOperations.assembleAppendWhitelistMessages(
48
+ adminNode.wallet,
49
+ txValidity,
50
+ targetAddress
51
+ );
52
+ }
53
+
54
+ export async function buildAppendWhitelistPayloadWithTxValidity(
55
+ context,
56
+ txValidity,
57
+ readerAddress = null
58
+ ) {
59
+ const adminNode = context.adminBootstrap;
60
+ const targetAddress = readerAddress ?? selectReaderPeer(context).wallet.address;
61
+ return CompleteStateMessageOperations.assembleAppendWhitelistMessages(
62
+ adminNode.wallet,
63
+ txValidity,
64
+ targetAddress
65
+ );
66
+ }
67
+
68
+ export async function buildBanWriterPayload(context, readerAddress) {
69
+ const adminNode = context.adminBootstrap;
70
+ const txValidity = await deriveIndexerSequenceState(adminNode.base);
71
+ return CompleteStateMessageOperations.assembleBanWriterMessage(
72
+ adminNode.wallet,
73
+ readerAddress,
74
+ txValidity
75
+ );
76
+ }
77
+
78
+ export function mutateAppendWhitelistPayloadForInvalidSchema(t, validPayload) {
79
+ const decodedPayload = safeDecodeApplyOperation(validPayload);
80
+ t.ok(decodedPayload, 'fixtures decode');
81
+ decodedPayload.aco.tx = b4a.alloc(decodedPayload.aco.tx.length);
82
+ return safeEncodeApplyOperation(decodedPayload);
83
+ }
84
+
85
+ export async function assertAppendWhitelistFailureState(
86
+ t,
87
+ context,
88
+ { skipSync = false, readerAddress = null, expectedLicenseCount = 1 } = {}
89
+ ) {
90
+ const adminNode = context.adminBootstrap;
91
+ const targetAddress = readerAddress ?? selectReaderPeer(context).wallet.address;
92
+
93
+ await assertReaderAbsent(t, adminNode.base, targetAddress);
94
+
95
+ if (typeof expectedLicenseCount === 'number') {
96
+ await assertLicenseCount(t, adminNode.base, expectedLicenseCount);
97
+ }
98
+
99
+ if (!skipSync) {
100
+ await context.sync();
101
+ for (const peer of context.peers) {
102
+ await assertReaderAbsent(t, peer.base, targetAddress);
103
+ }
104
+ }
105
+ }
106
+
107
+ export async function assertAppendWhitelistSuccessState(
108
+ t,
109
+ context,
110
+ { readerAddress = null, expectedLicenseCount = 2, skipSync = false } = {}
111
+ ) {
112
+ const adminNode = context.adminBootstrap;
113
+ const targetAddress = readerAddress ?? selectReaderPeer(context).wallet.address;
114
+
115
+ await assertReaderWhitelisted(t, adminNode.base, targetAddress, {
116
+ expectedLicenseCount
117
+ });
118
+
119
+ if (!skipSync) {
120
+ await context.sync();
121
+ const peer = findPeerByAddress(context, targetAddress);
122
+ if (peer) {
123
+ await assertReaderWhitelisted(t, peer.base, targetAddress, {
124
+ expectedLicenseCount
125
+ });
126
+ }
127
+ }
128
+ }
129
+
130
+ export async function assertReaderWhitelisted(
131
+ t,
132
+ base,
133
+ readerAddress,
134
+ { expectedLicenseCount } = {}
135
+ ) {
136
+ const nodeEntryRecord = await base.view.get(readerAddress);
137
+ t.ok(nodeEntryRecord, 'reader node entry exists');
138
+
139
+ const decodedEntry = nodeEntryUtils.decode(nodeEntryRecord.value);
140
+ t.ok(decodedEntry, 'reader node entry decodes');
141
+ t.is(decodedEntry.isWhitelisted, true, 'reader flagged as whitelisted');
142
+ t.is(decodedEntry.isWriter, false, 'reader not flagged as writer');
143
+ t.is(decodedEntry.isIndexer, false, 'reader not flagged as indexer');
144
+ t.ok(!b4a.equals(decodedEntry.license, ZERO_LICENSE), 'reader license assigned');
145
+
146
+ const licenseId = lengthEntryUtils.decodeBE(decodedEntry.license);
147
+ const licenseIndexEntry = await base.view.get(`${EntryType.LICENSE_INDEX}${licenseId}`);
148
+ t.ok(licenseIndexEntry, 'license index entry exists for reader');
149
+ const readerAddressBuffer = addressUtils.addressToBuffer(readerAddress);
150
+ t.ok(readerAddressBuffer.length > 0, 'reader address encodes to buffer');
151
+ t.ok(
152
+ licenseIndexEntry && b4a.equals(licenseIndexEntry.value, readerAddressBuffer),
153
+ 'license index stores reader address'
154
+ );
155
+
156
+ if (typeof expectedLicenseCount === 'number') {
157
+ await assertLicenseCount(t, base, expectedLicenseCount);
158
+ }
159
+
160
+ return { licenseId, decodedEntry };
161
+ }
162
+
163
+ async function bootstrapAdmin(context) {
164
+ const adminNode = context.adminBootstrap;
165
+ const payload = await buildAddAdminRequesterPayload(context);
166
+
167
+ await adminNode.base.append(payload);
168
+ await adminNode.base.update();
169
+ await eventFlush();
170
+ }
171
+
172
+ async function assertReaderAbsent(t, base, address) {
173
+ const nodeEntryRecord = await base.view.get(address);
174
+ t.is(nodeEntryRecord, null, 'reader node entry remains absent');
175
+ }
176
+
177
+ function findPeerByAddress(context, address) {
178
+ return context.peers.find(peer => peer.wallet?.address === address) ?? null;
179
+ }
180
+
181
+ async function assertLicenseCount(t, base, expected) {
182
+ const licenseCountEntry = await base.view.get(EntryType.LICENSE_COUNT);
183
+ t.ok(licenseCountEntry, 'license count entry exists');
184
+ const licenseCount = lengthEntryUtils.decodeBE(licenseCountEntry.value);
185
+ t.is(licenseCount, expected, 'license count matches expected value');
186
+ }
187
+
188
+ export default setupAppendWhitelistScenario;
189
+ export {
190
+ assertReaderAbsent
191
+ };
@@ -0,0 +1,220 @@
1
+ import InvalidPayloadValidationScenario from '../common/payload-structure/invalidPayloadValidationScenario.js';
2
+ import InvalidAddressValidationScenario from '../common/payload-structure/invalidAddressValidationScenario.js';
3
+ import createAddressWithInvalidPublicKeyScenario from '../common/payload-structure/addressWithInvalidPublicKeyScenario.js';
4
+ import AdminEntryMissingScenario from '../common/access-control/adminEntryMissingScenario.js';
5
+ import AdminEntryDecodeFailureScenario from '../common/access-control/adminEntryDecodeFailureScenario.js';
6
+ import AdminOnlyGuardScenario from '../common/access-control/adminOnlyGuardScenario.js';
7
+ import AdminPublicKeyDecodeFailureScenario from '../common/access-control/adminPublicKeyDecodeFailureScenario.js';
8
+ import AdminConsistencyMismatchScenario from '../common/access-control/adminConsistencyMismatchScenario.js';
9
+ import InvalidHashValidationScenario from '../common/payload-structure/invalidHashValidationScenario.js';
10
+ import InvalidSignatureValidationScenario, { SignatureMutationStrategy } from '../common/payload-structure/invalidSignatureValidationScenario.js';
11
+ import TransactionValidityMismatchScenario from '../common/transactionValidityMismatchScenario.js';
12
+ import IndexerSequenceStateInvalidScenario from '../common/indexer/indexerSequenceStateInvalidScenario.js';
13
+ import OperationAlreadyAppliedScenario from '../common/operationAlreadyAppliedScenario.js';
14
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
15
+ import createAdminEntryUpdateFailureScenario from '../common/balances/adminEntryUpdateFailureScenario.js';
16
+ import createNodeEntryInitializationFailureScenario from '../common/nodeEntryInitializationFailureScenario.js';
17
+ import appendWhitelistHappyPathScenario from './appendWhitelistHappyPathScenario.js';
18
+ import appendWhitelistExistingReaderHappyPathScenario from './appendWhitelistExistingReaderHappyPathScenario.js';
19
+ import appendWhitelistBanAndReapplyScenario from './appendWhitelistBanAndReapplyScenario.js';
20
+ import appendWhitelistFeeAfterDisableScenario from './appendWhitelistFeeAfterDisableScenario.js';
21
+ import appendWhitelistNodeAlreadyWhitelistedScenario from './appendWhitelistNodeAlreadyWhitelistedScenario.js';
22
+ import appendWhitelistInsufficientAdminBalanceScenario from './appendWhitelistInsufficientAdminBalanceScenario.js';
23
+ import { buildDisableInitializationPayload } from '../disableInitialization/disableInitializationScenarioHelpers.js';
24
+ import setupAppendWhitelistScenario, {
25
+ buildAppendWhitelistPayload,
26
+ buildAppendWhitelistPayloadWithTxValidity,
27
+ mutateAppendWhitelistPayloadForInvalidSchema,
28
+ assertAppendWhitelistFailureState,
29
+ assertAppendWhitelistSuccessState
30
+ } from './appendWhitelistScenarioHelpers.js';
31
+
32
+ appendWhitelistHappyPathScenario();
33
+ appendWhitelistExistingReaderHappyPathScenario();
34
+ appendWhitelistBanAndReapplyScenario();
35
+ appendWhitelistFeeAfterDisableScenario();
36
+ appendWhitelistNodeAlreadyWhitelistedScenario();
37
+
38
+ new InvalidPayloadValidationScenario({
39
+ title: 'State.apply appendWhitelist rejects payloads that fail schema validation',
40
+ setupScenario: setupAppendWhitelistScenario,
41
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
42
+ mutatePayload: mutateAppendWhitelistPayloadForInvalidSchema,
43
+ assertStateUnchanged: assertAppendWhitelistFailureState,
44
+ expectedLogs: ['Contract schema validation failed.']
45
+ }).performScenario();
46
+
47
+ new InvalidAddressValidationScenario({
48
+ title: 'State.apply appendWhitelist recipient address is invalid',
49
+ setupScenario: setupAppendWhitelistScenario,
50
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
51
+ assertStateUnchanged: assertAppendWhitelistFailureState,
52
+ addressPath: ['address'],
53
+ expectedLogs: ['Recipient address is invalid.']
54
+ }).performScenario();
55
+
56
+ createAddressWithInvalidPublicKeyScenario({
57
+ title: 'State.apply appendWhitelist recipient public key is invalid',
58
+ setupScenario: setupAppendWhitelistScenario,
59
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
60
+ assertStateUnchanged: assertAppendWhitelistFailureState,
61
+ addressPath: ['address'],
62
+ expectedLogs: ['Failed to decode recipient public key.']
63
+ }).performScenario();
64
+
65
+ new AdminEntryMissingScenario({
66
+ title: 'State.apply appendWhitelist aborts when admin entry cannot be verified',
67
+ setupScenario: setupAppendWhitelistScenario,
68
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
69
+ assertStateUnchanged: (t, context) =>
70
+ assertAppendWhitelistFailureState(t, context, { skipSync: true }),
71
+ expectedLogs: ['Failed to verify admin entry.']
72
+ }).performScenario();
73
+
74
+ new AdminEntryDecodeFailureScenario({
75
+ title: 'State.apply appendWhitelist aborts when admin entry cannot be decoded',
76
+ setupScenario: setupAppendWhitelistScenario,
77
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
78
+ assertStateUnchanged: (t, context) =>
79
+ assertAppendWhitelistFailureState(t, context, { skipSync: true }),
80
+ expectedLogs: ['Failed to decode admin entry.']
81
+ }).performScenario();
82
+
83
+ new AdminOnlyGuardScenario({
84
+ title: 'State.apply appendWhitelist rejects non-admin nodes',
85
+ setupScenario: setupAppendWhitelistScenario,
86
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
87
+ assertStateUnchanged: (t, context) =>
88
+ assertAppendWhitelistFailureState(t, context, { skipSync: true }),
89
+ expectedLogs: ['Node is not allowed to perform this operation. (ADMIN ONLY)']
90
+ }).performScenario();
91
+
92
+ new AdminPublicKeyDecodeFailureScenario({
93
+ title: 'State.apply appendWhitelist aborts when admin public key cannot be decoded',
94
+ setupScenario: setupAppendWhitelistScenario,
95
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
96
+ assertStateUnchanged: (t, context) =>
97
+ assertAppendWhitelistFailureState(t, context, { skipSync: true }),
98
+ expectedLogs: ['Failed to decode admin public key.']
99
+ }).performScenario();
100
+
101
+ new AdminConsistencyMismatchScenario({
102
+ title: 'State.apply appendWhitelist rejects when admin key mismatch occurs',
103
+ setupScenario: setupAppendWhitelistScenario,
104
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
105
+ assertStateUnchanged: (t, context) =>
106
+ assertAppendWhitelistFailureState(t, context, { skipSync: true }),
107
+ expectedLogs: ['System admin and node public keys do not match.']
108
+ }).performScenario();
109
+
110
+ new InvalidAddressValidationScenario({
111
+ title: 'State.apply appendWhitelist node address is invalid',
112
+ setupScenario: setupAppendWhitelistScenario,
113
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
114
+ assertStateUnchanged: assertAppendWhitelistFailureState,
115
+ addressPath: ['aco', 'ia'],
116
+ expectedLogs: ['Failed to verify node address.']
117
+ }).performScenario();
118
+
119
+ createAddressWithInvalidPublicKeyScenario({
120
+ title: 'State.apply appendWhitelist node public key is invalid',
121
+ setupScenario: setupAppendWhitelistScenario,
122
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
123
+ assertStateUnchanged: assertAppendWhitelistFailureState,
124
+ addressPath: ['aco', 'ia'],
125
+ expectedLogs: ['Failed to decode node public key.']
126
+ }).performScenario();
127
+
128
+ new InvalidHashValidationScenario({
129
+ title: 'State.apply appendWhitelist requester message hash mismatch',
130
+ setupScenario: setupAppendWhitelistScenario,
131
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
132
+ assertStateUnchanged: assertAppendWhitelistFailureState,
133
+ expectedLogs: ['Message hash does not match the tx_hash.']
134
+ }).performScenario();
135
+
136
+ new InvalidSignatureValidationScenario({
137
+ title: 'State.apply appendWhitelist requester signature is invalid (foreign signature)',
138
+ setupScenario: setupAppendWhitelistScenario,
139
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
140
+ assertStateUnchanged: assertAppendWhitelistFailureState,
141
+ expectedLogs: ['Failed to verify message signature.']
142
+ }).performScenario();
143
+
144
+ new InvalidSignatureValidationScenario({
145
+ title: 'State.apply appendWhitelist requester signature is invalid (zero fill)',
146
+ setupScenario: setupAppendWhitelistScenario,
147
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
148
+ assertStateUnchanged: assertAppendWhitelistFailureState,
149
+ strategy: SignatureMutationStrategy.ZERO_FILL,
150
+ expectedLogs: ['Failed to verify message signature.']
151
+ }).performScenario();
152
+
153
+ new InvalidSignatureValidationScenario({
154
+ title: 'State.apply appendWhitelist requester signature is invalid (type mismatch)',
155
+ setupScenario: setupAppendWhitelistScenario,
156
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
157
+ assertStateUnchanged: assertAppendWhitelistFailureState,
158
+ strategy: SignatureMutationStrategy.TYPE_MISMATCH,
159
+ expectedLogs: ['Failed to verify message signature.']
160
+ }).performScenario();
161
+
162
+ new IndexerSequenceStateInvalidScenario({
163
+ title: 'State.apply appendWhitelist rejects payloads when indexer sequence state is invalid',
164
+ setupScenario: setupAppendWhitelistScenario,
165
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
166
+ assertStateUnchanged: (t, context, validPayload, invalidPayload) =>
167
+ assertAppendWhitelistFailureState(t, context, {
168
+ skipSync: true,
169
+ readerAddress: invalidPayload?.address
170
+ }),
171
+ expectedLogs: ['Indexer sequence state is invalid.']
172
+ }).performScenario();
173
+
174
+ new TransactionValidityMismatchScenario({
175
+ title: 'State.apply appendWhitelist rejects payload when tx validity mismatches indexer state',
176
+ setupScenario: setupAppendWhitelistScenario,
177
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
178
+ assertStateUnchanged: assertAppendWhitelistFailureState,
179
+ txValidityPath: ['aco', 'txv'],
180
+ rebuildPayloadWithTxValidity: ({ context, mutatedTxValidity }) =>
181
+ buildAppendWhitelistPayloadWithTxValidity(context, mutatedTxValidity),
182
+ expectedLogs: ['Transaction was not executed.']
183
+ }).performScenario();
184
+
185
+ new OperationAlreadyAppliedScenario({
186
+ title: 'State.apply appendWhitelist rejects duplicate operations',
187
+ setupScenario: setupAppendWhitelistScenario,
188
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
189
+ assertStateUnchanged: (t, context) =>
190
+ assertAppendWhitelistSuccessState(t, context),
191
+ expectedLogs: ['Operation has already been applied.']
192
+ }).performScenario();
193
+
194
+ appendWhitelistNodeAlreadyWhitelistedScenario();
195
+
196
+ appendWhitelistInsufficientAdminBalanceScenario();
197
+
198
+ createAdminEntryUpdateFailureScenario({
199
+ title: 'State.apply appendWhitelist rejects when admin entry update fails',
200
+ setupScenario: setupAppendWhitelistScenario,
201
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
202
+ beforeApply: async context => {
203
+ const adminNode = context.adminBootstrap;
204
+ const disablePayload = await buildDisableInitializationPayload(context);
205
+ await adminNode.base.append(disablePayload);
206
+ await adminNode.base.update();
207
+ await eventFlush();
208
+ },
209
+ assertStateUnchanged: (t, context) =>
210
+ assertAppendWhitelistFailureState(t, context, { skipSync: true })
211
+ }).performScenario();
212
+
213
+ createNodeEntryInitializationFailureScenario({
214
+ title: 'State.apply appendWhitelist rejects when node entry initialization fails',
215
+ setupScenario: setupAppendWhitelistScenario,
216
+ buildValidPayload: context => buildAppendWhitelistPayload(context),
217
+ assertStateUnchanged: (t, context) =>
218
+ assertAppendWhitelistFailureState(t, context, { expectedLicenseCount: 2, skipSync: true }),
219
+ selectNode: context => context.adminBootstrap
220
+ }).performScenario();