-
Notifications
You must be signed in to change notification settings - Fork 44
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #2291 from planetarium/move-NCStagePolicy
Move `NCStagePolicy` to `NineChronicles.Headless`
- Loading branch information
Showing
9 changed files
with
481 additions
and
5 deletions.
There are no files selected for viewing
Submodule Lib9c
updated
6 files
2 changes: 1 addition & 1 deletion
2
...onicles.Headless.AccessControlCenter/AccessControlService/IMutableAccessControlService.cs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
324 changes: 324 additions & 0 deletions
324
NineChronicles.Headless.Tests/Policies/StagePolicyTest.cs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,324 @@ | ||
namespace NineChronicles.Headless.Tests.Policies | ||
{ | ||
using System; | ||
using System.Collections.Generic; | ||
using System.Collections.Immutable; | ||
using System.Linq; | ||
using System.Threading.Tasks; | ||
using Lib9c.Renderers; | ||
using Libplanet.Action; | ||
using Libplanet.Blockchain; | ||
using Libplanet.Blockchain.Policies; | ||
using Libplanet.Crypto; | ||
using Libplanet.Store; | ||
using Libplanet.Store.Trie; | ||
using Libplanet.Types.Blocks; | ||
using Libplanet.Types.Tx; | ||
using Nekoyume; | ||
using Nekoyume.Action; | ||
using Nekoyume.Action.Loader; | ||
using Nekoyume.Blockchain.Policy; | ||
using Nekoyume.Model; | ||
using Nekoyume.Model.State; | ||
using NineChronicles.Headless.Policies; | ||
using Xunit; | ||
|
||
public class StagePolicyTest | ||
{ | ||
private readonly PrivateKey[] _accounts; | ||
|
||
private readonly Dictionary<Address, Transaction[]> _txs; | ||
|
||
public StagePolicyTest() | ||
{ | ||
_accounts = new[] | ||
{ | ||
new PrivateKey(), | ||
new PrivateKey(), | ||
new PrivateKey(), | ||
new PrivateKey(), | ||
}; | ||
_txs = _accounts.ToDictionary( | ||
acc => acc.ToAddress(), | ||
acc => Enumerable | ||
.Range(0, 10) | ||
.Select( | ||
n => Transaction.Create( | ||
n, | ||
acc, | ||
default, | ||
new ActionBase[0].ToPlainValues() | ||
) | ||
) | ||
.ToArray() | ||
); | ||
} | ||
|
||
[Fact] | ||
public void Stage() | ||
{ | ||
NCStagePolicy stagePolicy = new NCStagePolicy(TimeSpan.FromHours(1), 2); | ||
BlockChain chain = MakeChainWithStagePolicy(stagePolicy); | ||
|
||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][0]); | ||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][1]); | ||
stagePolicy.Stage(chain, _txs[_accounts[1].ToAddress()][0]); | ||
stagePolicy.Stage(chain, _txs[_accounts[2].ToAddress()][0]); | ||
|
||
AssertTxs( | ||
chain, | ||
stagePolicy, | ||
_txs[_accounts[0].ToAddress()][0], | ||
_txs[_accounts[0].ToAddress()][1], | ||
_txs[_accounts[1].ToAddress()][0], | ||
_txs[_accounts[2].ToAddress()][0] | ||
); | ||
} | ||
|
||
[Fact] | ||
public void StageOverQuota() | ||
{ | ||
NCStagePolicy stagePolicy = new NCStagePolicy(TimeSpan.FromHours(1), 2); | ||
BlockChain chain = MakeChainWithStagePolicy(stagePolicy); | ||
|
||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][0]); | ||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][1]); | ||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][2]); | ||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][3]); | ||
|
||
AssertTxs( | ||
chain, | ||
stagePolicy, | ||
_txs[_accounts[0].ToAddress()][0], | ||
_txs[_accounts[0].ToAddress()][1] | ||
); | ||
} | ||
|
||
[Fact] | ||
public void StageOverQuotaInverseOrder() | ||
{ | ||
NCStagePolicy stagePolicy = new NCStagePolicy(TimeSpan.FromHours(1), 2); | ||
BlockChain chain = MakeChainWithStagePolicy(stagePolicy); | ||
|
||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][3]); | ||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][2]); | ||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][1]); | ||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][0]); | ||
|
||
AssertTxs( | ||
chain, | ||
stagePolicy, | ||
_txs[_accounts[0].ToAddress()][0], | ||
_txs[_accounts[0].ToAddress()][1] | ||
); | ||
} | ||
|
||
[Fact] | ||
public void StageOverQuotaOutOfOrder() | ||
{ | ||
NCStagePolicy stagePolicy = new NCStagePolicy(TimeSpan.FromHours(1), 2); | ||
BlockChain chain = MakeChainWithStagePolicy(stagePolicy); | ||
|
||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][2]); | ||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][1]); | ||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][3]); | ||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][0]); | ||
|
||
AssertTxs( | ||
chain, | ||
stagePolicy, | ||
_txs[_accounts[0].ToAddress()][0], | ||
_txs[_accounts[0].ToAddress()][1] | ||
); | ||
} | ||
|
||
[Fact] | ||
public void StageSameNonce() | ||
{ | ||
NCStagePolicy stagePolicy = new NCStagePolicy(TimeSpan.FromHours(1), 2); | ||
BlockChain chain = MakeChainWithStagePolicy(stagePolicy); | ||
var txA = Transaction.Create(0, _accounts[0], default, new ActionBase[0].ToPlainValues()); | ||
var txB = Transaction.Create(0, _accounts[0], default, new ActionBase[0].ToPlainValues()); | ||
var txC = Transaction.Create(0, _accounts[0], default, new ActionBase[0].ToPlainValues()); | ||
|
||
stagePolicy.Stage(chain, txA); | ||
stagePolicy.Stage(chain, txB); | ||
stagePolicy.Stage(chain, txC); | ||
|
||
AssertTxs(chain, stagePolicy, txA, txB); | ||
} | ||
|
||
[Fact] | ||
public async Task StateFromMultiThread() | ||
{ | ||
NCStagePolicy stagePolicy = new NCStagePolicy(TimeSpan.FromHours(1), 2); | ||
BlockChain chain = MakeChainWithStagePolicy(stagePolicy); | ||
|
||
await Task.WhenAll( | ||
Enumerable | ||
.Range(0, 40) | ||
.Select(i => Task.Run(() => | ||
{ | ||
stagePolicy.Stage(chain, _txs[_accounts[i / 10].ToAddress()][i % 10]); | ||
})) | ||
); | ||
AssertTxs( | ||
chain, | ||
stagePolicy, | ||
_txs[_accounts[0].ToAddress()][0], | ||
_txs[_accounts[0].ToAddress()][1], | ||
_txs[_accounts[1].ToAddress()][0], | ||
_txs[_accounts[1].ToAddress()][1], | ||
_txs[_accounts[2].ToAddress()][0], | ||
_txs[_accounts[2].ToAddress()][1], | ||
_txs[_accounts[3].ToAddress()][0], | ||
_txs[_accounts[3].ToAddress()][1] | ||
); | ||
} | ||
|
||
[Fact] | ||
public void IterateAfterUnstage() | ||
{ | ||
NCStagePolicy stagePolicy = new NCStagePolicy(TimeSpan.FromHours(1), 2); | ||
BlockChain chain = MakeChainWithStagePolicy(stagePolicy); | ||
|
||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][0]); | ||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][1]); | ||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][2]); | ||
stagePolicy.Stage(chain, _txs[_accounts[0].ToAddress()][3]); | ||
|
||
AssertTxs( | ||
chain, | ||
stagePolicy, | ||
_txs[_accounts[0].ToAddress()][0], | ||
_txs[_accounts[0].ToAddress()][1] | ||
); | ||
|
||
stagePolicy.Unstage(chain, _txs[_accounts[0].ToAddress()][0].Id); | ||
|
||
AssertTxs( | ||
chain, | ||
stagePolicy, | ||
_txs[_accounts[0].ToAddress()][1], | ||
_txs[_accounts[0].ToAddress()][2] | ||
); | ||
} | ||
|
||
[Fact] | ||
public void CalculateNextTxNonceCorrectWhenTxOverQuota() | ||
{ | ||
NCStagePolicy stagePolicy = new NCStagePolicy(TimeSpan.FromHours(1), 2); | ||
BlockChain chain = MakeChainWithStagePolicy(stagePolicy); | ||
|
||
long nextTxNonce = chain.GetNextTxNonce(_accounts[0].ToAddress()); | ||
Assert.Equal(0, nextTxNonce); | ||
var txA = Transaction.Create(nextTxNonce, _accounts[0], default, new ActionBase[0].ToPlainValues()); | ||
stagePolicy.Stage(chain, txA); | ||
|
||
nextTxNonce = chain.GetNextTxNonce(_accounts[0].ToAddress()); | ||
Assert.Equal(1, nextTxNonce); | ||
var txB = Transaction.Create(nextTxNonce, _accounts[0], default, new ActionBase[0].ToPlainValues()); | ||
stagePolicy.Stage(chain, txB); | ||
|
||
nextTxNonce = chain.GetNextTxNonce(_accounts[0].ToAddress()); | ||
Assert.Equal(2, nextTxNonce); | ||
var txC = Transaction.Create(nextTxNonce, _accounts[0], default, new ActionBase[0].ToPlainValues()); | ||
stagePolicy.Stage(chain, txC); | ||
|
||
nextTxNonce = chain.GetNextTxNonce(_accounts[0].ToAddress()); | ||
Assert.Equal(3, nextTxNonce); | ||
|
||
AssertTxs( | ||
chain, | ||
stagePolicy, | ||
txA, | ||
txB); | ||
} | ||
|
||
private void AssertTxs(BlockChain blockChain, NCStagePolicy policy, params Transaction[] txs) | ||
{ | ||
foreach (Transaction tx in txs) | ||
{ | ||
Assert.Equal(tx, policy.Get(blockChain, tx.Id, filtered: true)); | ||
} | ||
|
||
Assert.Equal( | ||
txs.ToHashSet(), | ||
policy.Iterate(blockChain, filtered: true).ToHashSet() | ||
); | ||
} | ||
|
||
private BlockChain MakeChainWithStagePolicy(NCStagePolicy stagePolicy) | ||
{ | ||
BlockPolicySource blockPolicySource = new BlockPolicySource(); | ||
IBlockPolicy policy = blockPolicySource.GetPolicy(); | ||
BlockChain chain = | ||
BlockChainHelper.MakeBlockChain( | ||
blockRenderers: new[] { new BlockRenderer() }, | ||
policy: policy, | ||
stagePolicy: stagePolicy); | ||
return chain; | ||
} | ||
|
||
public static class BlockChainHelper | ||
{ | ||
public static BlockChain MakeBlockChain( | ||
BlockRenderer[] blockRenderers, | ||
IBlockPolicy policy = null, | ||
IStagePolicy stagePolicy = null, | ||
IStore store = null, | ||
IStateStore stateStore = null) | ||
{ | ||
PrivateKey adminPrivateKey = new PrivateKey(); | ||
|
||
policy ??= new BlockPolicy(); | ||
stagePolicy ??= new VolatileStagePolicy(); | ||
store ??= new DefaultStore(null); | ||
stateStore ??= new TrieStateStore(new DefaultKeyValueStore(null)); | ||
Block genesis = MakeGenesisBlock(adminPrivateKey.ToAddress(), ImmutableHashSet<Address>.Empty); | ||
return BlockChain.Create( | ||
policy, | ||
stagePolicy, | ||
store, | ||
stateStore, | ||
genesis, | ||
new ActionEvaluator( | ||
policyBlockActionGetter: _ => policy.BlockAction, | ||
stateStore: stateStore, | ||
actionTypeLoader: new NCActionLoader() | ||
), | ||
renderers: blockRenderers); | ||
} | ||
|
||
public static Block MakeGenesisBlock( | ||
Address adminAddress, | ||
IImmutableSet<Address> activatedAddresses, | ||
AuthorizedMinersState authorizedMinersState = null, | ||
DateTimeOffset? timestamp = null, | ||
PendingActivationState[] pendingActivations = null | ||
) | ||
{ | ||
PrivateKey privateKey = new PrivateKey(); | ||
if (pendingActivations is null) | ||
{ | ||
var nonce = new byte[] { 0x00, 0x01, 0x02, 0x03 }; | ||
(ActivationKey activationKey, PendingActivationState pendingActivation) = | ||
ActivationKey.Create(privateKey, nonce); | ||
pendingActivations = new[] { pendingActivation }; | ||
} | ||
|
||
var sheets = TableSheetsImporter.ImportSheets(); | ||
return BlockHelper.ProposeGenesisBlock( | ||
sheets, | ||
new GoldDistribution[0], | ||
pendingActivations, | ||
new AdminState(adminAddress, 1500000), | ||
activatedAccounts: activatedAddresses, | ||
isActivateAdminAddress: false, | ||
credits: null, | ||
privateKey: privateKey, | ||
timestamp: timestamp ?? DateTimeOffset.MinValue); | ||
} | ||
} | ||
} | ||
} |
9 changes: 9 additions & 0 deletions
9
NineChronicles.Headless/AccessControlService/IAccessControlService.cs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,9 @@ | ||
using Libplanet.Crypto; | ||
|
||
namespace NineChronicles.Headless.AccessControlService | ||
{ | ||
public interface IAccessControlService | ||
{ | ||
public int? GetTxQuota(Address address); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.