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,50 @@
|
|
|
1
|
+
import { BALANCE_ZERO } from '../../../../../src/core/state/utils/balance.js';
|
|
2
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
3
|
+
import {
|
|
4
|
+
setupAddWriterScenario,
|
|
5
|
+
buildAddWriterPayload,
|
|
6
|
+
assertAddWriterFailureState,
|
|
7
|
+
applyWithStakeEntryMutation
|
|
8
|
+
} from './addWriterScenarioHelpers.js';
|
|
9
|
+
|
|
10
|
+
const STAKE_BALANCE_MARK = Symbol('stake-balance-update-target');
|
|
11
|
+
|
|
12
|
+
export default function addWriterStakeBalanceUpdateFailureScenario() {
|
|
13
|
+
new OperationValidationScenarioBase({
|
|
14
|
+
title: 'State.apply addWriter rejects payloads when stake balance update fails',
|
|
15
|
+
setupScenario: setupAddWriterScenario,
|
|
16
|
+
buildValidPayload: buildAddWriterPayload,
|
|
17
|
+
mutatePayload: (_t, payload) => payload,
|
|
18
|
+
applyInvalidPayload: async (context, invalidPayload) => {
|
|
19
|
+
const balancePrototype = Object.getPrototypeOf(BALANCE_ZERO);
|
|
20
|
+
const originalUpdate = balancePrototype.update;
|
|
21
|
+
let shouldFail = true;
|
|
22
|
+
|
|
23
|
+
balancePrototype.update = function patchedUpdate(nodeEntryBuffer) {
|
|
24
|
+
if (shouldFail && nodeEntryBuffer?.[STAKE_BALANCE_MARK]) {
|
|
25
|
+
shouldFail = false;
|
|
26
|
+
delete nodeEntryBuffer[STAKE_BALANCE_MARK];
|
|
27
|
+
return null;
|
|
28
|
+
}
|
|
29
|
+
return originalUpdate.call(this, nodeEntryBuffer);
|
|
30
|
+
};
|
|
31
|
+
|
|
32
|
+
try {
|
|
33
|
+
await applyWithStakeEntryMutation(
|
|
34
|
+
context,
|
|
35
|
+
invalidPayload,
|
|
36
|
+
nodeEntryBuffer => {
|
|
37
|
+
if (nodeEntryBuffer && typeof nodeEntryBuffer === 'object') {
|
|
38
|
+
nodeEntryBuffer[STAKE_BALANCE_MARK] = true;
|
|
39
|
+
}
|
|
40
|
+
return nodeEntryBuffer;
|
|
41
|
+
}
|
|
42
|
+
);
|
|
43
|
+
} finally {
|
|
44
|
+
balancePrototype.update = originalUpdate;
|
|
45
|
+
}
|
|
46
|
+
},
|
|
47
|
+
assertStateUnchanged: (t, context) => assertAddWriterFailureState(t, context, { skipSync: true }),
|
|
48
|
+
expectedLogs: ['Failed to update node entry with new balance']
|
|
49
|
+
}).performScenario();
|
|
50
|
+
}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import { BALANCE_ZERO } from '../../../../../src/core/state/utils/balance.js';
|
|
2
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
3
|
+
import {
|
|
4
|
+
setupAddWriterScenario,
|
|
5
|
+
buildAddWriterPayload,
|
|
6
|
+
assertAddWriterFailureState,
|
|
7
|
+
applyWithStakeEntryMutation
|
|
8
|
+
} from './addWriterScenarioHelpers.js';
|
|
9
|
+
|
|
10
|
+
export default function addWriterStakeInsufficientBalanceScenario() {
|
|
11
|
+
new OperationValidationScenarioBase({
|
|
12
|
+
title: 'State.apply addWriter rejects payloads when stake balance is insufficient',
|
|
13
|
+
setupScenario: setupAddWriterScenario,
|
|
14
|
+
buildValidPayload: buildAddWriterPayload,
|
|
15
|
+
mutatePayload: (_t, payload) => payload,
|
|
16
|
+
applyInvalidPayload: (context, invalidPayload) =>
|
|
17
|
+
applyWithStakeEntryMutation(
|
|
18
|
+
context,
|
|
19
|
+
invalidPayload,
|
|
20
|
+
nodeEntryBuffer => nodeEntryBuffer,
|
|
21
|
+
decodedEntry => {
|
|
22
|
+
if (!decodedEntry) return decodedEntry;
|
|
23
|
+
return { ...decodedEntry, balance: BALANCE_ZERO.value };
|
|
24
|
+
}
|
|
25
|
+
),
|
|
26
|
+
assertStateUnchanged: (t, context) => assertAddWriterFailureState(t, context, { skipSync: true }),
|
|
27
|
+
expectedLogs: ['Insufficient balance to stake']
|
|
28
|
+
}).performScenario();
|
|
29
|
+
}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import b4a from 'b4a';
|
|
2
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
3
|
+
import {
|
|
4
|
+
setupAddWriterScenario,
|
|
5
|
+
buildAddWriterPayload,
|
|
6
|
+
assertAddWriterFailureState,
|
|
7
|
+
applyWithStakeEntryMutation
|
|
8
|
+
} from './addWriterScenarioHelpers.js';
|
|
9
|
+
|
|
10
|
+
export default function addWriterStakeInvalidBalanceScenario() {
|
|
11
|
+
new OperationValidationScenarioBase({
|
|
12
|
+
title: 'State.apply addWriter rejects payloads when stake balance is invalid',
|
|
13
|
+
setupScenario: setupAddWriterScenario,
|
|
14
|
+
buildValidPayload: buildAddWriterPayload,
|
|
15
|
+
mutatePayload: (_t, payload) => payload,
|
|
16
|
+
applyInvalidPayload: (context, invalidPayload) =>
|
|
17
|
+
applyWithStakeEntryMutation(
|
|
18
|
+
context,
|
|
19
|
+
invalidPayload,
|
|
20
|
+
nodeEntryBuffer => nodeEntryBuffer,
|
|
21
|
+
decodedEntry => {
|
|
22
|
+
if (!decodedEntry) return decodedEntry;
|
|
23
|
+
return { ...decodedEntry, balance: b4a.alloc(1) };
|
|
24
|
+
}
|
|
25
|
+
),
|
|
26
|
+
assertStateUnchanged: (t, context) => assertAddWriterFailureState(t, context, { skipSync: true }),
|
|
27
|
+
expectedLogs: ['Invalid node balance']
|
|
28
|
+
}).performScenario();
|
|
29
|
+
}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import b4a from 'b4a';
|
|
2
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
3
|
+
import {
|
|
4
|
+
setupAddWriterScenario,
|
|
5
|
+
buildAddWriterPayload,
|
|
6
|
+
assertAddWriterFailureState,
|
|
7
|
+
applyWithStakeEntryMutation
|
|
8
|
+
} from './addWriterScenarioHelpers.js';
|
|
9
|
+
|
|
10
|
+
export default function addWriterStakeInvalidEntryScenario() {
|
|
11
|
+
new OperationValidationScenarioBase({
|
|
12
|
+
title: 'State.apply addWriter rejects payloads when stake entry buffer is invalid',
|
|
13
|
+
setupScenario: setupAddWriterScenario,
|
|
14
|
+
buildValidPayload: buildAddWriterPayload,
|
|
15
|
+
mutatePayload: (_t, payload) => payload,
|
|
16
|
+
applyInvalidPayload: (context, invalidPayload) =>
|
|
17
|
+
applyWithStakeEntryMutation(context, invalidPayload, () => b4a.alloc(1)),
|
|
18
|
+
assertStateUnchanged: (t, context) => assertAddWriterFailureState(t, context, { skipSync: true }),
|
|
19
|
+
expectedLogs: ['Invalid node entry buffer']
|
|
20
|
+
}).performScenario();
|
|
21
|
+
}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
2
|
+
import nodeEntryUtils from '../../../../../src/core/state/utils/nodeEntry.js';
|
|
3
|
+
import {
|
|
4
|
+
setupAddWriterScenario,
|
|
5
|
+
buildAddWriterPayload,
|
|
6
|
+
assertAddWriterFailureState,
|
|
7
|
+
applyWithStakeEntryMutation
|
|
8
|
+
} from './addWriterScenarioHelpers.js';
|
|
9
|
+
|
|
10
|
+
export default function addWriterStakeStakedBalanceFailureScenario() {
|
|
11
|
+
new OperationValidationScenarioBase({
|
|
12
|
+
title: 'State.apply addWriter rejects payloads when setting staked balance fails',
|
|
13
|
+
setupScenario: setupAddWriterScenario,
|
|
14
|
+
buildValidPayload: buildAddWriterPayload,
|
|
15
|
+
mutatePayload: (_t, payload) => payload,
|
|
16
|
+
applyInvalidPayload: async (context, invalidPayload) => {
|
|
17
|
+
const originalSetStakedBalance = nodeEntryUtils.setStakedBalance;
|
|
18
|
+
let shouldFail = true;
|
|
19
|
+
|
|
20
|
+
nodeEntryUtils.setStakedBalance = function patchedSetStakedBalance(nodeEntryBuffer, stakedBalance) {
|
|
21
|
+
if (shouldFail) {
|
|
22
|
+
shouldFail = false;
|
|
23
|
+
return null;
|
|
24
|
+
}
|
|
25
|
+
return originalSetStakedBalance(nodeEntryBuffer, stakedBalance);
|
|
26
|
+
};
|
|
27
|
+
|
|
28
|
+
try {
|
|
29
|
+
await applyWithStakeEntryMutation(context, invalidPayload, nodeEntryBuffer => nodeEntryBuffer);
|
|
30
|
+
} finally {
|
|
31
|
+
nodeEntryUtils.setStakedBalance = originalSetStakedBalance;
|
|
32
|
+
}
|
|
33
|
+
},
|
|
34
|
+
assertStateUnchanged: (t, context) => assertAddWriterFailureState(t, context, { skipSync: true }),
|
|
35
|
+
expectedLogs: ['Failed to set staked balance in node entry']
|
|
36
|
+
}).performScenario();
|
|
37
|
+
}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { BALANCE_ZERO, BALANCE_TO_STAKE } from '../../../../../src/core/state/utils/balance.js';
|
|
2
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
3
|
+
import {
|
|
4
|
+
setupAddWriterScenario,
|
|
5
|
+
buildAddWriterPayload,
|
|
6
|
+
assertAddWriterFailureState,
|
|
7
|
+
applyWithStakeEntryMutation
|
|
8
|
+
} from './addWriterScenarioHelpers.js';
|
|
9
|
+
|
|
10
|
+
export default function addWriterStakeSubtractFailureScenario() {
|
|
11
|
+
new OperationValidationScenarioBase({
|
|
12
|
+
title: 'State.apply addWriter rejects payloads when stake balance subtraction fails',
|
|
13
|
+
setupScenario: setupAddWriterScenario,
|
|
14
|
+
buildValidPayload: buildAddWriterPayload,
|
|
15
|
+
mutatePayload: (_t, payload) => payload,
|
|
16
|
+
applyInvalidPayload: async (context, invalidPayload) => {
|
|
17
|
+
const balancePrototype = Object.getPrototypeOf(BALANCE_ZERO);
|
|
18
|
+
const originalSub = balancePrototype.sub;
|
|
19
|
+
let shouldFailSubtraction = true;
|
|
20
|
+
|
|
21
|
+
balancePrototype.sub = function patchedSub(balance) {
|
|
22
|
+
if (shouldFailSubtraction && balance === BALANCE_TO_STAKE) {
|
|
23
|
+
shouldFailSubtraction = false;
|
|
24
|
+
return null;
|
|
25
|
+
}
|
|
26
|
+
return originalSub.call(this, balance);
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
try {
|
|
30
|
+
await applyWithStakeEntryMutation(
|
|
31
|
+
context,
|
|
32
|
+
invalidPayload,
|
|
33
|
+
nodeEntryBuffer => nodeEntryBuffer
|
|
34
|
+
);
|
|
35
|
+
} finally {
|
|
36
|
+
balancePrototype.sub = originalSub;
|
|
37
|
+
}
|
|
38
|
+
},
|
|
39
|
+
assertStateUnchanged: (t, context) => assertAddWriterFailureState(t, context, { skipSync: true }),
|
|
40
|
+
expectedLogs: ['Failed to subtract stake balance']
|
|
41
|
+
}).performScenario();
|
|
42
|
+
}
|
|
@@ -0,0 +1,105 @@
|
|
|
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 './addWriterScenarioHelpers.js';
|
|
11
|
+
import { initializeBalances, whitelistAddress } from '../common/commonScenarioHelper.js';
|
|
12
|
+
import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
|
|
13
|
+
import { safeDecodeApplyOperation } from '../../../../../src/utils/protobuf/operationHelpers.js';
|
|
14
|
+
import addressUtils from '../../../../../src/core/state/utils/address.js';
|
|
15
|
+
|
|
16
|
+
export default function addWriterValidatorRewardScenario() {
|
|
17
|
+
test(
|
|
18
|
+
'State.apply addWriter rewards validator writers that process peer promotions',
|
|
19
|
+
async t => {
|
|
20
|
+
const context = await setupAddWriterScenario(t, { nodes: 3 });
|
|
21
|
+
const adminPeer = context.adminBootstrap;
|
|
22
|
+
const firstWriterPeer = selectWriterPeer(context, 0);
|
|
23
|
+
const secondReaderPeer = selectWriterPeer(context, 1);
|
|
24
|
+
if (!secondReaderPeer) {
|
|
25
|
+
t.fail('Validator reward scenario requires a second reader peer.');
|
|
26
|
+
return;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
await fundAndWhitelistPeer(context, secondReaderPeer);
|
|
30
|
+
|
|
31
|
+
const adminBalanceBefore = await readNodeBalanceBuffer(adminPeer);
|
|
32
|
+
const firstAddPayload = await buildAddWriterPayload(context, {
|
|
33
|
+
readerPeer: firstWriterPeer,
|
|
34
|
+
validatorPeer: adminPeer
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
await appendPayload(adminPeer.base, firstAddPayload);
|
|
38
|
+
|
|
39
|
+
await assertAddWriterSuccessState(t, context, {
|
|
40
|
+
readerPeer: firstWriterPeer,
|
|
41
|
+
validatorPeer: adminPeer,
|
|
42
|
+
writerInitialBalance: context.addWriterScenario?.writerInitialBalance,
|
|
43
|
+
validatorBalanceBefore: adminBalanceBefore,
|
|
44
|
+
payload: firstAddPayload,
|
|
45
|
+
expectedWriterIndex: 1
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
const writerBalanceBeforeReward = await readNodeBalanceBuffer(firstWriterPeer);
|
|
49
|
+
const secondAddPayload = await buildAddWriterPayload(context, {
|
|
50
|
+
readerPeer: secondReaderPeer,
|
|
51
|
+
validatorPeer: firstWriterPeer
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
assertPayloadValidator(t, secondAddPayload, firstWriterPeer.wallet.address);
|
|
55
|
+
|
|
56
|
+
await appendPayload(firstWriterPeer.base, secondAddPayload);
|
|
57
|
+
|
|
58
|
+
await assertAddWriterSuccessState(t, context, {
|
|
59
|
+
readerPeer: secondReaderPeer,
|
|
60
|
+
validatorPeer: firstWriterPeer,
|
|
61
|
+
writerInitialBalance:
|
|
62
|
+
context.addWriterScenario?.writerInitialBalance ?? defaultWriterFunding,
|
|
63
|
+
validatorBalanceBefore: writerBalanceBeforeReward,
|
|
64
|
+
payload: secondAddPayload,
|
|
65
|
+
expectedWriterIndex: 2
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
);
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
async function fundAndWhitelistPeer(context, peer) {
|
|
72
|
+
const funding = context.addWriterScenario?.writerInitialBalance ?? defaultWriterFunding;
|
|
73
|
+
await initializeBalances(context, [[peer.wallet.address, funding]]);
|
|
74
|
+
await whitelistAddress(context, peer.wallet.address);
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
async function appendPayload(nodeBase, payload) {
|
|
78
|
+
await nodeBase.append(payload);
|
|
79
|
+
await nodeBase.update();
|
|
80
|
+
await eventFlush();
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
async function readNodeBalanceBuffer(peer) {
|
|
84
|
+
const entry = await peer.base.view.get(peer.wallet.address);
|
|
85
|
+
if (!entry?.value) {
|
|
86
|
+
throw new Error('Unable to read node balance for validator reward scenario.');
|
|
87
|
+
}
|
|
88
|
+
const decoded = nodeEntryUtils.decode(entry.value);
|
|
89
|
+
if (!decoded?.balance) {
|
|
90
|
+
throw new Error('Validator entry decode failed while tracking rewards.');
|
|
91
|
+
}
|
|
92
|
+
return b4a.from(decoded.balance);
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
function assertPayloadValidator(t, payload, validatorAddress) {
|
|
96
|
+
const decoded = safeDecodeApplyOperation(payload);
|
|
97
|
+
t.ok(decoded, 'validator reward payload decodes');
|
|
98
|
+
const validatorBuffer = decoded?.rao?.va;
|
|
99
|
+
t.ok(validatorBuffer, 'payload carries validator address');
|
|
100
|
+
const expected = addressUtils.addressToBuffer(validatorAddress);
|
|
101
|
+
t.ok(
|
|
102
|
+
b4a.equals(validatorBuffer, expected),
|
|
103
|
+
'payload validator address matches processing writer'
|
|
104
|
+
);
|
|
105
|
+
}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
2
|
+
import {
|
|
3
|
+
setupAddWriterScenario,
|
|
4
|
+
buildAddWriterPayload,
|
|
5
|
+
buildRemoveWriterPayload,
|
|
6
|
+
assertAddWriterFailureState,
|
|
7
|
+
selectWriterPeer,
|
|
8
|
+
applyWithRequesterWriterKeyMismatch
|
|
9
|
+
} from './addWriterScenarioHelpers.js';
|
|
10
|
+
import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
|
|
11
|
+
|
|
12
|
+
async function setupScenarioWithRegisteredKey(t) {
|
|
13
|
+
const context = await setupAddWriterScenario(t);
|
|
14
|
+
const writerPeer = selectWriterPeer(context);
|
|
15
|
+
await promotePeerToWriter(context, writerPeer);
|
|
16
|
+
context.registeredWriterKey = writerPeer.base.local.key;
|
|
17
|
+
await demoteWriterPeer(context, writerPeer);
|
|
18
|
+
return context;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
async function promotePeerToWriter(context, peer) {
|
|
22
|
+
const payload = await buildAddWriterPayload(context, { readerPeer: peer });
|
|
23
|
+
await appendPayload(context, payload);
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
async function demoteWriterPeer(context, peer) {
|
|
27
|
+
const payload = await buildRemoveWriterPayload(context, { readerPeer: peer });
|
|
28
|
+
await appendPayload(context, payload);
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
async function appendPayload(context, payload) {
|
|
32
|
+
const node = context.adminBootstrap ?? context.bootstrap;
|
|
33
|
+
if (!node?.base) {
|
|
34
|
+
throw new Error('Writer key mismatch scenario requires an admin node with a writable base.');
|
|
35
|
+
}
|
|
36
|
+
await node.base.append(payload);
|
|
37
|
+
await node.base.update();
|
|
38
|
+
await eventFlush();
|
|
39
|
+
await context.sync();
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
export default function addWriterWriterKeyMismatchScenario() {
|
|
43
|
+
new OperationValidationScenarioBase({
|
|
44
|
+
title: 'State.apply addWriter rejects payloads when node entry writing key mismatches registered key',
|
|
45
|
+
setupScenario: setupScenarioWithRegisteredKey,
|
|
46
|
+
buildValidPayload: context =>
|
|
47
|
+
buildAddWriterPayload(context, { writerKeyBuffer: context.registeredWriterKey }),
|
|
48
|
+
mutatePayload: (_t, payload) => payload,
|
|
49
|
+
applyInvalidPayload: applyWithRequesterWriterKeyMismatch,
|
|
50
|
+
assertStateUnchanged: (t, context) =>
|
|
51
|
+
assertAddWriterFailureState(t, context, { skipSync: true, expectRegistryEntry: true }),
|
|
52
|
+
expectedLogs: ['Invalid writer key: either not owned by requester or different from assigned key.']
|
|
53
|
+
}).performScenario();
|
|
54
|
+
}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
2
|
+
import {
|
|
3
|
+
setupAddWriterScenario,
|
|
4
|
+
buildAddWriterPayload,
|
|
5
|
+
assertAddWriterFailureState,
|
|
6
|
+
selectWriterPeer,
|
|
7
|
+
defaultWriterFunding
|
|
8
|
+
} from './addWriterScenarioHelpers.js';
|
|
9
|
+
import { initializeBalances, whitelistAddress } from '../common/commonScenarioHelper.js';
|
|
10
|
+
import { eventFlush } from '../../../../helpers/autobaseTestHelpers.js';
|
|
11
|
+
|
|
12
|
+
async function setupScenarioWithForeignKey(t) {
|
|
13
|
+
const context = await setupAddWriterScenario(t, { nodes: 3 });
|
|
14
|
+
const foreignPeer = selectWriterPeer(context, 1);
|
|
15
|
+
await fundAndWhitelistPeer(context, foreignPeer);
|
|
16
|
+
await promotePeerToWriter(context, foreignPeer);
|
|
17
|
+
context.foreignWriterKey = foreignPeer.base.local.key;
|
|
18
|
+
return context;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
async function fundAndWhitelistPeer(context, peer) {
|
|
22
|
+
const funding = context.addWriterScenario?.writerInitialBalance ?? defaultWriterFunding;
|
|
23
|
+
await initializeBalances(context, [[peer.wallet.address, funding]]);
|
|
24
|
+
await whitelistAddress(context, peer.wallet.address);
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
async function promotePeerToWriter(context, peer) {
|
|
28
|
+
const payload = await buildAddWriterPayload(context, { readerPeer: peer });
|
|
29
|
+
await appendPayload(context, payload);
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
async function appendPayload(context, payload) {
|
|
33
|
+
const node = context.adminBootstrap ?? context.bootstrap;
|
|
34
|
+
if (!node?.base) {
|
|
35
|
+
throw new Error('Writer key ownership scenario requires an admin node with a writable base.');
|
|
36
|
+
}
|
|
37
|
+
await node.base.append(payload);
|
|
38
|
+
await node.base.update();
|
|
39
|
+
await eventFlush();
|
|
40
|
+
await context.sync();
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
export default function addWriterWriterKeyOwnershipScenario() {
|
|
44
|
+
new OperationValidationScenarioBase({
|
|
45
|
+
title: 'State.apply addWriter rejects payloads when writer key belongs to another node',
|
|
46
|
+
setupScenario: setupScenarioWithForeignKey,
|
|
47
|
+
buildValidPayload: context =>
|
|
48
|
+
buildAddWriterPayload(context, { writerKeyBuffer: context.foreignWriterKey }),
|
|
49
|
+
mutatePayload: (_t, payload) => payload,
|
|
50
|
+
applyInvalidPayload: appendPayload,
|
|
51
|
+
assertStateUnchanged: assertAddWriterFailureState,
|
|
52
|
+
expectedLogs: ['Invalid writer key: either not owned by requester or different from assigned key.']
|
|
53
|
+
}).performScenario();
|
|
54
|
+
}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import OperationValidationScenarioBase from '../common/base/OperationValidationScenarioBase.js';
|
|
2
|
+
import { safeDecodeApplyOperation, safeEncodeApplyOperation } from '../../../../../src/utils/protobuf/operationHelpers.js';
|
|
3
|
+
import { ZERO_WK } from '../../../../../src/utils/buffer.js';
|
|
4
|
+
import {
|
|
5
|
+
setupAddWriterScenario,
|
|
6
|
+
buildAddWriterPayload,
|
|
7
|
+
assertAddWriterFailureState,
|
|
8
|
+
applyWithRoleAccessBypass
|
|
9
|
+
} from './addWriterScenarioHelpers.js';
|
|
10
|
+
|
|
11
|
+
function mutateWriterKeyToZero(t, validPayload) {
|
|
12
|
+
const decoded = safeDecodeApplyOperation(validPayload);
|
|
13
|
+
t.ok(decoded, 'fixtures decode');
|
|
14
|
+
if (!decoded.rao) return validPayload;
|
|
15
|
+
decoded.rao.iw = ZERO_WK;
|
|
16
|
+
return safeEncodeApplyOperation(decoded);
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export default function addWriterZeroWriterKeyScenario() {
|
|
20
|
+
new OperationValidationScenarioBase({
|
|
21
|
+
title: 'State.apply addWriter rejects zero writer key requests',
|
|
22
|
+
setupScenario: setupAddWriterScenario,
|
|
23
|
+
buildValidPayload: buildAddWriterPayload,
|
|
24
|
+
mutatePayload: mutateWriterKeyToZero,
|
|
25
|
+
applyInvalidPayload: applyWithRoleAccessBypass,
|
|
26
|
+
assertStateUnchanged: assertAddWriterFailureState,
|
|
27
|
+
expectedLogs: ['Writer cannot initialize with zero-writer-key.']
|
|
28
|
+
}).performScenario();
|
|
29
|
+
}
|