org.bitcoinj.params.TestNet3Params Maven / Gradle / Ivy
/*
* Copyright 2013 Google Inc.
* Copyright 2014 Andreas Schildbach
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.bitcoinj.params;
import java.math.BigInteger;
import java.util.Date;
import com.google.common.base.Preconditions;
import org.bitcoinj.core.AbstractBlockChain;
import org.bitcoinj.core.Block;
import org.bitcoinj.core.NetworkParameters;
import org.bitcoinj.core.StoredBlock;
import org.bitcoinj.core.Utils;
import org.bitcoinj.core.VerificationException;
import org.bitcoinj.store.BlockStore;
import org.bitcoinj.store.BlockStoreException;
import static com.google.common.base.Preconditions.checkState;
/**
* Parameters for the testnet, a separate public instance of Bitcoin that has relaxed rules suitable for development
* and testing of applications and new Bitcoin versions.
*/
public class TestNet3Params extends AbstractBitcoinNetParams {
public TestNet3Params() {
super();
id = ID_TESTNET;
// Genesis hash is 000000000933ea01ad0ee984209779baaec3ced90fa3f408719526f8d77f4943
packetMagic = 0xf4e5f3f4L;
interval = INTERVAL;
targetTimespan = TARGET_TIMESPAN;
maxTarget = Utils.decodeCompactBits(0x1d00ffffL);
port = 18333;
addressHeader = 111;
p2shHeader = 196;
acceptableAddressCodes = new int[] { addressHeader, p2shHeader };
dumpedPrivateKeyHeader = 239;
genesisBlock.setTime(1296688602L);
genesisBlock.setDifficultyTarget(0x1d00ffffL);
genesisBlock.setNonce(414098458);
spendableCoinbaseDepth = 100;
subsidyDecreaseBlockCount = 210000;
String genesisHash = genesisBlock.getHashAsString();
checkState(genesisHash.equals("000000000933ea01ad0ee984209779baaec3ced90fa3f408719526f8d77f4943"));
alertSigningKey = Utils.HEX.decode("04302390343f91cc401d56d68b123028bf52e5fca1939df127f63c6467cdf9c8e2c14b61104cf817d0b780da337893ecc4aaff1309e536162dabbdb45200ca2b0a");
dnsSeeds = new String[] {
"testnet-seed.bitcoinabc.org",
"testnet-seed-abc.bitcoinforks.org",
"testnet-seed.bitcoinunlimited.info",
"testnet-seed.bitprim.org",
"testnet-seed.deadalnix.me",
"testnet-seeder.criptolayer.net"
};
addrSeeds = null;
bip32HeaderPub = 0x043587CF;
bip32HeaderPriv = 0x04358394;
majorityEnforceBlockUpgrade = TestNet2Params.TESTNET_MAJORITY_ENFORCE_BLOCK_UPGRADE;
majorityRejectBlockOutdated = TestNet2Params.TESTNET_MAJORITY_REJECT_BLOCK_OUTDATED;
majorityWindow = TestNet2Params.TESTNET_MAJORITY_WINDOW;
// Aug, 1 hard fork
uahfHeight = 1155876;
/** Activation time at which the cash HF kicks in. */
cashHardForkActivationTime = 1510600000;
daaHeight = 1188697;
}
private static TestNet3Params instance;
public static synchronized TestNet3Params get() {
if (instance == null) {
instance = new TestNet3Params();
}
return instance;
}
@Override
public String getPaymentProtocolId() {
return PAYMENT_PROTOCOL_ID_TESTNET;
}
// February 16th 2012
private static final Date testnetDiffDate = new Date(1329264000000L);
@Override
public void checkDifficultyTransitions(final StoredBlock storedPrev, final Block nextBlock,
final BlockStore blockStore, AbstractBlockChain blockChain) throws VerificationException, BlockStoreException {
if (storedPrev.getHeight() < daaHeight && !isDifficultyTransitionPoint(storedPrev) && nextBlock.getTime().after(testnetDiffDate)) {
Block prev = storedPrev.getHeader();
// After 15th February 2012 the rules on the testnet change to avoid people running up the difficulty
// and then leaving, making it too hard to mine a block. On non-difficulty transition points, easy
// blocks are allowed if there has been a span of 20 minutes without one.
final long timeDelta = nextBlock.getTimeSeconds() - prev.getTimeSeconds();
// There is an integer underflow bug in bitcoin-qt that means mindiff blocks are accepted when time
// goes backwards.
if (timeDelta >= 0 && timeDelta <= NetworkParameters.TARGET_SPACING * 2) {
// Walk backwards until we find a block that doesn't have the easiest proof of work, then check
// that difficulty is equal to that one.
StoredBlock cursor = storedPrev;
while (!cursor.getHeader().equals(getGenesisBlock()) &&
cursor.getHeight() % getInterval() != 0 &&
cursor.getHeader().getDifficultyTargetAsInteger().equals(getMaxTarget()))
cursor = cursor.getPrev(blockStore);
BigInteger cursorTarget = cursor.getHeader().getDifficultyTargetAsInteger();
BigInteger newTarget = nextBlock.getDifficultyTargetAsInteger();
if (!cursorTarget.equals(newTarget))
throw new VerificationException("Testnet block transition that is not allowed: " +
Long.toHexString(cursor.getHeader().getDifficultyTarget()) + " vs " +
Long.toHexString(nextBlock.getDifficultyTarget()));
}
} else {
super.checkDifficultyTransitions(storedPrev, nextBlock, blockStore, blockChain);
}
}
@Override
protected void checkNextCashWorkRequired(StoredBlock storedPrev,
Block nextBlock, BlockStore blockStore) {
// This cannot handle the genesis block and early blocks in general.
//assert(pindexPrev);
// Compute the difficulty based on the full adjustement interval.
int nHeight = storedPrev.getHeight();
Preconditions.checkState(nHeight >= this.interval);
// Get the last suitable block of the difficulty interval.
try {
// Special difficulty rule for testnet:
// If the new block's timestamp is more than 2* 10 minutes then allow
// mining of a min-difficulty block.
Block prev = storedPrev.getHeader();
final long timeDelta = nextBlock.getTimeSeconds() - prev.getTimeSeconds();
if (timeDelta >= 0 && timeDelta > NetworkParameters.TARGET_SPACING * 2) {
if (!maxTarget.equals(nextBlock.getDifficultyTargetAsInteger()))
throw new VerificationException("Testnet block transition that is not allowed: " +
Long.toHexString(Utils.encodeCompactBits(maxTarget)) + " (required min difficulty) vs " +
Long.toHexString(nextBlock.getDifficultyTarget()));
return;
}
StoredBlock pindexLast = GetSuitableBlock(storedPrev, blockStore);
//assert (pindexLast);
// Get the first suitable block of the difficulty interval.
int nHeightFirst = nHeight - 144;
StoredBlock pindexFirst = storedPrev;
for (int i = 144; i > 0; --i)
{
pindexFirst = pindexFirst.getPrev(blockStore);
if(pindexFirst == null)
return;
}
pindexFirst = GetSuitableBlock(pindexFirst, blockStore);
//assert (pindexFirst);
// Compute the target based on time and work done during the interval.
BigInteger nextTarget =
ComputeTarget(pindexFirst, pindexLast);
verifyDifficulty(nextTarget, nextBlock);
}
catch (BlockStoreException x)
{
//this means we don't have enough blocks, yet. let it go until we do.
return;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy