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.
- package/.dockerignore +16 -0
- package/.github/workflows/acceptance-tests.yml +7 -0
- package/.github/workflows/publish.yml +40 -0
- package/.github/workflows/{CI.yml → unit-tests.yml} +1 -1
- package/README.md +175 -50
- package/docker-compose.yml +16 -0
- package/dockerfile +41 -0
- package/docs/fee_distribution.md +89 -0
- package/msb.mjs +12 -14
- package/package.json +8 -4
- package/rpc/constants.mjs +4 -1
- package/rpc/handlers.mjs +109 -66
- package/rpc/routes/v1.mjs +3 -1
- package/rpc/rpc_services.js +126 -0
- package/rpc/utils/confirmedParameter.mjs +17 -0
- package/rpc/utils/url.mjs +38 -0
- package/src/core/network/Network.js +27 -10
- package/src/core/network/identity/NetworkWalletFactory.js +78 -0
- package/src/core/network/services/ConnectionManager.js +2 -2
- package/src/core/network/services/ValidatorObserverService.js +7 -4
- package/src/core/state/State.js +28 -22
- package/src/index.js +197 -385
- package/src/utils/cliCommands.js +280 -0
- package/src/utils/constants.js +3 -1
- package/tests/acceptance/v1/account/account.test.mjs +123 -0
- package/tests/acceptance/v1/balance/balance.test.mjs +55 -0
- package/tests/acceptance/v1/broadcast-transaction/broadcast-transaction.test.mjs +111 -0
- package/tests/acceptance/v1/confirmed-length/confirmed-length.test.mjs +19 -0
- package/tests/acceptance/v1/fee/fee.test.mjs +11 -0
- package/tests/acceptance/v1/rpc.test.mjs +62 -291
- package/tests/acceptance/v1/tx/tx.test.mjs +98 -0
- package/tests/acceptance/v1/tx-details/tx-details.test.mjs +195 -0
- package/tests/acceptance/v1/tx-hashes/tx-hashes.test.mjs +72 -0
- package/tests/acceptance/v1/tx-payloads-bulk/tx-payloads-bulk.test.mjs +27 -0
- package/tests/acceptance/v1/txv/txv.test.mjs +11 -0
- package/tests/acceptance/v1/unconfirmed-length/unconfirmed-length.test.mjs +11 -0
- package/tests/helpers/StateNetworkFactory.js +157 -0
- package/tests/helpers/autobaseTestHelpers.js +369 -0
- package/tests/helpers/createTestSignature.js +12 -0
- package/tests/helpers/transactionPayloads.mjs +78 -0
- package/tests/unit/network/NetworkWalletFactory.test.js +156 -0
- package/tests/unit/state/apply/addAdmin/addAdminHappyPathScenario.js +38 -0
- package/tests/unit/state/apply/addAdmin/addAdminScenarioHelpers.js +273 -0
- package/tests/unit/state/apply/addAdmin/adminEntryEncodingFailureScenario.js +30 -0
- package/tests/unit/state/apply/addAdmin/adminEntryExistsScenario.js +78 -0
- package/tests/unit/state/apply/addAdmin/nodeEntryInitializationFailureScenario.js +30 -0
- package/tests/unit/state/apply/addAdmin/nonBootstrapNodeScenario.js +68 -0
- package/tests/unit/state/apply/addAdmin/state.apply.addAdmin.test.js +155 -0
- package/tests/unit/state/apply/addIndexer/addIndexerHappyPathScenario.js +39 -0
- package/tests/unit/state/apply/addIndexer/addIndexerMultipleIndexersInTheNetworkScenario.js +167 -0
- package/tests/unit/state/apply/addIndexer/addIndexerPretenderAlreadyIndexerScenario.js +21 -0
- package/tests/unit/state/apply/addIndexer/addIndexerPretenderNotWriterScenario.js +21 -0
- package/tests/unit/state/apply/addIndexer/addIndexerRemoveAndReAddScenario.js +186 -0
- package/tests/unit/state/apply/addIndexer/addIndexerScenarioHelpers.js +445 -0
- package/tests/unit/state/apply/addIndexer/addIndexerWriterKeyAlreadyRegisteredScenario.js +32 -0
- package/tests/unit/state/apply/addIndexer/state.apply.addIndexer.test.js +297 -0
- package/tests/unit/state/apply/addWriter/addWriterHappyPathScenario.js +41 -0
- package/tests/unit/state/apply/addWriter/addWriterInvalidValidatorSignatureScenario.js +32 -0
- package/tests/unit/state/apply/addWriter/addWriterNewWkScenario.js +149 -0
- package/tests/unit/state/apply/addWriter/addWriterRequesterAlreadyWriterScenario.js +21 -0
- package/tests/unit/state/apply/addWriter/addWriterRequesterBalanceInsufficientScenario.js +21 -0
- package/tests/unit/state/apply/addWriter/addWriterRequesterEntryDecodeFailureScenario.js +19 -0
- package/tests/unit/state/apply/addWriter/addWriterRequesterEntryMissingScenario.js +19 -0
- package/tests/unit/state/apply/addWriter/addWriterRequesterIndexerScenario.js +21 -0
- package/tests/unit/state/apply/addWriter/addWriterRequesterNotWhitelistedScenario.js +21 -0
- package/tests/unit/state/apply/addWriter/addWriterScenarioHelpers.js +757 -0
- package/tests/unit/state/apply/addWriter/addWriterStakeBalanceUpdateFailureScenario.js +50 -0
- package/tests/unit/state/apply/addWriter/addWriterStakeInsufficientBalanceScenario.js +29 -0
- package/tests/unit/state/apply/addWriter/addWriterStakeInvalidBalanceScenario.js +29 -0
- package/tests/unit/state/apply/addWriter/addWriterStakeInvalidEntryScenario.js +21 -0
- package/tests/unit/state/apply/addWriter/addWriterStakeStakedBalanceFailureScenario.js +37 -0
- package/tests/unit/state/apply/addWriter/addWriterStakeSubtractFailureScenario.js +42 -0
- package/tests/unit/state/apply/addWriter/addWriterValidatorRewardScenario.js +105 -0
- package/tests/unit/state/apply/addWriter/addWriterWriterKeyMismatchScenario.js +54 -0
- package/tests/unit/state/apply/addWriter/addWriterWriterKeyOwnershipScenario.js +54 -0
- package/tests/unit/state/apply/addWriter/addWriterZeroWriterKeyScenario.js +29 -0
- package/tests/unit/state/apply/addWriter/state.apply.addWriter.test.js +309 -0
- package/tests/unit/state/apply/adminRecovery/adminRecoveryHappyPathScenario.js +30 -0
- package/tests/unit/state/apply/adminRecovery/adminRecoveryScenarioHelpers.js +866 -0
- package/tests/unit/state/apply/adminRecovery/state.apply.adminRecovery.test.js +439 -0
- package/tests/unit/state/apply/appendWhitelist/appendWhitelistBanAndReapplyScenario.js +78 -0
- package/tests/unit/state/apply/appendWhitelist/appendWhitelistExistingReaderHappyPathScenario.js +98 -0
- package/tests/unit/state/apply/appendWhitelist/appendWhitelistFeeAfterDisableScenario.js +66 -0
- package/tests/unit/state/apply/appendWhitelist/appendWhitelistHappyPathScenario.js +55 -0
- package/tests/unit/state/apply/appendWhitelist/appendWhitelistInsufficientAdminBalanceScenario.js +103 -0
- package/tests/unit/state/apply/appendWhitelist/appendWhitelistNodeAlreadyWhitelistedScenario.js +60 -0
- package/tests/unit/state/apply/appendWhitelist/appendWhitelistScenarioHelpers.js +191 -0
- package/tests/unit/state/apply/appendWhitelist/state.apply.appendWhitelist.test.js +220 -0
- package/tests/unit/state/apply/balanceInitialization/balanceInitializationHappyPathScenario.js +82 -0
- package/tests/unit/state/apply/balanceInitialization/balanceInitializationScenarioHelpers.js +106 -0
- package/tests/unit/state/apply/balanceInitialization/invalidAmountScenario.js +45 -0
- package/tests/unit/state/apply/balanceInitialization/nodeEntryBalanceUpdateFailureScenario.js +81 -0
- package/tests/unit/state/apply/balanceInitialization/state.apply.balanceInitialization.test.js +189 -0
- package/tests/unit/state/apply/banValidator/banValidatorBanAndReWhitelistScenario.js +155 -0
- package/tests/unit/state/apply/banValidator/banValidatorHappyPathScenario.js +36 -0
- package/tests/unit/state/apply/banValidator/banValidatorScenarioHelpers.js +534 -0
- package/tests/unit/state/apply/banValidator/banValidatorSequentialBansScenario.js +74 -0
- package/tests/unit/state/apply/banValidator/banValidatorTargetDecodeFailureScenario.js +19 -0
- package/tests/unit/state/apply/banValidator/banValidatorTargetIndexerScenario.js +32 -0
- package/tests/unit/state/apply/banValidator/banValidatorTargetNodeEntryMissingScenario.js +19 -0
- package/tests/unit/state/apply/banValidator/banValidatorTargetRoleUpdateFailureScenario.js +19 -0
- package/tests/unit/state/apply/banValidator/banValidatorWhitelistedNonWriterScenario.js +38 -0
- package/tests/unit/state/apply/banValidator/banValidatorWhitelistedZeroBalanceScenario.js +91 -0
- package/tests/unit/state/apply/banValidator/banValidatorWithdrawFailureScenario.js +19 -0
- package/tests/unit/state/apply/banValidator/state.apply.banValidator.test.js +266 -0
- package/tests/unit/state/apply/bootstrapDeployment/bootstrapDeploymentDuplicateRegistrationScenario.js +142 -0
- package/tests/unit/state/apply/bootstrapDeployment/bootstrapDeploymentHappyPathScenario.js +26 -0
- package/tests/unit/state/apply/bootstrapDeployment/bootstrapDeploymentIncompleteOperationScenario.js +94 -0
- package/tests/unit/state/apply/bootstrapDeployment/bootstrapDeploymentInvalidDeploymentEntryScenario.js +37 -0
- package/tests/unit/state/apply/bootstrapDeployment/bootstrapDeploymentMultipleBootstrapScenario.js +86 -0
- package/tests/unit/state/apply/bootstrapDeployment/bootstrapDeploymentScenarioHelpers.js +344 -0
- package/tests/unit/state/apply/bootstrapDeployment/invalidValidatorNodeEntryScenario.js +57 -0
- package/tests/unit/state/apply/bootstrapDeployment/state.apply.bootstrapDeployment.test.js +429 -0
- package/tests/unit/state/apply/common/access-control/adminConsistencyMismatchScenario.js +119 -0
- package/tests/unit/state/apply/common/access-control/adminEntryDecodeFailureScenario.js +130 -0
- package/tests/unit/state/apply/common/access-control/adminEntryExistsScenario.js +93 -0
- package/tests/unit/state/apply/common/access-control/adminEntryMissingScenario.js +108 -0
- package/tests/unit/state/apply/common/access-control/adminOnlyGuardScenario.js +126 -0
- package/tests/unit/state/apply/common/access-control/adminPublicKeyDecodeFailureScenario.js +120 -0
- package/tests/unit/state/apply/common/access-control/roleAccessOperationValidationScenario.js +50 -0
- package/tests/unit/state/apply/common/adminControlOperationValidationScenario.js +56 -0
- package/tests/unit/state/apply/common/balances/adminEntryUpdateFailureScenario.js +52 -0
- package/tests/unit/state/apply/common/balances/base/requesterBalanceScenarioBase.js +197 -0
- package/tests/unit/state/apply/common/balances/feeDecodeFailureScenario.js +52 -0
- package/tests/unit/state/apply/common/balances/requesterBalanceDecodeFailureScenario.js +15 -0
- package/tests/unit/state/apply/common/balances/requesterBalanceFeeApplicationFailureScenario.js +11 -0
- package/tests/unit/state/apply/common/balances/requesterBalanceInsufficientScenario.js +15 -0
- package/tests/unit/state/apply/common/balances/requesterBalanceUpdateFailureScenario.js +11 -0
- package/tests/unit/state/apply/common/balances/validatorEntryRewardFailureScenario.js +11 -0
- package/tests/unit/state/apply/common/balances/validatorEntryUpdateFailureScenario.js +11 -0
- package/tests/unit/state/apply/common/balances/validatorNodeEntryDecodeFailureScenario.js +40 -0
- package/tests/unit/state/apply/common/base/OperationValidationScenarioBase.js +114 -0
- package/tests/unit/state/apply/common/commonScenarioHelper.js +103 -0
- package/tests/unit/state/apply/common/indexer/indexerNodeEntryDecodeFailureScenario.js +36 -0
- package/tests/unit/state/apply/common/indexer/indexerNodeEntryMissingScenario.js +36 -0
- package/tests/unit/state/apply/common/indexer/indexerRoleUpdateFailureScenario.js +29 -0
- package/tests/unit/state/apply/common/indexer/indexerSequenceStateInvalidScenario.js +66 -0
- package/tests/unit/state/apply/common/invalidMessageComponentValidationScenario.js +84 -0
- package/tests/unit/state/apply/common/nodeEntryInitializationFailureScenario.js +47 -0
- package/tests/unit/state/apply/common/operationAlreadyAppliedScenario.js +85 -0
- package/tests/unit/state/apply/common/payload-structure/addressWithInvalidPublicKeyScenario.js +52 -0
- package/tests/unit/state/apply/common/payload-structure/initializationDisabledScenario.js +49 -0
- package/tests/unit/state/apply/common/payload-structure/invalidAddressValidationScenario.js +73 -0
- package/tests/unit/state/apply/common/payload-structure/invalidHashValidationScenario.js +71 -0
- package/tests/unit/state/apply/common/payload-structure/invalidPayloadValidationScenario.js +31 -0
- package/tests/unit/state/apply/common/payload-structure/invalidSignatureValidationScenario.js +142 -0
- package/tests/unit/state/apply/common/payload-structure/partialOperationValidationScenario.js +87 -0
- package/tests/unit/state/apply/common/requester/requesterNodeEntryBufferMissingScenario.js +70 -0
- package/tests/unit/state/apply/common/requester/requesterNodeEntryDecodeFailureScenario.js +72 -0
- package/tests/unit/state/apply/common/requester/requesterNodeEntryMissingScenario.js +36 -0
- package/tests/unit/state/apply/common/requesterAddressValidationScenario.js +44 -0
- package/tests/unit/state/apply/common/requesterPublicKeyValidationScenario.js +25 -0
- package/tests/unit/state/apply/common/transactionValidityMismatchScenario.js +98 -0
- package/tests/unit/state/apply/common/validatorConsistency/base/validatorConsistencyScenarioBase.js +201 -0
- package/tests/unit/state/apply/common/validatorConsistency/validatorEntryDecodeFailureScenario.js +17 -0
- package/tests/unit/state/apply/common/validatorConsistency/validatorEntryMissingScenario.js +44 -0
- package/tests/unit/state/apply/common/validatorConsistency/validatorInactiveScenario.js +19 -0
- package/tests/unit/state/apply/common/validatorConsistency/validatorWriterKeyMismatchScenario.js +18 -0
- package/tests/unit/state/apply/common/validatorEntryValidation/base/validatorEntryValidationScenarioBase.js +314 -0
- package/tests/unit/state/apply/common/validatorEntryValidation/validatorEntryInvalidBalanceScenario.js +18 -0
- package/tests/unit/state/apply/common/writerKeyExistsValidationScenario.js +43 -0
- package/tests/unit/state/apply/disableInitialization/disableInitializationAlreadyDisabledScenario.js +53 -0
- package/tests/unit/state/apply/disableInitialization/disableInitializationHappyPathScenario.js +24 -0
- package/tests/unit/state/apply/disableInitialization/disableInitializationScenarioHelpers.js +197 -0
- package/tests/unit/state/apply/disableInitialization/state.apply.disableInitialization.test.js +161 -0
- package/tests/unit/state/apply/missing-tests.md +18 -0
- package/tests/unit/state/apply/removeIndexer/removeIndexerHappyPathScenario.js +58 -0
- package/tests/unit/state/apply/removeIndexer/removeIndexerReAddAndRemoveAgainScenario.js +98 -0
- package/tests/unit/state/apply/removeIndexer/removeIndexerRemoveMultipleIndexersScenario.js +167 -0
- package/tests/unit/state/apply/removeIndexer/removeIndexerScenarioHelpers.js +428 -0
- package/tests/unit/state/apply/removeIndexer/removeIndexerTargetNotIndexerScenario.js +22 -0
- package/tests/unit/state/apply/removeIndexer/removeIndexerWriterKeyMissingScenario.js +20 -0
- package/tests/unit/state/apply/removeIndexer/state.apply.removeIndexer.test.js +291 -0
- package/tests/unit/state/apply/removeWriter/removeWriterAndAddWriterAgainScenario.js +87 -0
- package/tests/unit/state/apply/removeWriter/removeWriterHappyPathScenario.js +38 -0
- package/tests/unit/state/apply/removeWriter/removeWriterInvalidValidatorSignatureScenario.js +32 -0
- package/tests/unit/state/apply/removeWriter/removeWriterRequesterBalanceInsufficientScenario.js +21 -0
- package/tests/unit/state/apply/removeWriter/removeWriterRequesterEntryDecodeFailureScenario.js +19 -0
- package/tests/unit/state/apply/removeWriter/removeWriterRequesterEntryMissingScenario.js +19 -0
- package/tests/unit/state/apply/removeWriter/removeWriterRequesterIndexerScenario.js +21 -0
- package/tests/unit/state/apply/removeWriter/removeWriterRequesterNotWriterScenario.js +21 -0
- package/tests/unit/state/apply/removeWriter/removeWriterRequesterRoleUpdateFailureScenario.js +19 -0
- package/tests/unit/state/apply/removeWriter/removeWriterScenarioHelpers.js +344 -0
- package/tests/unit/state/apply/removeWriter/removeWriterThroughWriterValidatorScenario.js +113 -0
- package/tests/unit/state/apply/removeWriter/removeWriterUnstakeFailureScenario.js +33 -0
- package/tests/unit/state/apply/removeWriter/removeWriterWriterKeyMismatchScenario.js +21 -0
- package/tests/unit/state/apply/removeWriter/removeWriterWriterKeyOwnershipScenario.js +26 -0
- package/tests/unit/state/apply/removeWriter/removeWriterWriterKeyRegistryMissingScenario.js +22 -0
- package/tests/unit/state/apply/removeWriter/state.apply.removeWriter.test.js +307 -0
- package/tests/unit/state/apply/state.apply.test.js +24 -0
- package/tests/unit/state/apply/transfer/state.apply.transfer.test.js +819 -0
- package/tests/unit/state/apply/transfer/transferContractSchemaValidationScenario.js +22 -0
- package/tests/unit/state/apply/transfer/transferDoubleSpendAcrossValidatorsScenario.js +137 -0
- package/tests/unit/state/apply/transfer/transferDoubleSpendSameBatchScenario.js +63 -0
- package/tests/unit/state/apply/transfer/transferDoubleSpendSingleValidatorScenario.js +67 -0
- package/tests/unit/state/apply/transfer/transferExistingRecipientAmountScenario.js +31 -0
- package/tests/unit/state/apply/transfer/transferExistingRecipientZeroAmountScenario.js +31 -0
- package/tests/unit/state/apply/transfer/transferHandlerGuardScenarios.js +22 -0
- package/tests/unit/state/apply/transfer/transferHappyPathScenario.js +8 -0
- package/tests/unit/state/apply/transfer/transferInvalidIncomingDataScenario.js +66 -0
- package/tests/unit/state/apply/transfer/transferNewRecipientAmountScenario.js +31 -0
- package/tests/unit/state/apply/transfer/transferNewRecipientZeroAmountScenario.js +31 -0
- package/tests/unit/state/apply/transfer/transferScenarioHelpers.js +1167 -0
- package/tests/unit/state/apply/transfer/transferSelfTransferAmountScenario.js +38 -0
- package/tests/unit/state/apply/transfer/transferSelfTransferZeroAmountScenario.js +38 -0
- package/tests/unit/state/apply/transfer/transferValidatorRecipientAmountScenario.js +38 -0
- package/tests/unit/state/apply/transfer/transferValidatorRecipientZeroAmountScenario.js +38 -0
- package/tests/unit/state/apply/txOperation/state.apply.txOperation.test.js +318 -0
- package/tests/unit/state/apply/txOperation/txOperationBootstrapNotRegisteredScenario.js +70 -0
- package/tests/unit/state/apply/txOperation/txOperationDifferentValidatorCreatorHappyPathScenario.js +23 -0
- package/tests/unit/state/apply/txOperation/txOperationInvalidDeploymentEntryScenario.js +48 -0
- package/tests/unit/state/apply/txOperation/txOperationInvalidFeeAmountScenario.js +39 -0
- package/tests/unit/state/apply/txOperation/txOperationInvalidSubnetCreatorAddressScenario.js +46 -0
- package/tests/unit/state/apply/txOperation/txOperationRequesterCreatorHappyPathScenario.js +21 -0
- package/tests/unit/state/apply/txOperation/txOperationScenarioHelpers.js +429 -0
- package/tests/unit/state/apply/txOperation/txOperationStandardHappyPathScenario.js +21 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeAddCreatorBalanceFailureScenario.js +26 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeAddValidatorBalanceFailureScenario.js +25 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeAddValidatorBonusFailureScenario.js +27 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeDecodeCreatorEntryScenario.js +18 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeDecodeRequesterEntryScenario.js +17 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeDecodeValidatorEntryScenario.js +31 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeGuardBypassScenario.js +49 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeGuardScenarioFactory.js +92 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeInsufficientRequesterBalanceScenario.js +28 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeInvalidCreatorBalanceScenario.js +29 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeInvalidRequesterBalanceScenario.js +28 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeInvalidRequesterEntryScenario.js +17 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeInvalidValidatorBalanceScenario.js +33 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeMissingCreatorEntryScenario.js +18 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeSubtractFailureScenario.js +25 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeUpdateCreatorBalanceFailureScenario.js +26 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeUpdateFailureScenario.js +25 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeUpdateValidatorBalanceFailureScenario.js +26 -0
- package/tests/unit/state/apply/txOperation/txOperationTransferFeeUpdateValidatorBonusFailureScenario.js +27 -0
- package/tests/unit/state/apply/txOperation/txOperationValidatorCreatorHappyPathScenario.js +21 -0
- package/tests/unit/state/stateModule.test.js +1 -0
- package/tests/unit/state/stateTestUtils.js +5 -1
- package/.env +0 -3
|
@@ -0,0 +1,297 @@
|
|
|
1
|
+
import b4a from 'b4a';
|
|
2
|
+
import addIndexerHappyPathScenario from './addIndexerHappyPathScenario.js';
|
|
3
|
+
import addIndexerMultipleIndexersInTheNetworkScenario from './addIndexerMultipleIndexersInTheNetworkScenario.js';
|
|
4
|
+
import addIndexerRemoveAndReAddScenario from './addIndexerRemoveAndReAddScenario.js';
|
|
5
|
+
import addIndexerWriterKeyAlreadyRegisteredScenario from './addIndexerWriterKeyAlreadyRegisteredScenario.js';
|
|
6
|
+
import AdminControlOperationValidationScenario from '../common/adminControlOperationValidationScenario.js';
|
|
7
|
+
import RequesterAddressValidationScenario from '../common/requesterAddressValidationScenario.js';
|
|
8
|
+
import createRequesterPublicKeyValidationScenario from '../common/requesterPublicKeyValidationScenario.js';
|
|
9
|
+
import InvalidAddressValidationScenario from '../common/payload-structure/invalidAddressValidationScenario.js';
|
|
10
|
+
import createAddressWithInvalidPublicKeyScenario from '../common/payload-structure/addressWithInvalidPublicKeyScenario.js';
|
|
11
|
+
import AdminEntryMissingScenario from '../common/access-control/adminEntryMissingScenario.js';
|
|
12
|
+
import AdminEntryDecodeFailureScenario from '../common/access-control/adminEntryDecodeFailureScenario.js';
|
|
13
|
+
import AdminOnlyGuardScenario from '../common/access-control/adminOnlyGuardScenario.js';
|
|
14
|
+
import AdminPublicKeyDecodeFailureScenario from '../common/access-control/adminPublicKeyDecodeFailureScenario.js';
|
|
15
|
+
import AdminConsistencyMismatchScenario from '../common/access-control/adminConsistencyMismatchScenario.js';
|
|
16
|
+
import InvalidHashValidationScenario from '../common/payload-structure/invalidHashValidationScenario.js';
|
|
17
|
+
import InvalidSignatureValidationScenario, {
|
|
18
|
+
SignatureMutationStrategy
|
|
19
|
+
} from '../common/payload-structure/invalidSignatureValidationScenario.js';
|
|
20
|
+
import IndexerSequenceStateInvalidScenario from '../common/indexer/indexerSequenceStateInvalidScenario.js';
|
|
21
|
+
import TransactionValidityMismatchScenario from '../common/transactionValidityMismatchScenario.js';
|
|
22
|
+
import OperationAlreadyAppliedScenario from '../common/operationAlreadyAppliedScenario.js';
|
|
23
|
+
import RequesterBalanceScenarioBase from '../common/balances/base/requesterBalanceScenarioBase.js';
|
|
24
|
+
import RequesterBalanceInsufficientScenario from '../common/balances/requesterBalanceInsufficientScenario.js';
|
|
25
|
+
import RequesterBalanceFeeApplicationFailureScenario from '../common/balances/requesterBalanceFeeApplicationFailureScenario.js';
|
|
26
|
+
import addIndexerPretenderNotWriterScenario from './addIndexerPretenderNotWriterScenario.js';
|
|
27
|
+
import addIndexerPretenderAlreadyIndexerScenario from './addIndexerPretenderAlreadyIndexerScenario.js';
|
|
28
|
+
import IndexerNodeEntryMissingScenario from '../common/indexer/indexerNodeEntryMissingScenario.js';
|
|
29
|
+
import IndexerNodeEntryDecodeFailureScenario from '../common/indexer/indexerNodeEntryDecodeFailureScenario.js';
|
|
30
|
+
import IndexerRoleUpdateFailureScenario from '../common/indexer/indexerRoleUpdateFailureScenario.js';
|
|
31
|
+
import FeeDecodeFailureScenario from '../common/balances/feeDecodeFailureScenario.js';
|
|
32
|
+
import RequesterNodeEntryMissingScenario from '../common/requester/requesterNodeEntryMissingScenario.js';
|
|
33
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
34
|
+
import { applyWithRequesterEntryCorruption } from '../addWriter/addWriterScenarioHelpers.js';
|
|
35
|
+
import {
|
|
36
|
+
setupAddIndexerScenario,
|
|
37
|
+
buildAddIndexerPayload,
|
|
38
|
+
buildAddIndexerPayloadWithTxValidity,
|
|
39
|
+
assertAddIndexerFailureState,
|
|
40
|
+
assertAddIndexerSuccessState,
|
|
41
|
+
selectIndexerCandidatePeer,
|
|
42
|
+
applyWithIndexerRoleUpdateFailure
|
|
43
|
+
} from './addIndexerScenarioHelpers.js';
|
|
44
|
+
|
|
45
|
+
addIndexerHappyPathScenario();
|
|
46
|
+
addIndexerMultipleIndexersInTheNetworkScenario();
|
|
47
|
+
addIndexerRemoveAndReAddScenario();
|
|
48
|
+
|
|
49
|
+
// Handler validation order
|
|
50
|
+
new AdminControlOperationValidationScenario({
|
|
51
|
+
title: 'State.apply addIndexer rejects payloads when contract schema validation fails',
|
|
52
|
+
setupScenario: setupAddIndexerScenario,
|
|
53
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
54
|
+
assertStateUnchanged: assertAddIndexerFailureState,
|
|
55
|
+
expectedLogs: ['Contract schema validation failed.']
|
|
56
|
+
}).performScenario();
|
|
57
|
+
|
|
58
|
+
new RequesterAddressValidationScenario({
|
|
59
|
+
title: 'State.apply addIndexer rejects payloads when requester address is invalid',
|
|
60
|
+
setupScenario: setupAddIndexerScenario,
|
|
61
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
62
|
+
assertStateUnchanged: assertAddIndexerFailureState,
|
|
63
|
+
expectedLogs: ['Requester address is invalid.']
|
|
64
|
+
}).performScenario();
|
|
65
|
+
|
|
66
|
+
createRequesterPublicKeyValidationScenario({
|
|
67
|
+
title: 'State.apply addIndexer rejects payloads when requester public key is invalid',
|
|
68
|
+
setupScenario: setupAddIndexerScenario,
|
|
69
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
70
|
+
assertStateUnchanged: assertAddIndexerFailureState,
|
|
71
|
+
expectedLogs: ['Error while decoding requester public key.']
|
|
72
|
+
}).performScenario();
|
|
73
|
+
|
|
74
|
+
new InvalidAddressValidationScenario({
|
|
75
|
+
title: 'State.apply addIndexer rejects payloads when pretending indexer address is invalid',
|
|
76
|
+
setupScenario: setupAddIndexerScenario,
|
|
77
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
78
|
+
assertStateUnchanged: assertAddIndexerFailureState,
|
|
79
|
+
addressPath: ['aco', 'ia'],
|
|
80
|
+
expectedLogs: ['Pretending indexer address is invalid.']
|
|
81
|
+
}).performScenario();
|
|
82
|
+
|
|
83
|
+
createAddressWithInvalidPublicKeyScenario({
|
|
84
|
+
title: 'State.apply addIndexer rejects payloads when pretending indexer public key is invalid',
|
|
85
|
+
setupScenario: setupAddIndexerScenario,
|
|
86
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
87
|
+
assertStateUnchanged: assertAddIndexerFailureState,
|
|
88
|
+
addressPath: ['aco', 'ia'],
|
|
89
|
+
expectedLogs: ['Failed to decode pretending indexer public key.']
|
|
90
|
+
}).performScenario();
|
|
91
|
+
|
|
92
|
+
new AdminEntryMissingScenario({
|
|
93
|
+
title: 'State.apply addIndexer rejects payloads when admin entry is missing',
|
|
94
|
+
setupScenario: setupAddIndexerScenario,
|
|
95
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
96
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
97
|
+
expectedLogs: ['Invalid admin entry.']
|
|
98
|
+
}).performScenario();
|
|
99
|
+
|
|
100
|
+
new AdminEntryDecodeFailureScenario({
|
|
101
|
+
title: 'State.apply addIndexer rejects payloads when admin entry cannot be decoded',
|
|
102
|
+
setupScenario: setupAddIndexerScenario,
|
|
103
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
104
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
105
|
+
expectedLogs: ['Failed to decode admin entry.']
|
|
106
|
+
}).performScenario();
|
|
107
|
+
|
|
108
|
+
new AdminOnlyGuardScenario({
|
|
109
|
+
title: 'State.apply addIndexer rejects non-admin nodes',
|
|
110
|
+
setupScenario: setupAddIndexerScenario,
|
|
111
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
112
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
113
|
+
expectedLogs: ['Node is not allowed to perform this operation. (ADMIN ONLY)']
|
|
114
|
+
}).performScenario();
|
|
115
|
+
|
|
116
|
+
new AdminPublicKeyDecodeFailureScenario({
|
|
117
|
+
title: 'State.apply addIndexer rejects payloads when admin public key cannot be decoded',
|
|
118
|
+
setupScenario: setupAddIndexerScenario,
|
|
119
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
120
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
121
|
+
expectedLogs: ['Failed to decode admin public key.']
|
|
122
|
+
}).performScenario();
|
|
123
|
+
|
|
124
|
+
new AdminConsistencyMismatchScenario({
|
|
125
|
+
title: 'State.apply addIndexer rejects payloads when admin public key mismatches requester',
|
|
126
|
+
setupScenario: setupAddIndexerScenario,
|
|
127
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
128
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
129
|
+
expectedLogs: ['System admin and node public keys do not match.']
|
|
130
|
+
}).performScenario();
|
|
131
|
+
|
|
132
|
+
new InvalidHashValidationScenario({
|
|
133
|
+
title: 'State.apply addIndexer rejects payloads when message hash mismatches tx hash',
|
|
134
|
+
setupScenario: setupAddIndexerScenario,
|
|
135
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
136
|
+
assertStateUnchanged: assertAddIndexerFailureState,
|
|
137
|
+
expectedLogs: ['Message hash does not match the tx_hash.']
|
|
138
|
+
}).performScenario();
|
|
139
|
+
|
|
140
|
+
new InvalidSignatureValidationScenario({
|
|
141
|
+
title: 'State.apply addIndexer rejects payloads when admin signature is invalid (foreign signature)',
|
|
142
|
+
setupScenario: setupAddIndexerScenario,
|
|
143
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
144
|
+
assertStateUnchanged: assertAddIndexerFailureState,
|
|
145
|
+
expectedLogs: ['Failed to verify message signature.']
|
|
146
|
+
}).performScenario();
|
|
147
|
+
|
|
148
|
+
new InvalidSignatureValidationScenario({
|
|
149
|
+
title: 'State.apply addIndexer rejects payloads when admin signature is invalid (zero fill)',
|
|
150
|
+
setupScenario: setupAddIndexerScenario,
|
|
151
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
152
|
+
assertStateUnchanged: assertAddIndexerFailureState,
|
|
153
|
+
strategy: SignatureMutationStrategy.ZERO_FILL,
|
|
154
|
+
expectedLogs: ['Failed to verify message signature.']
|
|
155
|
+
}).performScenario();
|
|
156
|
+
|
|
157
|
+
new InvalidSignatureValidationScenario({
|
|
158
|
+
title: 'State.apply addIndexer rejects payloads when admin signature is invalid (type mismatch)',
|
|
159
|
+
setupScenario: setupAddIndexerScenario,
|
|
160
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
161
|
+
assertStateUnchanged: assertAddIndexerFailureState,
|
|
162
|
+
strategy: SignatureMutationStrategy.TYPE_MISMATCH,
|
|
163
|
+
expectedLogs: ['Failed to verify message signature.']
|
|
164
|
+
}).performScenario();
|
|
165
|
+
|
|
166
|
+
new IndexerSequenceStateInvalidScenario({
|
|
167
|
+
title: 'State.apply addIndexer rejects payloads when indexer sequence state is invalid',
|
|
168
|
+
setupScenario: setupAddIndexerScenario,
|
|
169
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
170
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
171
|
+
expectedLogs: ['Indexer sequence state is invalid.']
|
|
172
|
+
}).performScenario();
|
|
173
|
+
|
|
174
|
+
new TransactionValidityMismatchScenario({
|
|
175
|
+
title: 'State.apply addIndexer rejects payloads when transaction validity mismatches indexer state',
|
|
176
|
+
setupScenario: setupAddIndexerScenario,
|
|
177
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
178
|
+
assertStateUnchanged: assertAddIndexerFailureState,
|
|
179
|
+
txValidityPath: ['aco', 'txv'],
|
|
180
|
+
rebuildPayloadWithTxValidity: ({ context, mutatedTxValidity }) =>
|
|
181
|
+
buildAddIndexerPayloadWithTxValidity(context, mutatedTxValidity),
|
|
182
|
+
expectedLogs: ['Transaction was not executed.']
|
|
183
|
+
}).performScenario();
|
|
184
|
+
|
|
185
|
+
new OperationAlreadyAppliedScenario({
|
|
186
|
+
title: 'State.apply addIndexer rejects payloads when operation was already applied',
|
|
187
|
+
setupScenario: setupAddIndexerScenario,
|
|
188
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
189
|
+
assertStateUnchanged: (t, context, validPayload) =>
|
|
190
|
+
assertAddIndexerSuccessState(t, context, {
|
|
191
|
+
writerPeer: context.addIndexerScenario?.writerPeer ?? selectIndexerCandidatePeer(context),
|
|
192
|
+
adminPeer: context.adminBootstrap,
|
|
193
|
+
writerEntryBefore: context.addIndexerScenario?.writerEntryBefore,
|
|
194
|
+
adminEntryBefore: context.addIndexerScenario?.adminEntryBefore,
|
|
195
|
+
payload: validPayload
|
|
196
|
+
}),
|
|
197
|
+
expectedLogs: ['Operation has already been applied.']
|
|
198
|
+
}).performScenario();
|
|
199
|
+
|
|
200
|
+
new IndexerNodeEntryMissingScenario({
|
|
201
|
+
title: 'State.apply addIndexer rejects payloads when target indexer entry is missing',
|
|
202
|
+
setupScenario: setupAddIndexerScenario,
|
|
203
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
204
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
205
|
+
selectPeer: context => context.addIndexerScenario?.writerPeer ?? selectIndexerCandidatePeer(context),
|
|
206
|
+
expectedLogs: ['Failed to verify target indexer entry.']
|
|
207
|
+
}).performScenario();
|
|
208
|
+
|
|
209
|
+
new IndexerNodeEntryDecodeFailureScenario({
|
|
210
|
+
title: 'State.apply addIndexer rejects payloads when target indexer entry cannot be decoded',
|
|
211
|
+
setupScenario: setupAddIndexerScenario,
|
|
212
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
213
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
214
|
+
selectPeer: context => context.addIndexerScenario?.writerPeer ?? selectIndexerCandidatePeer(context),
|
|
215
|
+
expectedLogs: ['Failed to decode pretender indexer node entry.']
|
|
216
|
+
}).performScenario();
|
|
217
|
+
|
|
218
|
+
addIndexerPretenderNotWriterScenario();
|
|
219
|
+
addIndexerPretenderAlreadyIndexerScenario();
|
|
220
|
+
|
|
221
|
+
new IndexerRoleUpdateFailureScenario({
|
|
222
|
+
title: 'State.apply addIndexer rejects payloads when node role update fails',
|
|
223
|
+
setupScenario: setupAddIndexerScenario,
|
|
224
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
225
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
226
|
+
applyRoleMutationFailure: applyWithIndexerRoleUpdateFailure,
|
|
227
|
+
expectedLogs: ['Failed to update node role.']
|
|
228
|
+
}).performScenario();
|
|
229
|
+
|
|
230
|
+
addIndexerWriterKeyAlreadyRegisteredScenario();
|
|
231
|
+
|
|
232
|
+
new FeeDecodeFailureScenario({
|
|
233
|
+
title: 'State.apply addIndexer rejects payloads when fee cannot be decoded',
|
|
234
|
+
setupScenario: setupAddIndexerScenario,
|
|
235
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
236
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
237
|
+
selectNode: context => context.adminBootstrap,
|
|
238
|
+
expectedLogs: ['Invalid fee amount.']
|
|
239
|
+
}).performScenario();
|
|
240
|
+
|
|
241
|
+
new RequesterNodeEntryMissingScenario({
|
|
242
|
+
title: 'State.apply addIndexer rejects payloads when requester node entry is missing',
|
|
243
|
+
setupScenario: setupAddIndexerScenario,
|
|
244
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
245
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
246
|
+
selectPeer: context => context.adminBootstrap,
|
|
247
|
+
expectedLogs: ['Invalid requester node entry buffer.']
|
|
248
|
+
}).performScenario();
|
|
249
|
+
|
|
250
|
+
new OperationValidationScenarioBase({
|
|
251
|
+
title: 'State.apply addIndexer rejects payloads when requester node entry cannot be decoded',
|
|
252
|
+
setupScenario: setupAddIndexerScenario,
|
|
253
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
254
|
+
mutatePayload: (_t, payload) => payload,
|
|
255
|
+
applyInvalidPayload: (context, invalidPayload) =>
|
|
256
|
+
applyWithRequesterEntryCorruption(context, invalidPayload, { peer: context.adminBootstrap }),
|
|
257
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
258
|
+
expectedLogs: ['Failed to decode requester node entry.']
|
|
259
|
+
}).performScenario();
|
|
260
|
+
|
|
261
|
+
new RequesterBalanceScenarioBase({
|
|
262
|
+
title: 'State.apply addIndexer rejects payloads when admin balance cannot be decoded',
|
|
263
|
+
setupScenario: setupAddIndexerScenario,
|
|
264
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
265
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
266
|
+
mutateDecodedEntry: decoded => ({ ...decoded, balance: b4a.alloc(1) }),
|
|
267
|
+
selectPeer: context => context.adminBootstrap,
|
|
268
|
+
expectedLogs: ['Invalid admin balance.']
|
|
269
|
+
}).performScenario();
|
|
270
|
+
|
|
271
|
+
new RequesterBalanceInsufficientScenario({
|
|
272
|
+
title: 'State.apply addIndexer rejects payloads when requester balance is insufficient',
|
|
273
|
+
setupScenario: setupAddIndexerScenario,
|
|
274
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
275
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
276
|
+
selectPeer: context => context.adminBootstrap,
|
|
277
|
+
expectedLogs: ['Insufficient requester balance.']
|
|
278
|
+
}).performScenario();
|
|
279
|
+
|
|
280
|
+
new RequesterBalanceFeeApplicationFailureScenario({
|
|
281
|
+
title: 'State.apply addIndexer rejects payloads when requester fee cannot be applied',
|
|
282
|
+
setupScenario: setupAddIndexerScenario,
|
|
283
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
284
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
285
|
+
selectPeer: context => context.adminBootstrap,
|
|
286
|
+
expectedLogs: ['Failed to apply fee to requester balance.']
|
|
287
|
+
}).performScenario();
|
|
288
|
+
|
|
289
|
+
new RequesterBalanceScenarioBase({
|
|
290
|
+
title: 'State.apply addIndexer rejects payloads when requester balance update fails',
|
|
291
|
+
setupScenario: setupAddIndexerScenario,
|
|
292
|
+
buildValidPayload: context => buildAddIndexerPayload(context),
|
|
293
|
+
assertStateUnchanged: (t, context) => assertAddIndexerFailureState(t, context, { skipSync: true }),
|
|
294
|
+
selectPeer: context => context.adminBootstrap,
|
|
295
|
+
failNextBalanceUpdate: true,
|
|
296
|
+
expectedLogs: ['Failed to update requester node.']
|
|
297
|
+
}).performScenario();
|
|
@@ -0,0 +1,41 @@
|
|
|
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 {
|
|
6
|
+
setupAddWriterScenario,
|
|
7
|
+
selectWriterPeer,
|
|
8
|
+
buildAddWriterPayload,
|
|
9
|
+
assertAddWriterSuccessState
|
|
10
|
+
} from './addWriterScenarioHelpers.js';
|
|
11
|
+
|
|
12
|
+
export default function addWriterHappyPathScenario() {
|
|
13
|
+
test('State.apply addWriter promotes whitelisted nodes via validator cosignature - happy path', async t => {
|
|
14
|
+
const context = await setupAddWriterScenario(t);
|
|
15
|
+
const validatorPeer = context.adminBootstrap;
|
|
16
|
+
const writerPeer = selectWriterPeer(context);
|
|
17
|
+
|
|
18
|
+
const adminEntryBefore = await validatorPeer.base.view.get(validatorPeer.wallet.address);
|
|
19
|
+
t.ok(adminEntryBefore, 'admin entry exists before addWriter');
|
|
20
|
+
const decodedAdminBefore = nodeEntryUtils.decode(adminEntryBefore.value);
|
|
21
|
+
t.ok(decodedAdminBefore, 'admin entry decodes before addWriter');
|
|
22
|
+
const adminBalanceBefore = b4a.from(decodedAdminBefore.balance);
|
|
23
|
+
|
|
24
|
+
const payload = await buildAddWriterPayload(context, {
|
|
25
|
+
readerPeer: writerPeer,
|
|
26
|
+
validatorPeer
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
await validatorPeer.base.append(payload);
|
|
30
|
+
await validatorPeer.base.update();
|
|
31
|
+
await eventFlush();
|
|
32
|
+
|
|
33
|
+
await assertAddWriterSuccessState(t, context, {
|
|
34
|
+
readerPeer: writerPeer,
|
|
35
|
+
validatorPeer,
|
|
36
|
+
writerInitialBalance: context.addWriterScenario?.writerInitialBalance,
|
|
37
|
+
validatorBalanceBefore: adminBalanceBefore,
|
|
38
|
+
payload
|
|
39
|
+
});
|
|
40
|
+
});
|
|
41
|
+
}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import b4a from 'b4a';
|
|
2
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
3
|
+
import { safeDecodeApplyOperation, safeEncodeApplyOperation } from '../../../../../src/utils/protobuf/operationHelpers.js';
|
|
4
|
+
import {
|
|
5
|
+
setupAddWriterScenario,
|
|
6
|
+
buildAddWriterPayload,
|
|
7
|
+
assertAddWriterFailureState,
|
|
8
|
+
applyWithRoleAccessBypass
|
|
9
|
+
} from './addWriterScenarioHelpers.js';
|
|
10
|
+
|
|
11
|
+
function mutateValidatorSignature(t, validPayload) {
|
|
12
|
+
const decoded = safeDecodeApplyOperation(validPayload);
|
|
13
|
+
t.ok(decoded, 'fixtures decode');
|
|
14
|
+
const parent = decoded.rao;
|
|
15
|
+
if (!parent?.vs) return validPayload;
|
|
16
|
+
const mutated = b4a.from(parent.vs);
|
|
17
|
+
mutated[mutated.length - 1] ^= 0xff;
|
|
18
|
+
parent.vs = mutated;
|
|
19
|
+
return safeEncodeApplyOperation(decoded);
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export default function addWriterInvalidValidatorSignatureScenario() {
|
|
23
|
+
new OperationValidationScenarioBase({
|
|
24
|
+
title: 'State.apply addWriter validator signature is invalid',
|
|
25
|
+
setupScenario: setupAddWriterScenario,
|
|
26
|
+
buildValidPayload: buildAddWriterPayload,
|
|
27
|
+
mutatePayload: mutateValidatorSignature,
|
|
28
|
+
applyInvalidPayload: applyWithRoleAccessBypass,
|
|
29
|
+
assertStateUnchanged: assertAddWriterFailureState,
|
|
30
|
+
expectedLogs: ['Failed to verify validator message signature.']
|
|
31
|
+
}).performScenario();
|
|
32
|
+
}
|
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
import { test } from 'brittle';
|
|
2
|
+
import b4a from 'b4a';
|
|
3
|
+
import { randomBytes } from 'crypto';
|
|
4
|
+
import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
|
|
5
|
+
import nodeEntryUtils from '../../../../../src/core/state/utils/nodeEntry.js';
|
|
6
|
+
import {
|
|
7
|
+
setupAddWriterScenario,
|
|
8
|
+
selectWriterPeer,
|
|
9
|
+
buildAddWriterPayload,
|
|
10
|
+
buildRemoveWriterPayload,
|
|
11
|
+
assertAddWriterSuccessState,
|
|
12
|
+
assertWriterRemovalState
|
|
13
|
+
} from './addWriterScenarioHelpers.js';
|
|
14
|
+
import {
|
|
15
|
+
toBalance,
|
|
16
|
+
BALANCE_FEE,
|
|
17
|
+
BALANCE_TO_STAKE,
|
|
18
|
+
PERCENT_75
|
|
19
|
+
} from '../../../../../src/core/state/utils/balance.js';
|
|
20
|
+
|
|
21
|
+
export default function addWriterNewWkScenario() {
|
|
22
|
+
test('State.apply addWriter rotates writing keys after removing/re-adding writer role', async t => {
|
|
23
|
+
const context = await setupAddWriterScenario(t);
|
|
24
|
+
const validatorPeer = context.adminBootstrap;
|
|
25
|
+
const writerPeer = selectWriterPeer(context);
|
|
26
|
+
const originalWriterKey = b4a.from(writerPeer.base.local.key);
|
|
27
|
+
const rewardPerOperation = BALANCE_FEE.percentage(PERCENT_75);
|
|
28
|
+
|
|
29
|
+
const writerInitialBalance = toBalance(context.addWriterScenario.writerInitialBalance);
|
|
30
|
+
let validatorBalanceTracker = await readNodeBalanceBuffer(validatorPeer);
|
|
31
|
+
|
|
32
|
+
const firstAddPayload = await buildAddWriterPayload(context, {
|
|
33
|
+
readerPeer: writerPeer,
|
|
34
|
+
validatorPeer
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
await validatorPeer.base.append(firstAddPayload);
|
|
38
|
+
await validatorPeer.base.update();
|
|
39
|
+
await eventFlush();
|
|
40
|
+
|
|
41
|
+
await assertAddWriterSuccessState(t, context, {
|
|
42
|
+
readerPeer: writerPeer,
|
|
43
|
+
validatorPeer,
|
|
44
|
+
writerInitialBalance: context.addWriterScenario.writerInitialBalance,
|
|
45
|
+
validatorBalanceBefore: validatorBalanceTracker,
|
|
46
|
+
payload: firstAddPayload,
|
|
47
|
+
writerKeyBuffer: originalWriterKey,
|
|
48
|
+
expectedWriterIndex: 1
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
validatorBalanceTracker = advanceBalanceTracker(validatorBalanceTracker, rewardPerOperation);
|
|
52
|
+
|
|
53
|
+
const balanceAfterInitialAdd = writerInitialBalance
|
|
54
|
+
.sub(BALANCE_FEE)
|
|
55
|
+
?.sub(BALANCE_TO_STAKE);
|
|
56
|
+
if (!balanceAfterInitialAdd) {
|
|
57
|
+
throw new Error('Failed to derive writer balance after initial add.');
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
const removeWriterPayload = await buildRemoveWriterPayload(context, {
|
|
61
|
+
readerPeer: writerPeer,
|
|
62
|
+
validatorPeer,
|
|
63
|
+
writerKeyBuffer: originalWriterKey
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
await validatorPeer.base.append(removeWriterPayload);
|
|
67
|
+
await validatorPeer.base.update();
|
|
68
|
+
await eventFlush();
|
|
69
|
+
|
|
70
|
+
const writerBalanceAfterRemoval = balanceAfterInitialAdd
|
|
71
|
+
.add(BALANCE_TO_STAKE)
|
|
72
|
+
?.sub(BALANCE_FEE);
|
|
73
|
+
if (!writerBalanceAfterRemoval) {
|
|
74
|
+
throw new Error('Failed to derive writer balance after removal.');
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
await assertWriterRemovalState(t, context, {
|
|
78
|
+
readerPeer: writerPeer,
|
|
79
|
+
validatorPeer,
|
|
80
|
+
writerKeyBuffer: originalWriterKey,
|
|
81
|
+
expectedBalanceBuffer: b4a.from(writerBalanceAfterRemoval.value),
|
|
82
|
+
payload: removeWriterPayload
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
validatorBalanceTracker = advanceBalanceTracker(validatorBalanceTracker, rewardPerOperation);
|
|
86
|
+
|
|
87
|
+
const rotatedWriterKey = randomBytes(32);
|
|
88
|
+
const writerBalanceBeforeReAdd = b4a.from(writerBalanceAfterRemoval.value);
|
|
89
|
+
|
|
90
|
+
const secondAddPayload = await buildAddWriterPayload(context, {
|
|
91
|
+
readerPeer: writerPeer,
|
|
92
|
+
validatorPeer,
|
|
93
|
+
writerKeyBuffer: rotatedWriterKey
|
|
94
|
+
});
|
|
95
|
+
|
|
96
|
+
await validatorPeer.base.append(secondAddPayload);
|
|
97
|
+
await validatorPeer.base.update();
|
|
98
|
+
await eventFlush();
|
|
99
|
+
|
|
100
|
+
await assertAddWriterSuccessState(t, context, {
|
|
101
|
+
readerPeer: writerPeer,
|
|
102
|
+
validatorPeer,
|
|
103
|
+
writerInitialBalance: writerBalanceBeforeReAdd,
|
|
104
|
+
validatorBalanceBefore: validatorBalanceTracker,
|
|
105
|
+
payload: secondAddPayload,
|
|
106
|
+
expectedWriterIndex: 2,
|
|
107
|
+
writerKeyBuffer: rotatedWriterKey
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
validatorBalanceTracker = advanceBalanceTracker(validatorBalanceTracker, rewardPerOperation);
|
|
111
|
+
|
|
112
|
+
const finalAdminBalance = await readNodeBalanceBuffer(validatorPeer);
|
|
113
|
+
t.ok(
|
|
114
|
+
b4a.equals(finalAdminBalance, validatorBalanceTracker),
|
|
115
|
+
'validator accumulated rewards from two addWriter operations and one removeWriter'
|
|
116
|
+
);
|
|
117
|
+
|
|
118
|
+
const writerBalanceAfterReAdd = writerBalanceAfterRemoval
|
|
119
|
+
.sub(BALANCE_FEE)
|
|
120
|
+
?.sub(BALANCE_TO_STAKE);
|
|
121
|
+
if (!writerBalanceAfterReAdd) {
|
|
122
|
+
throw new Error('Failed to compute writer balance after re-add.');
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
const writerEntry = await writerPeer.base.view.get(writerPeer.wallet.address);
|
|
126
|
+
const decodedWriter = nodeEntryUtils.decode(writerEntry.value);
|
|
127
|
+
const expectedWriterBalance = writerBalanceAfterReAdd.value;
|
|
128
|
+
t.ok(
|
|
129
|
+
b4a.equals(decodedWriter.balance, expectedWriterBalance),
|
|
130
|
+
'writer ends with balance reduced by three fees and new stake'
|
|
131
|
+
);
|
|
132
|
+
t.ok(
|
|
133
|
+
b4a.equals(decodedWriter.wk, rotatedWriterKey),
|
|
134
|
+
'writer entry reflects the rotated writing key'
|
|
135
|
+
);
|
|
136
|
+
});
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
function advanceBalanceTracker(currentBuffer, incrementBalance) {
|
|
140
|
+
const current = toBalance(currentBuffer);
|
|
141
|
+
const next = current.add(incrementBalance);
|
|
142
|
+
return b4a.from(next.value);
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
async function readNodeBalanceBuffer(peer) {
|
|
146
|
+
const entry = await peer.base.view.get(peer.wallet.address);
|
|
147
|
+
const decoded = nodeEntryUtils.decode(entry.value);
|
|
148
|
+
return b4a.from(decoded.balance);
|
|
149
|
+
}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
2
|
+
import {
|
|
3
|
+
setupAddWriterScenario,
|
|
4
|
+
buildAddWriterPayload,
|
|
5
|
+
assertAddWriterFailureState,
|
|
6
|
+
applyWithRequesterRoleOverride
|
|
7
|
+
} from './addWriterScenarioHelpers.js';
|
|
8
|
+
import nodeRoleUtils from '../../../../../src/core/state/utils/roles.js';
|
|
9
|
+
|
|
10
|
+
export default function addWriterRequesterAlreadyWriterScenario() {
|
|
11
|
+
new OperationValidationScenarioBase({
|
|
12
|
+
title: 'State.apply addWriter rejects payloads when requester is already a writer',
|
|
13
|
+
setupScenario: setupAddWriterScenario,
|
|
14
|
+
buildValidPayload: buildAddWriterPayload,
|
|
15
|
+
mutatePayload: (_t, payload) => payload,
|
|
16
|
+
applyInvalidPayload: (context, invalidPayload) =>
|
|
17
|
+
applyWithRequesterRoleOverride(context, invalidPayload, nodeRoleUtils.NodeRole.WRITER),
|
|
18
|
+
assertStateUnchanged: (t, context) => assertAddWriterFailureState(t, context, { skipSync: true }),
|
|
19
|
+
expectedLogs: ['Node must be whitelisted, and cannot be a writer or an indexer.']
|
|
20
|
+
}).performScenario();
|
|
21
|
+
}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import b4a from 'b4a';
|
|
2
|
+
import RequesterBalanceScenarioBase from '../common/balances/base/requesterBalanceScenarioBase.js';
|
|
3
|
+
import {
|
|
4
|
+
setupAddWriterScenario,
|
|
5
|
+
buildAddWriterPayload,
|
|
6
|
+
assertAddWriterFailureState,
|
|
7
|
+
selectWriterPeer
|
|
8
|
+
} from './addWriterScenarioHelpers.js';
|
|
9
|
+
|
|
10
|
+
export default function addWriterRequesterBalanceInsufficientScenario() {
|
|
11
|
+
new RequesterBalanceScenarioBase({
|
|
12
|
+
title: 'State.apply addWriter rejects payloads when requester balance is insufficient',
|
|
13
|
+
setupScenario: setupAddWriterScenario,
|
|
14
|
+
buildValidPayload: buildAddWriterPayload,
|
|
15
|
+
assertStateUnchanged: (t, context) => assertAddWriterFailureState(t, context, { skipSync: true }),
|
|
16
|
+
selectPeer: selectWriterPeer,
|
|
17
|
+
mutateDecodedEntry: decoded =>
|
|
18
|
+
decoded ? { ...decoded, balance: b4a.alloc(decoded.balance.length) } : decoded,
|
|
19
|
+
expectedLogs: ['Insufficient requester balance.', 'Add writer operation ignored.']
|
|
20
|
+
}).performScenario();
|
|
21
|
+
}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
2
|
+
import {
|
|
3
|
+
setupAddWriterScenario,
|
|
4
|
+
buildAddWriterPayload,
|
|
5
|
+
assertAddWriterFailureState,
|
|
6
|
+
applyWithRequesterEntryCorruption
|
|
7
|
+
} from './addWriterScenarioHelpers.js';
|
|
8
|
+
|
|
9
|
+
export default function addWriterRequesterEntryDecodeFailureScenario() {
|
|
10
|
+
new OperationValidationScenarioBase({
|
|
11
|
+
title: 'State.apply addWriter rejects payloads when requester entry cannot be decoded',
|
|
12
|
+
setupScenario: setupAddWriterScenario,
|
|
13
|
+
buildValidPayload: buildAddWriterPayload,
|
|
14
|
+
mutatePayload: (_t, payload) => payload,
|
|
15
|
+
applyInvalidPayload: applyWithRequesterEntryCorruption,
|
|
16
|
+
assertStateUnchanged: (t, context) => assertAddWriterFailureState(t, context, { skipSync: true }),
|
|
17
|
+
expectedLogs: ['Failed to decode node entry.']
|
|
18
|
+
}).performScenario();
|
|
19
|
+
}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
2
|
+
import {
|
|
3
|
+
setupAddWriterScenario,
|
|
4
|
+
buildAddWriterPayload,
|
|
5
|
+
assertAddWriterFailureState,
|
|
6
|
+
applyWithRequesterEntryRemoval
|
|
7
|
+
} from './addWriterScenarioHelpers.js';
|
|
8
|
+
|
|
9
|
+
export default function addWriterRequesterEntryMissingScenario() {
|
|
10
|
+
new OperationValidationScenarioBase({
|
|
11
|
+
title: 'State.apply addWriter rejects payloads when requester entry is missing',
|
|
12
|
+
setupScenario: setupAddWriterScenario,
|
|
13
|
+
buildValidPayload: buildAddWriterPayload,
|
|
14
|
+
mutatePayload: (_t, payload) => payload,
|
|
15
|
+
applyInvalidPayload: applyWithRequesterEntryRemoval,
|
|
16
|
+
assertStateUnchanged: (t, context) => assertAddWriterFailureState(t, context, { skipSync: true }),
|
|
17
|
+
expectedLogs: ['Failed to verify requester node address.', 'Failed to add writer.']
|
|
18
|
+
}).performScenario();
|
|
19
|
+
}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
2
|
+
import {
|
|
3
|
+
setupAddWriterScenario,
|
|
4
|
+
buildAddWriterPayload,
|
|
5
|
+
assertAddWriterFailureState,
|
|
6
|
+
applyWithRequesterRoleOverride
|
|
7
|
+
} from './addWriterScenarioHelpers.js';
|
|
8
|
+
import nodeRoleUtils from '../../../../../src/core/state/utils/roles.js';
|
|
9
|
+
|
|
10
|
+
export default function addWriterRequesterIndexerScenario() {
|
|
11
|
+
new OperationValidationScenarioBase({
|
|
12
|
+
title: 'State.apply addWriter rejects payloads when requester is already an indexer',
|
|
13
|
+
setupScenario: setupAddWriterScenario,
|
|
14
|
+
buildValidPayload: buildAddWriterPayload,
|
|
15
|
+
mutatePayload: (_t, payload) => payload,
|
|
16
|
+
applyInvalidPayload: (context, invalidPayload) =>
|
|
17
|
+
applyWithRequesterRoleOverride(context, invalidPayload, nodeRoleUtils.NodeRole.INDEXER),
|
|
18
|
+
assertStateUnchanged: (t, context) => assertAddWriterFailureState(t, context, { skipSync: true }),
|
|
19
|
+
expectedLogs: ['Node must be whitelisted, and cannot be a writer or an indexer.']
|
|
20
|
+
}).performScenario();
|
|
21
|
+
}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
2
|
+
import {
|
|
3
|
+
setupAddWriterScenario,
|
|
4
|
+
buildAddWriterPayload,
|
|
5
|
+
assertAddWriterFailureState,
|
|
6
|
+
applyWithRequesterRoleOverride
|
|
7
|
+
} from './addWriterScenarioHelpers.js';
|
|
8
|
+
import nodeRoleUtils from '../../../../../src/core/state/utils/roles.js';
|
|
9
|
+
|
|
10
|
+
export default function addWriterRequesterNotWhitelistedScenario() {
|
|
11
|
+
new OperationValidationScenarioBase({
|
|
12
|
+
title: 'State.apply addWriter rejects payloads when requester is not whitelisted',
|
|
13
|
+
setupScenario: setupAddWriterScenario,
|
|
14
|
+
buildValidPayload: buildAddWriterPayload,
|
|
15
|
+
mutatePayload: (_t, payload) => payload,
|
|
16
|
+
applyInvalidPayload: (context, invalidPayload) =>
|
|
17
|
+
applyWithRequesterRoleOverride(context, invalidPayload, nodeRoleUtils.NodeRole.READER),
|
|
18
|
+
assertStateUnchanged: (t, context) => assertAddWriterFailureState(t, context, { skipSync: true }),
|
|
19
|
+
expectedLogs: ['Node must be whitelisted, and cannot be a writer or an indexer.']
|
|
20
|
+
}).performScenario();
|
|
21
|
+
}
|