diff --git a/ethereumj-core/src/main/java/org/ethereum/core/BlockchainImpl.java b/ethereumj-core/src/main/java/org/ethereum/core/BlockchainImpl.java index c5a043924c..4307db99a6 100644 --- a/ethereumj-core/src/main/java/org/ethereum/core/BlockchainImpl.java +++ b/ethereumj-core/src/main/java/org/ethereum/core/BlockchainImpl.java @@ -562,7 +562,9 @@ public synchronized BlockSummary add(Repository repo, final Block block) { public synchronized BlockSummary addImpl(Repository repo, final Block block) { if (exitOn < block.getNumber()) { - System.out.print("Exiting after block.number: " + bestBlock.getNumber()); + String msg = String.format("Exiting after block.number: %d", bestBlock.getNumber()); + logger.info(msg); + System.out.println(msg); dbFlushManager.flushSync(); System.exit(-1); } diff --git a/ethereumj-core/src/main/java/org/ethereum/core/Transaction.java b/ethereumj-core/src/main/java/org/ethereum/core/Transaction.java index 18f93b0e3f..761695fc56 100644 --- a/ethereumj-core/src/main/java/org/ethereum/core/Transaction.java +++ b/ethereumj-core/src/main/java/org/ethereum/core/Transaction.java @@ -473,7 +473,7 @@ public byte[] getEncodedRaw() { return rlpRaw; } - public byte[] getEncoded() { + public synchronized byte[] getEncoded() { if (rlpEncoded != null) return rlpEncoded; diff --git a/ethereumj-core/src/main/java/org/ethereum/net/rlpx/discover/NodeManager.java b/ethereumj-core/src/main/java/org/ethereum/net/rlpx/discover/NodeManager.java index 58dee3a7b1..1711126886 100644 --- a/ethereumj-core/src/main/java/org/ethereum/net/rlpx/discover/NodeManager.java +++ b/ethereumj-core/src/main/java/org/ethereum/net/rlpx/discover/NodeManager.java @@ -117,7 +117,7 @@ public ScheduledExecutorService getPongTimer() { return pongTimer; } - void setBootNodes(List bootNodes) { + public void setBootNodes(List bootNodes) { this.bootNodes = bootNodes; } diff --git a/ethereumj-core/src/main/java/org/ethereum/util/blockchain/StandaloneBlockchain.java b/ethereumj-core/src/main/java/org/ethereum/util/blockchain/StandaloneBlockchain.java index 1d9c592b6b..7a8c7229c2 100644 --- a/ethereumj-core/src/main/java/org/ethereum/util/blockchain/StandaloneBlockchain.java +++ b/ethereumj-core/src/main/java/org/ethereum/util/blockchain/StandaloneBlockchain.java @@ -25,6 +25,7 @@ import org.ethereum.config.blockchain.DaoNoHFConfig; import org.ethereum.config.blockchain.FrontierConfig; import org.ethereum.config.blockchain.HomesteadConfig; +import org.ethereum.config.blockchain.PetersburgConfig; import org.ethereum.core.*; import org.ethereum.core.genesis.GenesisLoader; import org.ethereum.crypto.ECKey; @@ -765,8 +766,8 @@ public synchronized void updateBatch(Map rows) { } // Override blockchain net config for fast mining - public static ByzantiumConfig getEasyMiningConfig() { - return new ByzantiumConfig(new DaoNoHFConfig(new HomesteadConfig(new HomesteadConfig.HomesteadConstants() { + public static PetersburgConfig getEasyMiningConfig() { + return new PetersburgConfig(new DaoNoHFConfig(new HomesteadConfig(new HomesteadConfig.HomesteadConstants() { @Override public BigInteger getMINIMUM_DIFFICULTY() { return BigInteger.ONE; diff --git a/ethereumj-core/src/main/java/org/ethereum/vm/program/Program.java b/ethereumj-core/src/main/java/org/ethereum/vm/program/Program.java index 659caf9af5..b8c11c5847 100644 --- a/ethereumj-core/src/main/java/org/ethereum/vm/program/Program.java +++ b/ethereumj-core/src/main/java/org/ethereum/vm/program/Program.java @@ -549,23 +549,24 @@ private void createContractImpl(DataWord value, byte[] programCode, byte[] newAd } // 4. CREATE THE CONTRACT OUT OF RETURN - byte[] code = result.getHReturn(); - - long storageCost = getLength(code) * getBlockchainConfig().getGasCost().getCREATE_DATA(); - long afterSpend = programInvoke.getGas().longValue() - storageCost - result.getGasUsed(); - if (afterSpend < 0) { - if (!blockchainConfig.getConstants().createEmptyContractOnOOG()) { - result.setException(Program.Exception.notEnoughSpendingGas("No gas to return just created contract", + if (!result.isRevert() && result.getException() == null) { + byte[] code = result.getHReturn(); + long storageCost = getLength(code) * getBlockchainConfig().getGasCost().getCREATE_DATA(); + long afterSpend = programInvoke.getGas().longValue() - result.getGasUsed() - storageCost; + if (afterSpend < 0) { + if (!blockchainConfig.getConstants().createEmptyContractOnOOG()) { + result.setException(Program.Exception.notEnoughSpendingGas("No gas to return just created contract", + storageCost, this)); + } else { + track.saveCode(newAddress, EMPTY_BYTE_ARRAY); + } + } else if (getLength(code) > blockchainConfig.getConstants().getMAX_CONTRACT_SZIE()) { + result.setException(Program.Exception.notEnoughSpendingGas("Contract size too large: " + getLength(result.getHReturn()), storageCost, this)); } else { - track.saveCode(newAddress, EMPTY_BYTE_ARRAY); + result.spendGas(storageCost); + track.saveCode(newAddress, code); } - } else if (getLength(code) > blockchainConfig.getConstants().getMAX_CONTRACT_SZIE()) { - result.setException(Program.Exception.notEnoughSpendingGas("Contract size too large: " + getLength(result.getHReturn()), - storageCost, this)); - } else if (!result.isRevert()){ - result.spendGas(storageCost); - track.saveCode(newAddress, code); } getResult().merge(result); diff --git a/ethereumj-core/src/main/resources/ethereumj.conf b/ethereumj-core/src/main/resources/ethereumj.conf index 5a550e27cb..9da9d9751f 100644 --- a/ethereumj-core/src/main/resources/ethereumj.conf +++ b/ethereumj-core/src/main/resources/ethereumj.conf @@ -22,20 +22,25 @@ peer.discovery = { "52.74.57.123:30303", # Parity discovery nodes + "193.70.55.37:30303", + "144.217.139.5:30303", + "139.99.51.203:30303", + "139.99.160.213:30303", + "163.172.131.191:30303", + "212.47.247.103:30303", + "163.172.157.114:30303" + "138.201.223.35:30303", + "138.201.144.135:30303", + "51.15.42.252:30303", + "163.172.171.38:30303" "163.172.187.252:30303", - "163.172.157.114:30303", "136.243.154.244:30303", "88.212.206.70:30303", "37.128.191.230:30303", "46.20.235.22:30303", "216.158.85.185:30303", - "212.47.247.103:30303", - "138.201.144.135:30303", - "45.55.33.62:30303", - "188.166.255.12:30303", - "159.203.210.80:30303", - "51.15.42.252:30303", - "163.172.171.38:30303" + "52.79.241.155:30303", + "52.78.149.82:30303", ] # external IP/hostname which is reported as our host during discovery diff --git a/ethereumj-core/src/main/resources/version.properties b/ethereumj-core/src/main/resources/version.properties index 0129a96eb8..90c1bb8591 100644 --- a/ethereumj-core/src/main/resources/version.properties +++ b/ethereumj-core/src/main/resources/version.properties @@ -1,3 +1,3 @@ -versionNumber='1.11.0' +versionNumber='1.12.0' // Remove org.ethereum.db.migrate.MigrateHeaderSourceTotalDiff with databaseVersion > 6 databaseVersion=6 diff --git a/ethereumj-core/src/test/java/org/ethereum/core/ChainTest.java b/ethereumj-core/src/test/java/org/ethereum/core/ChainTest.java new file mode 100644 index 0000000000..654f1b6a4a --- /dev/null +++ b/ethereumj-core/src/test/java/org/ethereum/core/ChainTest.java @@ -0,0 +1,64 @@ +package org.ethereum.core; + +import org.ethereum.core.genesis.GenesisJson; +import org.ethereum.core.genesis.GenesisLoader; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.InputStream; +import java.math.BigInteger; + +import static org.junit.Assert.*; + +/** + * @author alexbraz + * @since 29/03/2019 + */ +public class ChainTest { + + private static final Logger logger = LoggerFactory.getLogger("test"); + + Block genesis = GenesisLoader.loadGenesis(getClass().getResourceAsStream("/genesis/olympic.json")); + GenesisJson genesisJson = GenesisLoader.loadGenesisJson((InputStream) getClass().getResourceAsStream("/genesis/olympic.json")); + + @Test + public void testContainsBlock() { + Chain c = new Chain(); + c.add(genesis); + assertEquals(genesis, c.getLast()); + } + + @Test + public void testBlockHashNotNull() { + Chain c = new Chain(); + c.add(genesis); + assertNotNull(c.getLast().getHash()); + } + + @Test + public void testDifficultyGenesisCorrectLoadedAndConverted() { + Chain c = new Chain(); + c.add(genesis); + assertEquals(new BigInteger(genesisJson.getDifficulty().replace("0x", ""), 16).intValue(), c.getLast().getDifficultyBI().intValue()); + } + + @Test + public void testParentOnTheChain() { + Chain c = new Chain(); + c.add(genesis); + Block block = new Block(genesis.getHeader(), genesis.getTransactionsList(), null); + assertFalse(c.isParentOnTheChain(block)); + } + + @Test + public void testParentOnTheChain2() { + Chain c = new Chain(); + c.add(genesis); + assertFalse(c.isParentOnTheChain(genesis)); + } + + + + +} diff --git a/ethereumj-core/src/test/java/org/ethereum/core/PremineRawTest.java b/ethereumj-core/src/test/java/org/ethereum/core/PremineRawTest.java new file mode 100644 index 0000000000..7d6f0aacfd --- /dev/null +++ b/ethereumj-core/src/test/java/org/ethereum/core/PremineRawTest.java @@ -0,0 +1,25 @@ +package org.ethereum.core; + +import org.junit.Test; + +import java.math.BigInteger; + +import static org.junit.Assert.*; + +/** + * @author alexbraz + * @since 29/03/2019 + */ +public class PremineRawTest { + + @Test + public void testPremineRawNotNull() { + + byte[] addr = "0xcf0f482f2c1ef1f221f09e3cf14122fce0424f94".getBytes(); + PremineRaw pr = new PremineRaw(addr, BigInteger.ONE, Denomination.ETHER); + + assertTrue(pr.getDenomination() == Denomination.ETHER); + assertEquals(pr.value, BigInteger.ONE); + assertNotNull(pr.getAddr()); + } +} diff --git a/ethereumj-core/src/test/java/org/ethereum/datasource/BatchSourceWriterTest.java b/ethereumj-core/src/test/java/org/ethereum/datasource/BatchSourceWriterTest.java new file mode 100644 index 0000000000..252f04733a --- /dev/null +++ b/ethereumj-core/src/test/java/org/ethereum/datasource/BatchSourceWriterTest.java @@ -0,0 +1,38 @@ +package org.ethereum.datasource; + +import org.junit.Test; + +import java.math.BigInteger; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertTrue; +import static org.mockito.Mockito.mock; + +/** + * @author alexbraz + * @since 29/03/2019 + */ +public class BatchSourceWriterTest { + + @Test + public void testFlush() { + BatchSource batchSource = mock(BatchSource.class); + BatchSourceWriter bsw = new BatchSourceWriter(batchSource); + bsw.put("KEY", BigInteger.ONE); + assertTrue(bsw.flushImpl()); + } + + @Test + public void testValues() { + BatchSource batchSource = mock(BatchSource.class); + BatchSourceWriter bsw = new BatchSourceWriter(batchSource); + bsw.put("ONE", BigInteger.ONE); + bsw.put("TEN", BigInteger.TEN); + bsw.put("ZERO", BigInteger.ZERO); + + bsw.buf.forEach((K, v) -> { + assertEquals(v, bsw.buf.get(K)); + }); + + } +} diff --git a/ethereumj-core/src/test/java/org/ethereum/datasource/BlockReplayTest.java b/ethereumj-core/src/test/java/org/ethereum/datasource/BlockReplayTest.java new file mode 100644 index 0000000000..6762903eb5 --- /dev/null +++ b/ethereumj-core/src/test/java/org/ethereum/datasource/BlockReplayTest.java @@ -0,0 +1,133 @@ +package org.ethereum.datasource; + +import org.ethereum.core.Block; +import org.ethereum.core.Genesis; +import org.ethereum.datasource.inmem.HashMapDB; +import org.ethereum.db.IndexedBlockStore; +import org.ethereum.db.TransactionStore; +import org.ethereum.listener.BlockReplay; +import org.ethereum.listener.EthereumListener; +import org.junit.Before; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.spongycastle.util.encoders.Hex; + +import java.io.File; +import java.io.IOException; +import java.math.BigInteger; +import java.net.URISyntaxException; +import java.net.URL; +import java.nio.charset.StandardCharsets; +import java.nio.file.Files; +import java.util.ArrayList; +import java.util.List; + +import static java.math.BigInteger.ZERO; +import static org.mockito.ArgumentMatchers.any; +import static org.mockito.Mockito.*; + +/** + * @author alexbraz + * @since 29/03/2019 + */ +public class BlockReplayTest { + private static final Logger logger = LoggerFactory.getLogger("test"); + + + BlockReplay replay; + EthereumListener listener; + private List blocks = new ArrayList<>(); + private BigInteger totDifficulty = ZERO; + Block genesis = Genesis.getInstance(); + + @Before + public void setup() throws URISyntaxException, IOException { + URL scenario1 = ClassLoader + .getSystemResource("blockstore/load.dmp"); + + File file = new File(scenario1.toURI()); + List strData = Files.readAllLines(file.toPath(), StandardCharsets.UTF_8); + + IndexedBlockStore indexedBlockStore = indexedBlockStore = new IndexedBlockStore(); + indexedBlockStore.init(new HashMapDB(), new HashMapDB()); + + TransactionStore txStore = new TransactionStore(new HashMapDB<>()); + + listener = mock(EthereumListener.class); + + replay = new BlockReplay(indexedBlockStore, txStore, listener, 0L); + for (String blockRLP : strData) { + + Block block = new Block( + Hex.decode(blockRLP)); + + if (block.getNumber() % 1000 == 0) + logger.info("adding block.hash: [{}] block.number: [{}]", + block.getShortHash(), + block.getNumber()); + + blocks.add(block); + totDifficulty = totDifficulty.add(block.getDifficultyBI()); + indexedBlockStore.saveBlock(block, totDifficulty, true); + } + + } + + @Test + public void testReplayBlock() { + IndexedBlockStore i = mock(IndexedBlockStore.class); + when(i.getChainBlockByNumber(anyLong())).thenReturn(genesis); + TransactionStore txStore = new TransactionStore(new HashMapDB<>()); + replay = new BlockReplay(i, txStore, listener, 0L); + replay.replay(); + + verify(listener, times(1)).onBlock(any()); + } + + @Test + public void testListenerNoConnection() { + replay.onNoConnections(); + verify(listener, times(1)).onNoConnections(); + + replay.onSyncDone(null); + verify(listener, times(1)).onSyncDone(any()); + + replay.onNodeDiscovered(any()); + verify(listener, times(1)).onNodeDiscovered(any()); + + replay.onEthStatusUpdated(any(), any()); + verify(listener, times(1)).onEthStatusUpdated(any(), any()); + + replay.onHandShakePeer(any(), any()); + verify(listener, times(1)).onHandShakePeer(any(), any()); + + replay.onPeerAddedToSyncPool(any()); + verify(listener, times(1)).onPeerAddedToSyncPool(any()); + + replay.onPeerDisconnect(anyString(), anyLong()); + verify(listener, times(1)).onPeerDisconnect(anyString(), anyLong()); + + replay.onPendingStateChanged(any()); + verify(listener, times(1)).onPendingStateChanged(any()); + + replay.onPendingTransactionUpdate(any(), any(), any()); + verify(listener, times(1)).onPendingTransactionUpdate(any(), any(), any()); + + replay.onRecvMessage(any(), any()); + verify(listener, times(1)).onRecvMessage(any(), any()); + + replay.onTransactionExecuted(any()); + verify(listener, times(1)).onTransactionExecuted(any()); + + replay.onSendMessage(any(), any()); + verify(listener, times(1)).onSendMessage(any(), any()); + + replay.onVMTraceCreated(anyString(), anyString()); + verify(listener, times(1)).onVMTraceCreated(anyString(), anyString()); + + } + + + +} diff --git a/ethereumj-core/src/test/java/org/ethereum/datasource/BloomFilterTest.java b/ethereumj-core/src/test/java/org/ethereum/datasource/BloomFilterTest.java new file mode 100644 index 0000000000..87ecaa431f --- /dev/null +++ b/ethereumj-core/src/test/java/org/ethereum/datasource/BloomFilterTest.java @@ -0,0 +1,392 @@ +package org.ethereum.datasource; + +import org.junit.Test; + +import java.io.UnsupportedEncodingException; +import java.nio.ByteBuffer; +import java.util.ArrayList; +import java.util.List; +import java.util.Random; +import java.util.UUID; + +import static org.junit.Assert.*; + +/** + * The class mostly borrowed from http://github.com/magnuss/java-bloomfilter + * + * @author Magnus Skjegstad + */ +public class BloomFilterTest { + static Random r = new Random(); + + public static byte[] getBytesFromUUID(UUID uuid) { + ByteBuffer bb = ByteBuffer.wrap(new byte[16]); + bb.putLong(uuid.getMostSignificantBits()); + bb.putLong(uuid.getLeastSignificantBits()); + + return bb.array(); + } + + @Test + public void testConstructorCNK() throws Exception { + System.out.println("BloomFilter(c,n,k)"); + + for (int i = 0; i < 10000; i++) { + double c = r.nextInt(20) + 1; + int n = r.nextInt(10000) + 1; + int k = r.nextInt(20) + 1; + BloomFilter bf = new BloomFilter(c, n, k); + assertEquals(bf.getK(), k); + assertEquals(bf.getExpectedBitsPerElement(), c, 0); + assertEquals(bf.getExpectedNumberOfElements(), n); + assertEquals(bf.size(), c*n, 0); + } + } + + /** + * Test of equals method, of class BloomFilter. + * @throws UnsupportedEncodingException + */ + @Test + public void testEquals() throws UnsupportedEncodingException { + System.out.println("equals"); + BloomFilter instance1 = new BloomFilter(1000, 100); + BloomFilter instance2 = new BloomFilter(1000, 100); + + for (int i = 0; i < 100; i++) { + byte[] val = UUID.randomUUID().toString().getBytes("UTF-8"); + instance1.add(val); + instance2.add(val); + } + + assert(instance1.equals(instance2)); + assert(instance2.equals(instance1)); + + byte[] anotherEntryByteArray = "Another Entry".getBytes("UTF-8"); + instance1.add(anotherEntryByteArray); // make instance1 and instance2 different before clearing + + instance1.clear(); + instance2.clear(); + + assert(instance1.equals(instance2)); + assert(instance2.equals(instance1)); + + for (int i = 0; i < 100; i++) { + byte[] val = UUID.randomUUID().toString().getBytes("UTF-8"); + instance1.add(val); + instance2.add(val); + } + + assertTrue(instance1.equals(instance2)); + assertTrue(instance2.equals(instance1)); + } + + /** + * Test of hashCode method, of class BloomFilter. + * @throws UnsupportedEncodingException + */ + @Test + public void testHashCode() throws UnsupportedEncodingException { + System.out.println("hashCode"); + + BloomFilter instance1 = new BloomFilter(1000, 100); + BloomFilter instance2 = new BloomFilter(1000, 100); + + assertTrue(instance1.hashCode() == instance2.hashCode()); + + for (int i = 0; i < 100; i++) { + byte[] val = UUID.randomUUID().toString().getBytes("UTF-8"); + instance1.add(val); + instance2.add(val); + } + + assertTrue(instance1.hashCode() == instance2.hashCode()); + + instance1.clear(); + instance2.clear(); + + assertTrue(instance1.hashCode() == instance2.hashCode()); + + instance1 = new BloomFilter(100, 10); + instance2 = new BloomFilter(100, 9); + assertFalse(instance1.hashCode() == instance2.hashCode()); + + instance1 = new BloomFilter(100, 10); + instance2 = new BloomFilter(99, 9); + assertFalse(instance1.hashCode() == instance2.hashCode()); + + instance1 = new BloomFilter(100, 10); + instance2 = new BloomFilter(50, 10); + assertFalse(instance1.hashCode() == instance2.hashCode()); + } + + /** + * Test of expectedFalsePositiveProbability method, of class BloomFilter. + */ + @Test + public void testExpectedFalsePositiveProbability() { + // These probabilities are taken from the bloom filter probability table at + // http://pages.cs.wisc.edu/~cao/papers/summary-cache/node8.html + System.out.println("expectedFalsePositiveProbability"); + BloomFilter instance = new BloomFilter(1000, 100); + double expResult = 0.00819; // m/n=10, k=7 + double result = instance.expectedFalsePositiveProbability(); + assertEquals(instance.getK(), 7); + assertEquals(expResult, result, 0.000009); + + instance = new BloomFilter(100, 10); + expResult = 0.00819; // m/n=10, k=7 + result = instance.expectedFalsePositiveProbability(); + assertEquals(instance.getK(), 7); + assertEquals(expResult, result, 0.000009); + + instance = new BloomFilter(20, 10); + expResult = 0.393; // m/n=2, k=1 + result = instance.expectedFalsePositiveProbability(); + assertEquals(1, instance.getK()); + assertEquals(expResult, result, 0.0005); + + instance = new BloomFilter(110, 10); + expResult = 0.00509; // m/n=11, k=8 + result = instance.expectedFalsePositiveProbability(); + assertEquals(8, instance.getK()); + assertEquals(expResult, result, 0.00001); + } + + /** + * Test of clear method, of class BloomFilter. + */ + @Test + public void testClear() { + System.out.println("clear"); + BloomFilter instance = new BloomFilter(1000, 100); + for (int i = 0; i < instance.size(); i++) + instance.setBit(i, true); + instance.clear(); + for (int i = 0; i < instance.size(); i++) + assertSame(instance.getBit(i), false); + } + + /** + * Test of add method, of class BloomFilter. + * @throws Exception + */ + @Test + public void testAdd() throws Exception { + System.out.println("add"); + BloomFilter instance = new BloomFilter(1000, 100); + + for (int i = 0; i < 100; i++) { + byte[] bytes = getBytesFromUUID(UUID.randomUUID()); + instance.add(bytes); + assert(instance.contains(bytes)); + } + } + + /** + * Test of contains method, of class BloomFilter. + * @throws Exception + */ + @Test + public void testContains() throws Exception { + System.out.println("contains"); + BloomFilter instance = new BloomFilter(10000, 10); + + for (int i = 0; i < 10; i++) { + byte[] bytes = getBytesFromUUID(UUID.randomUUID()); + instance.add(bytes); + assert(instance.contains(bytes)); + } + + assertFalse(instance.contains(UUID.randomUUID().toString().getBytes("UTF-8"))); + } + + /** + * Test of getBit method, of class BloomFilter. + */ + @Test + public void testGetBit() { + System.out.println("getBit"); + BloomFilter instance = new BloomFilter(1000, 100); + Random r = new Random(); + + for (int i = 0; i < 100; i++) { + boolean b = r.nextBoolean(); + instance.setBit(i, b); + assertSame(instance.getBit(i), b); + } + } + + /** + * Test of setBit method, of class BloomFilter. + */ + @Test + public void testSetBit() { + System.out.println("setBit"); + + BloomFilter instance = new BloomFilter(1000, 100); + Random r = new Random(); + + for (int i = 0; i < 100; i++) { + instance.setBit(i, true); + assertSame(instance.getBit(i), true); + } + + for (int i = 0; i < 100; i++) { + instance.setBit(i, false); + assertSame(instance.getBit(i), false); + } + } + + /** + * Test of size method, of class BloomFilter. + */ + @Test + public void testSize() { + System.out.println("size"); + for (int i = 100; i < 1000; i++) { + BloomFilter instance = new BloomFilter(i, 10); + assertEquals(instance.size(), i); + } + } + + /** Test error rate * + * @throws UnsupportedEncodingException + */ + @Test + public void testFalsePositiveRate1() throws UnsupportedEncodingException { + // Numbers are from // http://pages.cs.wisc.edu/~cao/papers/summary-cache/node8.html + System.out.println("falsePositiveRate1"); + + for (int j = 10; j < 21; j++) { + System.out.print(j-9 + "/11"); + List v = new ArrayList(); + BloomFilter instance = new BloomFilter(100*j,100); + + for (int i = 0; i < 100; i++) { + byte[] bytes = new byte[100]; + r.nextBytes(bytes); + v.add(bytes); + } + for (byte[] bytes : v) { + instance.add(bytes); + } + + long f = 0; + double tests = 300000; + for (int i = 0; i < tests; i++) { + byte[] bytes = new byte[100]; + r.nextBytes(bytes); + if (instance.contains(bytes)) { + if (!v.contains(bytes)) { + f++; + } + } + } + + double ratio = f / tests; + + System.out.println(" - got " + ratio + ", math says " + instance.expectedFalsePositiveProbability()); + assertEquals(instance.expectedFalsePositiveProbability(), ratio, 0.01); + } + } + + /** Test for correct k **/ + @Test + public void testGetK() { + // Numbers are from http://pages.cs.wisc.edu/~cao/papers/summary-cache/node8.html + System.out.println("testGetK"); + BloomFilter instance = null; + + instance = new BloomFilter(2, 1); + assertEquals(1, instance.getK()); + + instance = new BloomFilter(3, 1); + assertEquals(2, instance.getK()); + + instance = new BloomFilter(4, 1); + assertEquals(3, instance.getK()); + + instance = new BloomFilter(5, 1); + assertEquals(3, instance.getK()); + + instance = new BloomFilter(6, 1); + assertEquals(4, instance.getK()); + + instance = new BloomFilter(7, 1); + assertEquals(5, instance.getK()); + + instance = new BloomFilter(8, 1); + assertEquals(6, instance.getK()); + + instance = new BloomFilter(9, 1); + assertEquals(6, instance.getK()); + + instance = new BloomFilter(10, 1); + assertEquals(7, instance.getK()); + + instance = new BloomFilter(11, 1); + assertEquals(8, instance.getK()); + + instance = new BloomFilter(12, 1); + assertEquals(8, instance.getK()); + } + + /** + * Test of contains method, of class BloomFilter. + */ + @Test + public void testContains_GenericType() throws UnsupportedEncodingException { + System.out.println("contains"); + int items = 100; + BloomFilter instance = new BloomFilter(0.01, items); + + for (int i = 0; i < items; i++) { + byte[] b = UUID.randomUUID().toString().getBytes("UTF-8"); + instance.add(b); + assertTrue(instance.contains(b)); + } + } + + /** + * Test of contains method, of class BloomFilter. + */ + @Test + public void testContains_byteArr() { + System.out.println("contains"); + + int items = 100; + BloomFilter instance = new BloomFilter(0.01, items); + + for (int i = 0; i < items; i++) { + byte[] bytes = new byte[500]; + r.nextBytes(bytes); + instance.add(bytes); + assertTrue(instance.contains(bytes)); + } + } + + /** + * Test of count method, of class BloomFilter. + */ + @Test + public void testCount() throws UnsupportedEncodingException { + System.out.println("count"); + int expResult = 100; + BloomFilter instance = new BloomFilter(0.01, expResult); + for (int i = 0; i < expResult; i++) { + byte[] bytes = new byte[100]; + r.nextBytes(bytes); + instance.add(bytes); + } + int result = instance.count(); + assertEquals(expResult, result); + + instance = new BloomFilter(0.01, expResult); + for (int i = 0; i < expResult; i++) { + instance.add(UUID.randomUUID().toString().getBytes("UTF-8")); + } + result = instance.count(); + assertEquals(expResult, result); + } +} diff --git a/ethereumj-core/src/test/java/org/ethereum/net/MessageRoundtripTest.java b/ethereumj-core/src/test/java/org/ethereum/net/MessageRoundtripTest.java new file mode 100644 index 0000000000..1fca941bc3 --- /dev/null +++ b/ethereumj-core/src/test/java/org/ethereum/net/MessageRoundtripTest.java @@ -0,0 +1,46 @@ +package org.ethereum.net; + +import org.ethereum.net.p2p.P2pMessage; +import org.junit.Test; + +import static org.junit.Assert.*; +import static org.mockito.Mockito.mock; + +/** + * @author alexbraz + * @since 29/03/2019 + */ +public class MessageRoundtripTest { + + @Test + public void testMessageNotAnswered(){ + P2pMessage message = mock(P2pMessage.class); + MessageRoundtrip messageRoundtrip = new MessageRoundtrip(message); + assertFalse(messageRoundtrip.isAnswered()); + } + + @Test + public void testMessageRetryTimes(){ + P2pMessage message = mock(P2pMessage.class); + MessageRoundtrip messageRoundtrip = new MessageRoundtrip(message); + messageRoundtrip.incRetryTimes(); + assertEquals(1L, messageRoundtrip.retryTimes); + } + + @Test + public void testMessageAnswered(){ + P2pMessage message = mock(P2pMessage.class); + MessageRoundtrip messageRoundtrip = new MessageRoundtrip(message); + messageRoundtrip.answer(); + assertTrue(messageRoundtrip.isAnswered()); + } + + @Test + public void testHasToRetry(){ + P2pMessage message = mock(P2pMessage.class); + MessageRoundtrip messageRoundtrip = new MessageRoundtrip(message); + messageRoundtrip.saveTime(); + assertFalse(messageRoundtrip.hasToRetry()); + } + +}