From 068dcba12285013602a82cf8bc13adf280420693 Mon Sep 17 00:00:00 2001 From: Mike Hearn Date: Mon, 2 May 2011 12:35:10 +0000 Subject: [PATCH] Switch to using SLF4J + the simple logger, allowing people to integrate BitCoinJ with whatever logging system they are already using. Resolves issue 16. --- lib/slf4j-api-1.6.1.jar | Bin 0 -> 25496 bytes lib/slf4j-simple-1.6.1.jar | Bin 0 -> 7669 bytes src/com/google/bitcoin/core/Block.java | 8 +++- src/com/google/bitcoin/core/BlockChain.java | 37 ++++++++------- .../google/bitcoin/core/DiskBlockStore.java | 13 ++++-- src/com/google/bitcoin/core/DnsDiscovery.java | 7 ++- src/com/google/bitcoin/core/IrcDiscovery.java | 7 ++- .../bitcoin/core/NetworkConnection.java | 25 +++++++--- src/com/google/bitcoin/core/Peer.java | 17 +++---- src/com/google/bitcoin/core/Script.java | 22 +++++---- src/com/google/bitcoin/core/Transaction.java | 6 ++- src/com/google/bitcoin/core/Utils.java | 16 ------- src/com/google/bitcoin/core/Wallet.java | 43 ++++++++++-------- tests/com/google/bitcoin/core/BlockTest.java | 6 ++- 14 files changed, 118 insertions(+), 89 deletions(-) create mode 100644 lib/slf4j-api-1.6.1.jar create mode 100644 lib/slf4j-simple-1.6.1.jar diff --git a/lib/slf4j-api-1.6.1.jar b/lib/slf4j-api-1.6.1.jar new file mode 100644 index 0000000000000000000000000000000000000000..42e0ad0de7773da9b94b12f503deda7f5a506015 GIT binary patch literal 25496 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 literal 0 HcmV?d00001 diff --git a/src/com/google/bitcoin/core/Block.java b/src/com/google/bitcoin/core/Block.java index 4dbc734de..6ac534cee 100644 --- a/src/com/google/bitcoin/core/Block.java +++ b/src/com/google/bitcoin/core/Block.java @@ -22,6 +22,9 @@ import java.io.OutputStream; import java.math.BigInteger; import java.util.*; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + import static com.google.bitcoin.core.Utils.*; /** @@ -33,6 +36,7 @@ import static com.google.bitcoin.core.Utils.*; * or request one specifically using {@link Peer#getBlock(byte[])}, or grab one from a downloaded {@link BlockChain}. */ public class Block extends Message { + private static final Logger log = LoggerFactory.getLogger(Block.class); private static final long serialVersionUID = 2738848929966035281L; /** How many bytes are required to represent a block header. */ @@ -263,8 +267,8 @@ public class Block extends Message { List tree = buildMerkleTree(); byte[] calculatedRoot = tree.get(tree.size() - 1); if (!Arrays.equals(calculatedRoot, merkleRoot)) { - LOG("Merkle tree did not verify: "); - for (byte[] b : tree) LOG(Utils.bytesToHexString(b)); + log.error("Merkle tree did not verify: "); + for (byte[] b : tree) log.error(Utils.bytesToHexString(b)); throw new VerificationException("Merkle hashes do not match: " + bytesToHexString(calculatedRoot) + " vs " + bytesToHexString(merkleRoot)); diff --git a/src/com/google/bitcoin/core/BlockChain.java b/src/com/google/bitcoin/core/BlockChain.java index 539ad0b91..c734c3cbe 100644 --- a/src/com/google/bitcoin/core/BlockChain.java +++ b/src/com/google/bitcoin/core/BlockChain.java @@ -19,7 +19,8 @@ package com.google.bitcoin.core; import java.math.BigInteger; import java.util.*; -import static com.google.bitcoin.core.Utils.LOG; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; /** * A BlockChain holds a series of {@link Block} objects, links them together, and knows how to verify that the @@ -46,6 +47,8 @@ import static com.google.bitcoin.core.Utils.LOG; * or if we connect to a peer that doesn't send us blocks in order. */ public class BlockChain { + private static final Logger log = LoggerFactory.getLogger(BlockChain.class); + /** Keeps a map of block hashes to StoredBlocks. */ protected BlockStore blockStore; @@ -78,7 +81,7 @@ public class BlockChain { try { this.blockStore = blockStore; chainHead = blockStore.getChainHead(); - LOG("chain head is: " + chainHead.getHeader().toString()); + log.info("chain head is:\n{}", chainHead.getHeader()); } catch (BlockStoreException e) { throw new RuntimeException(e); } @@ -103,9 +106,9 @@ public class BlockChain { private synchronized boolean add(Block block, boolean tryConnecting) throws BlockStoreException, VerificationException, ScriptException { - LOG("Adding block " + block.getHashAsString() + " to the chain"); + log.info("Adding block " + block.getHashAsString() + " to the chain"); if (blockStore.get(block.getHash()) != null) { - LOG("Already have block"); + log.info("Already have block"); return true; } @@ -113,8 +116,8 @@ public class BlockChain { try { block.verify(); } catch (VerificationException e) { - LOG("Failed to verify block: " + e.toString()); - LOG(block.toString()); + log.error("Failed to verify block:", e); + log.error(block.toString()); throw e; } @@ -125,7 +128,7 @@ public class BlockChain { // We can't find the previous block. Probably we are still in the process of downloading the chain and a // block was solved whilst we were doing it. We put it to one side and try to connect it later when we // have more blocks. - LOG("Block does not connect: " + block.getHashAsString()); + log.warn("Block does not connect: {}", block.getHashAsString()); unconnectedBlocks.add(block); return false; } else { @@ -152,7 +155,7 @@ public class BlockChain { if (storedPrev.equals(chainHead)) { // This block connects to the best known block, it is a normal continuation of the system. setChainHead(newStoredBlock); - LOG("Chain is now " + chainHead.getHeight() + " blocks high"); + log.info("Chain is now {} blocks high", chainHead.getHeight()); if (newTransactions != null) sendTransactionsToWallet(newStoredBlock, NewBlockType.BEST_CHAIN, newTransactions); } else { @@ -162,9 +165,9 @@ public class BlockChain { // to become the new best chain head. This simplifies handling of the re-org in the Wallet class. boolean causedSplit = newStoredBlock.moreWorkThan(chainHead); if (causedSplit) { - LOG("Block is causing a re-organize"); + log.info("Block is causing a re-organize"); } else { - LOG("Block forks the chain, but it did not cause a reorganize."); + log.info("Block forks the chain, but it did not cause a reorganize."); } // We may not have any transactions if we received only a header. That never happens today but will in @@ -187,10 +190,10 @@ public class BlockChain { // Firstly, calculate the block at which the chain diverged. We only need to examine the // chain from beyond this block to find differences. StoredBlock splitPoint = findSplit(newChainHead, chainHead); - LOG("Re-organize after split at height " + splitPoint.getHeight()); - LOG("Old chain head: " + chainHead.getHeader().getHashAsString()); - LOG("New chain head: " + newChainHead.getHeader().getHashAsString()); - LOG("Split at block: " + splitPoint.getHeader().getHashAsString()); + log.info("Re-organize after split at height {}", splitPoint.getHeight()); + log.info("Old chain head: {}", chainHead.getHeader().getHashAsString()); + log.info("New chain head: {}", newChainHead.getHeader().getHashAsString()); + log.info("Split at block: {}", splitPoint.getHeader().getHashAsString()); // Then build a list of all blocks in the old part of the chain and the new part. Set oldBlocks = getPartialChain(chainHead, splitPoint); Set newBlocks = getPartialChain(newChainHead, splitPoint); @@ -267,7 +270,7 @@ public class BlockChain { } catch (ScriptException e) { // We don't want scripts we don't understand to break the block chain, // so just note that this tx was not scanned here and continue. - LOG("Failed to parse a script: " + e.toString()); + log.warn("Failed to parse a script: " + e.toString()); } } } @@ -307,7 +310,7 @@ public class BlockChain { blocksConnectedThisRound++; } if (blocksConnectedThisRound > 0) { - LOG("Connected " + blocksConnectedThisRound + " floating blocks."); + log.info("Connected {} floating blocks.", blocksConnectedThisRound); } } while (blocksConnectedThisRound > 0); } @@ -353,7 +356,7 @@ public class BlockChain { newDifficulty = newDifficulty.divide(BigInteger.valueOf(params.targetTimespan)); if (newDifficulty.compareTo(params.proofOfWorkLimit) > 0) { - LOG("Difficulty hit proof of work limit: " + newDifficulty.toString(16)); + log.warn("Difficulty hit proof of work limit: {}", newDifficulty.toString(16)); newDifficulty = params.proofOfWorkLimit; } diff --git a/src/com/google/bitcoin/core/DiskBlockStore.java b/src/com/google/bitcoin/core/DiskBlockStore.java index 609882863..33eb04a5a 100644 --- a/src/com/google/bitcoin/core/DiskBlockStore.java +++ b/src/com/google/bitcoin/core/DiskBlockStore.java @@ -21,13 +21,16 @@ import java.nio.ByteBuffer; import java.util.HashMap; import java.util.Map; -import static com.google.bitcoin.core.Utils.LOG; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; /** * Stores the block chain to disk but still holds it in memory. This is intended for desktop apps and tests. * Constrained environments like mobile phones probably won't want to or be able to store all the block headers in RAM. */ public class DiskBlockStore implements BlockStore { + private static final Logger log = LoggerFactory.getLogger(DiskBlockStore.class); + private FileOutputStream stream; private Map blockMap; private Sha256Hash chainHead; @@ -40,7 +43,7 @@ public class DiskBlockStore implements BlockStore { load(file); stream = new FileOutputStream(file, true); // Do append. } catch (IOException e) { - LOG(e.toString()); + log.error("failed to load block store from file", e); createNewStore(params, file); } } @@ -70,7 +73,7 @@ public class DiskBlockStore implements BlockStore { } private void load(File file) throws IOException, BlockStoreException { - LOG("Reading block store from " + file.getAbsolutePath()); + log.info("Reading block store from {}", file); InputStream input = new BufferedInputStream(new FileInputStream(file)); // Read a version byte. int version = input.read(); @@ -85,7 +88,7 @@ public class DiskBlockStore implements BlockStore { byte[] chainHeadHash = new byte[32]; input.read(chainHeadHash); this.chainHead = new Sha256Hash(chainHeadHash); - LOG("Read chain head from disk: " + this.chainHead); + log.info("Read chain head from disk: {}", this.chainHead); long now = System.currentTimeMillis(); // Rest of file is raw block headers. byte[] headerBytes = new byte[Block.HEADER_SIZE]; @@ -126,7 +129,7 @@ public class DiskBlockStore implements BlockStore { throw new BlockStoreException(e); } long elapsed = System.currentTimeMillis() - now; - LOG("Block chain read complete in " + elapsed + "ms"); + log.info("Block chain read complete in {}ms", elapsed); } public synchronized void put(StoredBlock block) throws BlockStoreException { diff --git a/src/com/google/bitcoin/core/DnsDiscovery.java b/src/com/google/bitcoin/core/DnsDiscovery.java index 7fe76a925..568958df3 100644 --- a/src/com/google/bitcoin/core/DnsDiscovery.java +++ b/src/com/google/bitcoin/core/DnsDiscovery.java @@ -16,6 +16,9 @@ package com.google.bitcoin.core; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + import java.net.*; import java.util.HashSet; import java.util.Set; @@ -31,6 +34,8 @@ import java.util.Vector; * However, if all hosts passed fail to resolve a PeerDiscoveryException will be thrown during getPeers(). */ public class DnsDiscovery implements PeerDiscovery { + private static final Logger log = LoggerFactory.getLogger(DnsDiscovery.class); + private String[] hostNames; private NetworkParameters netParams; private static final String[] defaultHosts = new String[] {"bitseed.xf2.org","bitseed.bitcoin.org.uk"}; @@ -83,7 +88,7 @@ public class DnsDiscovery implements PeerDiscovery { } } catch (Exception e) { failedLookups++; - Utils.LOG("DNS lookup for " + hostName + " failed."); + log.info("DNS lookup for " + hostName + " failed."); if (failedLookups == hostNames.length) { // All the lookups failed. diff --git a/src/com/google/bitcoin/core/IrcDiscovery.java b/src/com/google/bitcoin/core/IrcDiscovery.java index baebe879e..c5cda6ebe 100644 --- a/src/com/google/bitcoin/core/IrcDiscovery.java +++ b/src/com/google/bitcoin/core/IrcDiscovery.java @@ -16,6 +16,9 @@ package com.google.bitcoin.core; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + import java.io.*; import java.net.*; import java.util.*; @@ -24,6 +27,8 @@ import java.util.*; * IrcDiscovery provides a way to find network peers by joining a pre-agreed rendevouz point on the LFnet IRC network. */ public class IrcDiscovery implements PeerDiscovery { + private static final Logger log = LoggerFactory.getLogger(IrcDiscovery.class); + private String channel; private int port = 6667; private String server; @@ -157,7 +162,7 @@ public class IrcDiscovery implements PeerDiscovery { // decodeChecked removes the checksum from the returned bytes. addressBytes = Base58.decodeChecked(user.substring(1)); } catch (AddressFormatException e) { - Utils.LOG("IRC nick does not parse as base58: " + user); + log.warn("IRC nick does not parse as base58: " + user); continue; } diff --git a/src/com/google/bitcoin/core/NetworkConnection.java b/src/com/google/bitcoin/core/NetworkConnection.java index e7e16e6ba..2cbf1f4bf 100644 --- a/src/com/google/bitcoin/core/NetworkConnection.java +++ b/src/com/google/bitcoin/core/NetworkConnection.java @@ -25,6 +25,9 @@ import java.net.InetSocketAddress; import java.net.Socket; import java.util.Date; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + import static com.google.bitcoin.core.Utils.*; /** @@ -36,6 +39,8 @@ import static com.google.bitcoin.core.Utils.*; * Construction is blocking whilst the protocol version is negotiated. */ public class NetworkConnection { + private static final Logger log = LoggerFactory.getLogger(NetworkConnection.class); + static final int COMMAND_LEN = 12; // Message strings. @@ -94,9 +99,13 @@ public class NetworkConnection { readMessage(); // Switch to the new protocol version. int peerVersion = versionMessage.clientVersion; - LOG(String.format("Connected to peer: version=%d, subVer='%s', services=0x%x, time=%s, blocks=%d", - peerVersion, versionMessage.subVer, - versionMessage.localServices, new Date(versionMessage.time * 1000).toString(), versionMessage.bestHeight)); + log.info("Connected to peer: version={}, subVer='{}', services=0x{}, time={}, blocks={}", new Object[] { + peerVersion, + versionMessage.subVer, + versionMessage.localServices, + new Date(versionMessage.time * 1000), + versionMessage.bestHeight + }); // BitCoinJ is a client mode implementation. That means there's not much point in us talking to other client // mode nodes because we can't download the data from them we need to find/verify transactions. if (!versionMessage.hasBlockChain()) @@ -240,8 +249,11 @@ public class NetworkConnection { } } - if (PROTOCOL_LOG) - LOG("Received " + size + " byte '" + command + "' message: " + Utils.bytesToHexString(payloadBytes)); + log.debug("Received {} byte '{}' message: {}", new Object[]{ + size, + command, + Utils.bytesToHexString(payloadBytes) + }); try { Message message; @@ -283,8 +295,7 @@ public class NetworkConnection { System.arraycopy(hash, 0, header, 4 + COMMAND_LEN + 4, 4); } - if (PROTOCOL_LOG) - LOG("Sending " + name + " message: " + bytesToHexString(payload)); + log.debug("Sending {} message: {}", name, bytesToHexString(payload)); // Another writeMessage call may be running concurrently. synchronized (out) { diff --git a/src/com/google/bitcoin/core/Peer.java b/src/com/google/bitcoin/core/Peer.java index 39af16a47..83a1ba587 100644 --- a/src/com/google/bitcoin/core/Peer.java +++ b/src/com/google/bitcoin/core/Peer.java @@ -23,7 +23,8 @@ import java.util.LinkedList; import java.util.List; import java.util.concurrent.*; -import static com.google.bitcoin.core.Utils.LOG; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; /** * A Peer handles the high level communication with a BitCoin node. It requires a NetworkConnection to be set up for @@ -31,6 +32,8 @@ import static com.google.bitcoin.core.Utils.LOG; * with the network. All these threads synchronize on the block chain. */ public class Peer { + private static final Logger log = LoggerFactory.getLogger(Peer.class); + private final NetworkConnection conn; private final NetworkParameters params; private Thread thread; @@ -88,13 +91,13 @@ public class Peer { // properly explore the network. } else { // TODO: Handle the other messages we can receive. - LOG("Received unhandled message: " + m.toString()); + log.warn("Received unhandled message: {}", m); } } } catch (Exception e) { if (e instanceof IOException && !running) { // This exception was expected because we are tearing down the socket as part of quitting. - LOG("Shutting down peer thread"); + log.info("Shutting down peer thread"); } else { // We caught an unexpected exception. e.printStackTrace(); @@ -144,12 +147,10 @@ public class Peer { } } catch (VerificationException e) { // We don't want verification failures to kill the thread. - LOG(e.toString()); - e.printStackTrace(); + log.warn("block verification failed", e); } catch (ScriptException e) { // We don't want script failures to kill the thread. - LOG(e.toString()); - e.printStackTrace(); + log.warn("script exception", e); } } @@ -297,7 +298,7 @@ public class Peer { // // So this is a complicated process but it has the advantage that we can download a chain of enormous length // in a relatively stateless manner and with constant/bounded memory usage. - LOG("Peer.blockChainDownload: " + Utils.bytesToHexString(toHash)); + log.info("blockChainDownload({})", Utils.bytesToHexString(toHash)); // TODO: Block locators should be abstracted out rather than special cased here. List blockLocator = new LinkedList(); diff --git a/src/com/google/bitcoin/core/Script.java b/src/com/google/bitcoin/core/Script.java index 338bcc33a..7b6f2104f 100644 --- a/src/com/google/bitcoin/core/Script.java +++ b/src/com/google/bitcoin/core/Script.java @@ -26,7 +26,9 @@ import java.util.ArrayList; import java.util.List; import java.util.Stack; -import static com.google.bitcoin.core.Utils.LOG; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + import static com.google.bitcoin.core.Utils.bytesToHexString; /** @@ -38,6 +40,8 @@ import static com.google.bitcoin.core.Utils.bytesToHexString; * transactions generated by the official client, but non-standard transactions will fail. */ public class Script { + private static Logger log = LoggerFactory.getLogger(Script.class); + // Some constants used for decoding the scripts. public static final int OP_PUSHDATA1 = 76; public static final int OP_PUSHDATA2 = 77; @@ -171,7 +175,7 @@ public class Script { chunks.add(getData(len)); } else if (opcode == OP_PUSHDATA4) { // Read a uint32, then read that many bytes of data. - LOG("PUSHDATA4: Unimplemented"); + log.error("PUSHDATA4: Unimplemented"); } else { chunks.add(new byte[] { (byte) opcode }); } @@ -260,11 +264,11 @@ public class Script { case OP_EQUALVERIFY: opEqualVerify(); break; case OP_CHECKSIG: opCheckSig(context); break; default: - if (tracing) LOG("Unknown/unimplemented opcode: " + opcode); + log.debug("Unknown/unimplemented opcode: {}", opcode); } } else { // Data block, push it onto the stack. - if (tracing) LOG("Push " + Utils.bytesToHexString(chunk)); + log.debug("Push {}", Utils.bytesToHexString(chunk)); stack.add(chunk); } } @@ -276,7 +280,7 @@ public class Script { void logStack() { for (int i = 0; i < stack.size(); i++) { - LOG("Stack[" + i + "]: " + Utils.bytesToHexString(stack.get(i))); + log.debug("Stack[{}]: {}",i , Utils.bytesToHexString(stack.get(i))); } } @@ -305,7 +309,7 @@ public class Script { byte[] sig = new byte[sigAndHashType.length - 1]; System.arraycopy(sigAndHashType, 0, sig, 0, sig.length); - if (tracing) LOG("CHECKSIG: hashtype=" + sigHash.toString() + " anyoneCanPay=" + anyoneCanPay); + log.debug("CHECKSIG: hashtype={} anyoneCanPay={}", sigHash, anyoneCanPay); if (context == null) { // TODO: Fix the unit tests to run scripts in transaction context then remove this. pushBool(true); @@ -327,7 +331,7 @@ public class Script { } private void opEqualVerify() throws ScriptException { - if (tracing) LOG("EQUALVERIFY"); + log.debug("EQUALVERIFY"); byte[] a = stack.pop(); byte[] b = stack.pop(); if (!Arrays.areEqual(a, b)) @@ -340,12 +344,12 @@ public class Script { byte[] buf = stack.pop(); byte[] hash = Utils.sha256hash160(buf); stack.add(hash); - if (tracing) LOG("HASH160: output is " + Utils.bytesToHexString(hash)); + log.debug("HASH160: output is {}", Utils.bytesToHexString(hash)); } /** Duplicates the top item on the stack */ private void opDup() { - if (tracing) LOG("DUP"); + log.debug("DUP"); stack.add(Arrays.clone(stack.lastElement())); } diff --git a/src/com/google/bitcoin/core/Transaction.java b/src/com/google/bitcoin/core/Transaction.java index 16e95d6e4..86dc71574 100644 --- a/src/com/google/bitcoin/core/Transaction.java +++ b/src/com/google/bitcoin/core/Transaction.java @@ -23,6 +23,9 @@ import java.io.Serializable; import java.math.BigInteger; import java.util.*; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + import static com.google.bitcoin.core.Utils.*; /** @@ -35,6 +38,7 @@ import static com.google.bitcoin.core.Utils.*; * or UI purposes. */ public class Transaction extends Message implements Serializable { + private static final Logger log = LoggerFactory.getLogger(Transaction.class); private static final long serialVersionUID = -8567546957352643140L; // These are serialized in both bitcoin and java serialization. @@ -319,7 +323,7 @@ public class Transaction extends Message implements Serializable { // The anyoneCanPay feature isn't used at the moment. boolean anyoneCanPay = false; byte[] hash = hashTransactionForSignature(hashType, anyoneCanPay); - Utils.LOG(" signInputs hash=" + Utils.bytesToHexString(hash)); + log.info(" signInputs hash={}", Utils.bytesToHexString(hash)); // Set the script to empty again for the next input. input.scriptBytes = TransactionInput.EMPTY_ARRAY; diff --git a/src/com/google/bitcoin/core/Utils.java b/src/com/google/bitcoin/core/Utils.java index ea96d09f4..f1d00a9ec 100644 --- a/src/com/google/bitcoin/core/Utils.java +++ b/src/com/google/bitcoin/core/Utils.java @@ -51,12 +51,6 @@ public class Utils { */ public static final BigInteger CENT = new BigInteger("1000000", 10); - private static final boolean logging; - - static { - logging = Boolean.parseBoolean(System.getProperty("bitcoinj.logging", "false")); - } - /** Convert an amount expressed in the way humans are used to into nanocoins. */ public static BigInteger toNanoCoins(int coins, int cents) { assert cents < 100; @@ -195,16 +189,6 @@ public class Utils { return ((bytes[offset] & 0xff) << 8) | bytes[offset + 1] & 0xff; } - static void LOG(String msg) { - // Set this to true to see debug prints from the library. - if (logging) { - System.err.print("BitCoin: "); - System.err.println(msg); - System.err.flush(); - System.out.flush(); - } - } - /** * Calculates RIPEMD160(SHA256(input)). This is used in Address calculations. */ diff --git a/src/com/google/bitcoin/core/Wallet.java b/src/com/google/bitcoin/core/Wallet.java index c0140cc51..b025a6c8d 100644 --- a/src/com/google/bitcoin/core/Wallet.java +++ b/src/com/google/bitcoin/core/Wallet.java @@ -16,11 +16,13 @@ package com.google.bitcoin.core; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + import java.io.*; import java.math.BigInteger; import java.util.*; -import static com.google.bitcoin.core.Utils.LOG; import static com.google.bitcoin.core.Utils.bitcoinValueToFriendlyString; /** @@ -33,6 +35,7 @@ import static com.google.bitcoin.core.Utils.bitcoinValueToFriendlyString; * pull in a potentially large (code-size) third party serialization library.

*/ public class Wallet implements Serializable { + private static final Logger log = LoggerFactory.getLogger(Wallet.class); private static final long serialVersionUID = 2L; // Algorithm for movement of transactions between pools. Outbound tx = us spending coins. Inbound tx = us @@ -203,14 +206,14 @@ public class Wallet implements Serializable { BigInteger valueSentToMe = tx.getValueSentToMe(this); BigInteger valueDifference = valueSentToMe.subtract(valueSentFromMe); - LOG("Wallet: Received tx" + (sideChain ? " on a side chain" :"") + " for " + + log.info("Wallet: Received tx" + (sideChain ? " on a side chain" :"") + " for " + bitcoinValueToFriendlyString(valueDifference) + " BTC"); // If this transaction is already in the wallet we may need to move it into a different pool. At the very // least we need to ensure we're manipulating the canonical object rather than a duplicate. Transaction wtx = null; if ((wtx = pending.remove(txHash)) != null) { - LOG(" <-pending"); + log.info(" <-pending"); // A transaction we created appeared in a block. Probably this is a spend we broadcast that has been // accepted by the network. // @@ -219,19 +222,19 @@ public class Wallet implements Serializable { if (bestChain) { if (valueSentToMe.equals(BigInteger.ZERO)) { // There were no change transactions so this tx is fully spent. - LOG(" ->spent"); + log.info(" ->spent"); boolean alreadyPresent = spent.put(wtx.getHash(), wtx) != null; assert !alreadyPresent : "TX in both pending and spent pools"; } else { // There was change back to us, or this tx was purely a spend back to ourselves (perhaps for // anonymization purposes). - LOG(" ->unspent"); + log.info(" ->unspent"); boolean alreadyPresent = unspent.put(wtx.getHash(), wtx) != null; assert !alreadyPresent : "TX in both pending and unspent pools"; } } else if (sideChain) { // The transaction was accepted on an inactive side chain, but not yet by the best chain. - LOG(" ->inactive"); + log.info(" ->inactive"); // It's OK for this to already be in the inactive pool because there can be multiple independent side // chains in which it appears: // @@ -240,7 +243,7 @@ public class Wallet implements Serializable { // \-> b4 (at this point it's already present in 'inactive' boolean alreadyPresent = inactive.put(wtx.getHash(), wtx) != null; if (alreadyPresent) - LOG("Saw a transaction be incorporated into multiple independent side chains"); + log.info("Saw a transaction be incorporated into multiple independent side chains"); // Put it back into the pending pool, because 'pending' means 'waiting to be included in best chain'. pending.put(wtx.getHash(), wtx); } @@ -250,14 +253,14 @@ public class Wallet implements Serializable { // This TX didn't originate with us. It could be sending us coins and also spending our own coins if keys // are being shared between different wallets. if (sideChain) { - LOG(" ->inactive"); + log.info(" ->inactive"); inactive.put(tx.getHash(), tx); } else if (bestChain) { processTxFromBestChain(tx); } } - LOG("Balance is now: " + bitcoinValueToFriendlyString(getBalance())); + log.info("Balance is now: " + bitcoinValueToFriendlyString(getBalance())); // Inform anyone interested that we have new coins. Note: we may be re-entered by the event listener, // so we must not make assumptions about our state after this loop returns! For example, @@ -281,12 +284,12 @@ public class Wallet implements Serializable { updateForSpends(tx); if (!tx.getValueSentToMe(this).equals(BigInteger.ZERO)) { // It's sending us coins. - LOG(" ->unspent"); + log.info(" ->unspent"); boolean alreadyPresent = unspent.put(tx.getHash(), tx) != null; assert !alreadyPresent : "TX was received twice"; } else { // It spent some of our coins and did not send us any. - LOG(" ->spent"); + log.info(" ->spent"); boolean alreadyPresent = spent.put(tx.getHash(), tx) != null; assert !alreadyPresent : "TX was received twice"; } @@ -302,16 +305,16 @@ public class Wallet implements Serializable { if (input.outpoint.connect(unspent.values())) { TransactionOutput output = input.outpoint.getConnectedOutput(); assert !output.isSpent : "Double spend accepted by the network?"; - LOG(" Saw some of my unspent outputs be spent by someone else who has my keys."); - LOG(" Total spent value is " + bitcoinValueToFriendlyString(output.getValue())); + log.info(" Saw some of my unspent outputs be spent by someone else who has my keys."); + log.info(" Total spent value is " + bitcoinValueToFriendlyString(output.getValue())); output.isSpent = true; Transaction connectedTx = input.outpoint.fromTx; if (connectedTx.getValueSentToMe(this, false).equals(BigInteger.ZERO)) { // There's nothing left I can spend in this transaction. if (unspent.remove(connectedTx.getHash()) != null); - LOG(" prevtx <-unspent"); + log.info(" prevtx <-unspent"); spent.put(connectedTx.getHash(), connectedTx); - LOG(" prevtx ->spent"); + log.info(" prevtx ->spent"); } } } @@ -400,7 +403,7 @@ public class Wallet implements Serializable { * @return a new {@link Transaction} or null if we cannot afford this send. */ synchronized Transaction createSend(Address address, BigInteger nanocoins, Address changeAddress) { - LOG("Creating send tx to " + address.toString() + " for " + + log.info("Creating send tx to " + address.toString() + " for " + bitcoinValueToFriendlyString(nanocoins)); // To send money to somebody else, we need to do gather up transactions with unspent outputs until we have // sufficient value. Many coin selection algorithms are possible, we use a simple but suboptimal one. @@ -418,7 +421,7 @@ public class Wallet implements Serializable { } // Can we afford this? if (valueGathered.compareTo(nanocoins) < 0) { - LOG("Insufficient value in wallet for send, missing " + + log.info("Insufficient value in wallet for send, missing " + bitcoinValueToFriendlyString(nanocoins.subtract(valueGathered))); // TODO: Should throw an exception here. return null; @@ -431,7 +434,7 @@ public class Wallet implements Serializable { // The value of the inputs is greater than what we want to send. Just like in real life then, // we need to take back some coins ... this is called "change". Add another output that sends the change // back to us. - LOG(" with " + bitcoinValueToFriendlyString(change) + " coins change"); + log.info(" with " + bitcoinValueToFriendlyString(change) + " coins change"); sendTx.addOutput(new TransactionOutput(params, change, changeAddress, sendTx)); } for (TransactionOutput output : gathered) { @@ -567,7 +570,7 @@ public class Wallet implements Serializable { // If there is no difference it means we the user doesn't really care about this re-org but we still need to // update the transaction block pointers for next time. boolean affectedUs = !oldChainTransactions.equals(newChainTransactions); - LOG(affectedUs ? "Re-org affected our transactions" : "Re-org had no effect on our transactions"); + log.info(affectedUs ? "Re-org affected our transactions" : "Re-org had no effect on our transactions"); if (!affectedUs) return; // Transactions that were in the old chain but aren't in the new chain. These will become inactive. @@ -579,7 +582,7 @@ public class Wallet implements Serializable { assert !(gone.isEmpty() && fresh.isEmpty()) : "There must have been some changes to get here"; for (Transaction tx : gone) { - LOG("tx not in new chain: <-unspent/spent ->inactive\n" + tx.toString()); + log.info("tx not in new chain: <-unspent/spent ->inactive\n" + tx.toString()); unspent.remove(tx.getHash()); spent.remove(tx.getHash()); inactive.put(tx.getHash(), tx); diff --git a/tests/com/google/bitcoin/core/BlockTest.java b/tests/com/google/bitcoin/core/BlockTest.java index 5b8cd7d36..f31ec03b6 100644 --- a/tests/com/google/bitcoin/core/BlockTest.java +++ b/tests/com/google/bitcoin/core/BlockTest.java @@ -18,6 +18,8 @@ package com.google.bitcoin.core; import com.google.bitcoin.bouncycastle.util.encoders.Hex; import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; @@ -29,7 +31,9 @@ import java.util.Arrays; import static org.junit.Assert.*; public class BlockTest { + private static Logger log = LoggerFactory.getLogger(BlockTest.class); static final NetworkParameters params = NetworkParameters.testNet(); + static final byte[] blockBytes; static { @@ -139,8 +143,6 @@ public class BlockTest { // Note that this will actually check the transactions are equal by doing bitcoin serialization and checking // the bytestreams are the same! A true "deep equals" is not implemented for Transaction. The primary purpose // of this test is to ensure no errors occur during the Java serialization/deserialization process. - Utils.LOG(Utils.bytesToHexString(tx.bitcoinSerialize())); - Utils.LOG(Utils.bytesToHexString(tx2.bitcoinSerialize())); assertEquals(tx, tx2); } }