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,344 @@
1
+ import b4a from 'b4a';
2
+ import {
3
+ setupAddWriterScenario,
4
+ selectWriterPeer,
5
+ buildRemoveWriterPayload,
6
+ assertWriterRemovalState,
7
+ assertValidatorReward,
8
+ promotePeerToWriter
9
+ } from '../addWriter/addWriterScenarioHelpers.js';
10
+ import PartialStateMessageOperations from '../../../../../src/messages/partialStateMessages/PartialStateMessageOperations.js';
11
+ import CompleteStateMessageOperations from '../../../../../src/messages/completeStateMessages/CompleteStateMessageOperations.js';
12
+ import nodeEntryUtils from '../../../../../src/core/state/utils/nodeEntry.js';
13
+ import addressUtils from '../../../../../src/core/state/utils/address.js';
14
+ import { toBalance, BALANCE_FEE, BALANCE_TO_STAKE } from '../../../../../src/core/state/utils/balance.js';
15
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
16
+ import { safeDecodeApplyOperation } from '../../../../../src/utils/protobuf/operationHelpers.js';
17
+ import { EntryType } from '../../../../../src/utils/constants.js';
18
+
19
+ export async function setupRemoveWriterScenario(t, options = {}) {
20
+ const context = await setupAddWriterScenario(t, options);
21
+ const writerPeer = selectWriterPeer(context);
22
+ await promotePeerToWriter(t, context, { readerPeer: writerPeer });
23
+ return context;
24
+ }
25
+
26
+ export async function assertRemoveWriterSuccessState(
27
+ t,
28
+ context,
29
+ {
30
+ writerPeer = selectWriterPeer(context),
31
+ validatorPeer = context.adminBootstrap,
32
+ writerEntryBefore,
33
+ validatorEntryBefore,
34
+ payload,
35
+ skipSync = false
36
+ } = {}
37
+ ) {
38
+ if (!writerEntryBefore?.value) {
39
+ throw new Error('assertRemoveWriterSuccessState requires the writer entry before removal.');
40
+ }
41
+ if (!validatorEntryBefore?.value) {
42
+ throw new Error('assertRemoveWriterSuccessState requires the validator entry before removal.');
43
+ }
44
+ if (!payload) {
45
+ throw new Error('assertRemoveWriterSuccessState requires the removeWriter payload.');
46
+ }
47
+
48
+ const decodedWriterBefore = nodeEntryUtils.decode(writerEntryBefore.value);
49
+ if (!decodedWriterBefore) {
50
+ throw new Error('Failed to decode writer entry before removal.');
51
+ }
52
+
53
+ const writerBalanceBefore = toBalance(decodedWriterBefore.balance);
54
+ t.ok(writerBalanceBefore, 'writer balance before removeWriter decodes');
55
+ if (!writerBalanceBefore) return;
56
+
57
+ const writerStakedBefore = toBalance(decodedWriterBefore.stakedBalance);
58
+ t.ok(writerStakedBefore, 'writer staked balance before removeWriter decodes');
59
+ if (!writerStakedBefore) return;
60
+
61
+ const balanceAfterFee = writerBalanceBefore.sub(BALANCE_FEE);
62
+ t.ok(balanceAfterFee, 'writer balance after deducting removeWriter fee computed');
63
+ if (!balanceAfterFee) return;
64
+
65
+ const expectedFinalBalance = balanceAfterFee.add(writerStakedBefore);
66
+ t.ok(expectedFinalBalance, 'writer balance after unstaking computed');
67
+ if (!expectedFinalBalance) return;
68
+
69
+ await assertWriterRemovalState(t, context, {
70
+ readerPeer: writerPeer,
71
+ validatorPeer,
72
+ writerKeyBuffer: decodedWriterBefore.wk,
73
+ expectedBalanceBuffer: expectedFinalBalance.value,
74
+ expectedLicenseBuffer: decodedWriterBefore.license,
75
+ payload,
76
+ skipSync
77
+ });
78
+
79
+ const decodedValidatorBefore = nodeEntryUtils.decode(validatorEntryBefore.value);
80
+ if (!decodedValidatorBefore) {
81
+ throw new Error('Failed to decode validator entry before removeWriter.');
82
+ }
83
+
84
+ await assertValidatorReward(t, validatorPeer, b4a.from(decodedValidatorBefore.balance));
85
+
86
+ await assertPayloadProcessedByValidator(t, payload, validatorPeer.wallet.address);
87
+ }
88
+
89
+ export async function assertRemoveWriterFailureState(t, context, { skipSync = false } = {}) {
90
+ const writerPeer = selectWriterPeer(context);
91
+ const validatorPeer = context.adminBootstrap;
92
+ const expected = deriveActiveWriterState(context);
93
+
94
+ await assertWriterStillPromoted(t, validatorPeer.base, writerPeer, expected);
95
+
96
+ if (!skipSync) {
97
+ await context.sync();
98
+ await assertWriterStillPromoted(t, writerPeer.base, writerPeer, expected);
99
+ }
100
+ }
101
+
102
+ function deriveActiveWriterState(context) {
103
+ const writerInitialBalance = context.addWriterScenario?.writerInitialBalance;
104
+ if (!writerInitialBalance) {
105
+ throw new Error('removeWriter scenarios require writerInitialBalance buffer.');
106
+ }
107
+ const initialBalance = toBalance(writerInitialBalance);
108
+ if (!initialBalance) {
109
+ throw new Error('Failed to decode writerInitialBalance buffer.');
110
+ }
111
+ const balanceAfterFee = initialBalance.sub(BALANCE_FEE);
112
+ if (!balanceAfterFee) {
113
+ throw new Error('Failed to compute writer balance after fee deduction.');
114
+ }
115
+ const availableBalance = balanceAfterFee.sub(BALANCE_TO_STAKE);
116
+ if (!availableBalance) {
117
+ throw new Error('Failed to compute writer liquid balance after staking.');
118
+ }
119
+ return {
120
+ balanceBuffer: availableBalance.value,
121
+ stakedBalanceBuffer: BALANCE_TO_STAKE.value
122
+ };
123
+ }
124
+
125
+ async function assertWriterStillPromoted(t, base, writerPeer, expected) {
126
+ const writerAddress = writerPeer.wallet.address;
127
+ const writerEntry = await base.view.get(writerAddress);
128
+ t.ok(writerEntry, 'writer entry still exists after failed removeWriter');
129
+ const decodedWriter = nodeEntryUtils.decode(writerEntry?.value);
130
+ t.ok(decodedWriter, 'writer entry decodes after failed removeWriter');
131
+ if (!decodedWriter) return;
132
+
133
+ t.is(decodedWriter.isWhitelisted, true, 'writer stays whitelisted');
134
+ t.is(decodedWriter.isWriter, true, 'writer role remains assigned');
135
+ t.is(decodedWriter.isIndexer, false, 'writer does not become an indexer');
136
+ t.ok(
137
+ b4a.equals(decodedWriter.balance, expected.balanceBuffer),
138
+ 'writer balance remains untouched'
139
+ );
140
+ t.ok(
141
+ b4a.equals(decodedWriter.stakedBalance, expected.stakedBalanceBuffer),
142
+ 'writer stake remains untouched'
143
+ );
144
+
145
+ const writerAddressBuffer = addressUtils.addressToBuffer(writerAddress);
146
+ const writingKeyHex = writerPeer.base.local.key.toString('hex');
147
+ const registryKey = EntryType.WRITER_ADDRESS + writingKeyHex;
148
+ const registryEntry = await base.view.get(registryKey);
149
+ t.ok(registryEntry, 'writer registry entry persists after failed removeWriter');
150
+ if (registryEntry?.value) {
151
+ t.ok(
152
+ b4a.equals(registryEntry.value, writerAddressBuffer),
153
+ 'writer registry still maps writing key to address'
154
+ );
155
+ }
156
+ }
157
+
158
+ async function assertPayloadProcessedByValidator(t, payload, expectedValidatorAddress) {
159
+ const decodedOperation = safeDecodeApplyOperation(payload);
160
+ t.ok(decodedOperation, 'removeWriter payload decodes');
161
+ const validatorAddressBuffer = decodedOperation?.rao?.va;
162
+ t.ok(validatorAddressBuffer, 'removeWriter payload carries validator address');
163
+ if (!validatorAddressBuffer) return;
164
+ const validatorAddress = addressUtils.bufferToAddress(validatorAddressBuffer);
165
+ t.ok(validatorAddress, 'removeWriter payload validator address decodes');
166
+ if (!validatorAddress) return;
167
+ t.is(
168
+ validatorAddress,
169
+ expectedValidatorAddress,
170
+ 'removeWriter payload signed by the expected validator'
171
+ );
172
+ }
173
+
174
+ export { buildRemoveWriterPayload, selectWriterPeer };
175
+
176
+ export async function buildRemoveWriterPayloadWithTxValidity(context, mutatedTxValidity, options = {}) {
177
+ if (!b4a.isBuffer(mutatedTxValidity)) {
178
+ throw new Error('buildRemoveWriterPayloadWithTxValidity requires a tx validity buffer.');
179
+ }
180
+ const { readerPeer = selectWriterPeer(context), validatorPeer = context.adminBootstrap, writerKeyBuffer = null } = options;
181
+ const writerKey = writerKeyBuffer ?? readerPeer.base.local.key;
182
+ const partial = await PartialStateMessageOperations.assembleRemoveWriterMessage(
183
+ readerPeer.wallet,
184
+ writerKey.toString('hex'),
185
+ mutatedTxValidity.toString('hex')
186
+ );
187
+ return CompleteStateMessageOperations.assembleRemoveWriterMessage(
188
+ validatorPeer.wallet,
189
+ partial.address,
190
+ b4a.from(partial.rao.tx, 'hex'),
191
+ mutatedTxValidity,
192
+ b4a.from(partial.rao.iw, 'hex'),
193
+ b4a.from(partial.rao.in, 'hex'),
194
+ b4a.from(partial.rao.is, 'hex')
195
+ );
196
+ }
197
+
198
+ export async function snapshotDowngradedWriterEntry(context) {
199
+ const writerPeer = selectWriterPeer(context);
200
+ const validatorPeer = context.adminBootstrap;
201
+ const entry = await validatorPeer.base.view.get(writerPeer.wallet.address);
202
+ if (!entry?.value) {
203
+ throw new Error('Snapshot requires writer entry after removeWriter.');
204
+ }
205
+ context.removeWriterScenario = {
206
+ ...(context.removeWriterScenario ?? {}),
207
+ writerAddress: writerPeer.wallet.address,
208
+ downgradedEntry: b4a.from(entry.value)
209
+ };
210
+ }
211
+
212
+ export async function assertDowngradedWriterSnapshot(t, context, { skipSync = false } = {}) {
213
+ const snapshot = context.removeWriterScenario?.downgradedEntry;
214
+ const writerAddress = context.removeWriterScenario?.writerAddress;
215
+ if (!snapshot || !writerAddress) {
216
+ throw new Error('Missing writer snapshot for removeWriter replay assertion.');
217
+ }
218
+
219
+ await assertEntryMatchesSnapshot(t, context.adminBootstrap.base, writerAddress, snapshot);
220
+
221
+ if (!skipSync) {
222
+ await context.sync();
223
+ const writerPeer = selectWriterPeer(context);
224
+ await assertEntryMatchesSnapshot(t, writerPeer.base, writerAddress, snapshot);
225
+ }
226
+ }
227
+
228
+ async function assertEntryMatchesSnapshot(t, base, address, snapshot) {
229
+ const entry = await base.view.get(address);
230
+ t.ok(entry, 'writer entry exists during replay assertion');
231
+ t.ok(entry?.value, 'writer entry has a value during replay assertion');
232
+ if (!entry?.value) return;
233
+ t.ok(b4a.equals(entry.value, snapshot), 'writer entry remains unchanged after replay attempt');
234
+ }
235
+
236
+ export async function applyWithWriterRegistryRemoval(context, invalidPayload) {
237
+ const writerPeer = selectWriterPeer(context);
238
+ const writingKeyHex = writerPeer.base.local.key.toString('hex');
239
+ await withWriterRegistryOverrideOnApply({
240
+ context,
241
+ writingKeyHex,
242
+ getOverride: () => null,
243
+ fn: async node => {
244
+ await node.base.append(invalidPayload);
245
+ await node.base.update();
246
+ await eventFlush();
247
+ }
248
+ });
249
+ }
250
+
251
+ export async function applyWithWriterRegistryForeignAddress(context, invalidPayload, foreignPeer) {
252
+ const writerPeer = selectWriterPeer(context);
253
+ const writingKeyHex = writerPeer.base.local.key.toString('hex');
254
+ const foreignAddress = foreignPeer?.wallet?.address;
255
+ if (!foreignAddress) {
256
+ throw new Error('Foreign registry override requires a peer with an address.');
257
+ }
258
+ const foreignEntry = {
259
+ value: addressUtils.addressToBuffer(foreignAddress)
260
+ };
261
+ await withWriterRegistryOverrideOnApply({
262
+ context,
263
+ writingKeyHex,
264
+ getOverride: () => foreignEntry,
265
+ fn: async node => {
266
+ await node.base.append(invalidPayload);
267
+ await node.base.update();
268
+ await eventFlush();
269
+ }
270
+ });
271
+ }
272
+
273
+ async function withWriterRegistryOverrideOnApply({
274
+ context,
275
+ writingKeyHex,
276
+ getOverride,
277
+ fn,
278
+ selectNode = defaultSelectNode
279
+ }) {
280
+ if (!writingKeyHex) {
281
+ throw new Error('Writer registry override requires a writing key.');
282
+ }
283
+ const node = selectNode(context);
284
+ const base = node.base;
285
+ const registryKey = EntryType.WRITER_ADDRESS + writingKeyHex;
286
+ const originalApply = base._handlers.apply;
287
+
288
+ base._handlers.apply = async function patchedApply(nodes, view, baseCtx) {
289
+ const originalBatch = view.batch;
290
+ view.batch = function patchedBatch(...args) {
291
+ const batch = originalBatch.apply(this, args);
292
+ if (!batch?.get) return batch;
293
+
294
+ const originalGet = batch.get.bind(batch);
295
+ batch.get = async key => {
296
+ if (key === registryKey) {
297
+ return getOverride();
298
+ }
299
+ return originalGet(key);
300
+ };
301
+
302
+ return batch;
303
+ };
304
+
305
+ try {
306
+ return await originalApply.call(this, nodes, view, baseCtx);
307
+ } finally {
308
+ view.batch = originalBatch;
309
+ }
310
+ };
311
+
312
+ try {
313
+ await fn(node);
314
+ } finally {
315
+ base._handlers.apply = originalApply;
316
+ }
317
+ }
318
+
319
+ function defaultSelectNode(context) {
320
+ return context.adminBootstrap ?? context.bootstrap ?? context.peers?.[0] ?? null;
321
+ }
322
+
323
+ export async function applyWithRequesterRoleMutationFailure(context, invalidPayload) {
324
+ const node = context.adminBootstrap ?? context.bootstrap;
325
+ if (!node?.base) {
326
+ throw new Error('Role mutation failure scenario requires a writable node.');
327
+ }
328
+ const originalSetRole = nodeEntryUtils.setRole;
329
+ let shouldFailNextSetRole = true;
330
+ nodeEntryUtils.setRole = function patchedSetRole(...args) {
331
+ if (shouldFailNextSetRole) {
332
+ shouldFailNextSetRole = false;
333
+ return null;
334
+ }
335
+ return originalSetRole(...args);
336
+ };
337
+ try {
338
+ await node.base.append(invalidPayload);
339
+ await node.base.update();
340
+ await eventFlush();
341
+ } finally {
342
+ nodeEntryUtils.setRole = originalSetRole;
343
+ }
344
+ }
@@ -0,0 +1,113 @@
1
+ import { test } from 'brittle';
2
+ import b4a from 'b4a';
3
+ import nodeEntryUtils from '../../../../../src/core/state/utils/nodeEntry.js';
4
+ import {
5
+ setupAddWriterScenario,
6
+ selectWriterPeer,
7
+ buildAddWriterPayload,
8
+ assertAddWriterSuccessState,
9
+ defaultWriterFunding
10
+ } from '../addWriter/addWriterScenarioHelpers.js';
11
+ import {
12
+ buildRemoveWriterPayload,
13
+ assertRemoveWriterSuccessState
14
+ } from './removeWriterScenarioHelpers.js';
15
+ import { initializeBalances, whitelistAddress } from '../common/commonScenarioHelper.js';
16
+ import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
17
+
18
+ export default function removeWriterThroughWriterValidatorScenario() {
19
+ test(
20
+ 'State.apply removeWriter rewards writer validators processing peer removals',
21
+ async t => {
22
+ const context = await setupAddWriterScenario(t, { nodes: 3 });
23
+ const adminPeer = context.adminBootstrap;
24
+ const validatorWriterPeer = selectWriterPeer(context, 0);
25
+ const targetWriterPeer = selectWriterPeer(context, 1);
26
+ if (!targetWriterPeer) {
27
+ t.fail('Writer validator scenario requires a second reader peer.');
28
+ return;
29
+ }
30
+
31
+ await fundAndWhitelistPeer(context, targetWriterPeer);
32
+
33
+ const adminBalanceBeforeFirstAdd = await readNodeBalanceBuffer(adminPeer);
34
+ const firstAddPayload = await buildAddWriterPayload(context, {
35
+ readerPeer: validatorWriterPeer,
36
+ validatorPeer: adminPeer
37
+ });
38
+
39
+ await appendPayload(adminPeer.base, firstAddPayload);
40
+
41
+ await assertAddWriterSuccessState(t, context, {
42
+ readerPeer: validatorWriterPeer,
43
+ validatorPeer: adminPeer,
44
+ writerInitialBalance: context.addWriterScenario?.writerInitialBalance,
45
+ validatorBalanceBefore: adminBalanceBeforeFirstAdd,
46
+ payload: firstAddPayload,
47
+ expectedWriterIndex: 1
48
+ });
49
+
50
+ const adminBalanceBeforeSecondAdd = await readNodeBalanceBuffer(adminPeer);
51
+ const targetWriterInitialBalance = await readNodeBalanceBuffer(targetWriterPeer);
52
+ const secondAddPayload = await buildAddWriterPayload(context, {
53
+ readerPeer: targetWriterPeer,
54
+ validatorPeer: adminPeer
55
+ });
56
+
57
+ await appendPayload(adminPeer.base, secondAddPayload);
58
+
59
+ await assertAddWriterSuccessState(t, context, {
60
+ readerPeer: targetWriterPeer,
61
+ validatorPeer: adminPeer,
62
+ writerInitialBalance:
63
+ targetWriterInitialBalance ?? context.addWriterScenario?.writerInitialBalance,
64
+ validatorBalanceBefore: adminBalanceBeforeSecondAdd,
65
+ payload: secondAddPayload,
66
+ expectedWriterIndex: 2
67
+ });
68
+
69
+ const writerEntryBeforeRemoval = await adminPeer.base.view.get(targetWriterPeer.wallet.address);
70
+ t.ok(writerEntryBeforeRemoval, 'target writer entry exists before removeWriter');
71
+ const validatorEntryBeforeRemoval = await validatorWriterPeer.base.view.get(
72
+ validatorWriterPeer.wallet.address
73
+ );
74
+ t.ok(validatorEntryBeforeRemoval, 'validator writer entry exists before removeWriter');
75
+
76
+ const removeWriterPayload = await buildRemoveWriterPayload(context, {
77
+ readerPeer: targetWriterPeer,
78
+ validatorPeer: validatorWriterPeer,
79
+ writerKeyBuffer: targetWriterPeer.base.local.key
80
+ });
81
+
82
+ await appendPayload(validatorWriterPeer.base, removeWriterPayload);
83
+
84
+ await assertRemoveWriterSuccessState(t, context, {
85
+ writerPeer: targetWriterPeer,
86
+ validatorPeer: validatorWriterPeer,
87
+ writerEntryBefore: writerEntryBeforeRemoval,
88
+ validatorEntryBefore: validatorEntryBeforeRemoval,
89
+ payload: removeWriterPayload
90
+ });
91
+ }
92
+ );
93
+ }
94
+
95
+ async function fundAndWhitelistPeer(context, peer) {
96
+ const funding = context.addWriterScenario?.writerInitialBalance ?? defaultWriterFunding;
97
+ await initializeBalances(context, [[peer.wallet.address, funding]]);
98
+ await whitelistAddress(context, peer.wallet.address);
99
+ }
100
+
101
+ async function appendPayload(base, payload) {
102
+ await base.append(payload);
103
+ await base.update();
104
+ await eventFlush();
105
+ }
106
+
107
+ async function readNodeBalanceBuffer(peer) {
108
+ const entry = await peer.base.view.get(peer.wallet.address);
109
+ if (!entry?.value) return null;
110
+ const decoded = nodeEntryUtils.decode(entry.value);
111
+ if (!decoded?.balance) return null;
112
+ return b4a.from(decoded.balance);
113
+ }
@@ -0,0 +1,33 @@
1
+ import b4a from 'b4a';
2
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
3
+ import {
4
+ setupRemoveWriterScenario,
5
+ buildRemoveWriterPayload,
6
+ assertRemoveWriterFailureState
7
+ } from './removeWriterScenarioHelpers.js';
8
+ import { applyWithStakeEntryMutation } from '../addWriter/addWriterScenarioHelpers.js';
9
+ import nodeEntryUtils from '../../../../../src/core/state/utils/nodeEntry.js';
10
+ import { BALANCE_ZERO } from '../../../../../src/core/state/utils/balance.js';
11
+
12
+ export default function removeWriterUnstakeFailureScenario() {
13
+ new OperationValidationScenarioBase({
14
+ title: 'State.apply removeWriter rejects payloads when unstaking requester balance fails',
15
+ setupScenario: setupRemoveWriterScenario,
16
+ buildValidPayload: context => buildRemoveWriterPayload(context),
17
+ mutatePayload: (_t, payload) => payload,
18
+ applyInvalidPayload: (context, invalidPayload) =>
19
+ applyWithStakeEntryMutation(context, invalidPayload, nodeEntryBuffer => {
20
+ if (!nodeEntryBuffer) return nodeEntryBuffer;
21
+ const mutatedBuffer = nodeEntryUtils.setStakedBalance(
22
+ b4a.from(nodeEntryBuffer),
23
+ BALANCE_ZERO.value
24
+ );
25
+ if (!mutatedBuffer) {
26
+ throw new Error('Failed to mutate staked balance for unstake failure scenario.');
27
+ }
28
+ return mutatedBuffer;
29
+ }),
30
+ assertStateUnchanged: (t, context) => assertRemoveWriterFailureState(t, context, { skipSync: true }),
31
+ expectedLogs: ['Failed to unstake balance for writer.']
32
+ }).performScenario();
33
+ }
@@ -0,0 +1,21 @@
1
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
2
+ import {
3
+ setupRemoveWriterScenario,
4
+ buildRemoveWriterPayload,
5
+ assertRemoveWriterFailureState
6
+ } from './removeWriterScenarioHelpers.js';
7
+ import { applyWithRequesterWriterKeyMismatch } from '../addWriter/addWriterScenarioHelpers.js';
8
+
9
+ export default function removeWriterWriterKeyMismatchScenario() {
10
+ new OperationValidationScenarioBase({
11
+ title: 'State.apply removeWriter rejects payloads when requester writer key mismatches entry',
12
+ setupScenario: setupRemoveWriterScenario,
13
+ buildValidPayload: context => buildRemoveWriterPayload(context),
14
+ mutatePayload: (_t, payload) => payload,
15
+ applyInvalidPayload: applyWithRequesterWriterKeyMismatch,
16
+ assertStateUnchanged: (t, context) => assertRemoveWriterFailureState(t, context, { skipSync: true }),
17
+ expectedLogs: [
18
+ "Writer key must be registered, match node's current key, and belong to the requester."
19
+ ]
20
+ }).performScenario();
21
+ }
@@ -0,0 +1,26 @@
1
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
2
+ import {
3
+ setupRemoveWriterScenario,
4
+ buildRemoveWriterPayload,
5
+ assertRemoveWriterFailureState,
6
+ applyWithWriterRegistryForeignAddress
7
+ } from './removeWriterScenarioHelpers.js';
8
+ import { selectValidatorPeerWithoutEntry } from '../addWriter/addWriterScenarioHelpers.js';
9
+
10
+ export default function removeWriterWriterKeyOwnershipScenario() {
11
+ new OperationValidationScenarioBase({
12
+ title: 'State.apply removeWriter rejects payloads when writer key belongs to another node',
13
+ setupScenario: t => setupRemoveWriterScenario(t, { nodes: 3 }),
14
+ buildValidPayload: context => buildRemoveWriterPayload(context),
15
+ mutatePayload: (_t, payload) => payload,
16
+ applyInvalidPayload: (context, invalidPayload) => {
17
+ const foreignPeer =
18
+ selectValidatorPeerWithoutEntry(context) ?? context.peers?.find(peer => peer !== context.adminBootstrap);
19
+ return applyWithWriterRegistryForeignAddress(context, invalidPayload, foreignPeer ?? context.peers[1]);
20
+ },
21
+ assertStateUnchanged: (t, context) => assertRemoveWriterFailureState(t, context, { skipSync: true }),
22
+ expectedLogs: [
23
+ "Writer key must be registered, match node's current key, and belong to the requester."
24
+ ]
25
+ }).performScenario();
26
+ }
@@ -0,0 +1,22 @@
1
+ import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
2
+ import {
3
+ setupRemoveWriterScenario,
4
+ buildRemoveWriterPayload,
5
+ assertRemoveWriterFailureState,
6
+ applyWithWriterRegistryRemoval
7
+ } from './removeWriterScenarioHelpers.js';
8
+
9
+ export default function removeWriterWriterKeyRegistryMissingScenario() {
10
+ new OperationValidationScenarioBase({
11
+ title:
12
+ 'State.apply removeWriter rejects payloads when writer key is not registered to the requester',
13
+ setupScenario: setupRemoveWriterScenario,
14
+ buildValidPayload: context => buildRemoveWriterPayload(context),
15
+ mutatePayload: (_t, payload) => payload,
16
+ applyInvalidPayload: applyWithWriterRegistryRemoval,
17
+ assertStateUnchanged: (t, context) => assertRemoveWriterFailureState(t, context, { skipSync: true }),
18
+ expectedLogs: [
19
+ "Writer key must be registered, match node's current key, and belong to the requester."
20
+ ]
21
+ }).performScenario();
22
+ }