/* * Copyright 2013 Google Inc. * * 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.protocols.channels; import org.bitcoinj.core.*; import org.bitcoinj.script.Script; import org.bitcoinj.script.ScriptBuilder; import org.bitcoinj.testing.TestWithWallet; import com.google.common.collect.Lists; import com.google.common.util.concurrent.ListenableFuture; import com.google.common.util.concurrent.SettableFuture; import org.junit.After; import org.junit.Before; import org.junit.Test; import java.math.BigInteger; import java.util.Arrays; import java.util.Iterator; import java.util.concurrent.BlockingQueue; import java.util.concurrent.ExecutionException; import java.util.concurrent.LinkedBlockingQueue; import static org.bitcoinj.core.Coin.*; import static org.bitcoinj.testing.FakeTxBuilder.createFakeTx; import static org.bitcoinj.testing.FakeTxBuilder.makeSolvedTestBlock; import static org.junit.Assert.*; public class PaymentChannelStateTest extends TestWithWallet { private ECKey serverKey; private Coin halfCoin; private Wallet serverWallet; private PaymentChannelServerState serverState; private PaymentChannelClientState clientState; private TransactionBroadcaster mockBroadcaster; private BlockingQueue<TxFuturePair> broadcasts; private static class TxFuturePair { Transaction tx; SettableFuture<Transaction> future; public TxFuturePair(Transaction tx, SettableFuture<Transaction> future) { this.tx = tx; this.future = future; } } @Override @Before public void setUp() throws Exception { super.setUp(); wallet.addExtension(new StoredPaymentChannelClientStates(wallet, new TransactionBroadcaster() { @Override public ListenableFuture<Transaction> broadcastTransaction(Transaction tx) { fail(); return null; } })); sendMoneyToWallet(COIN, AbstractBlockChain.NewBlockType.BEST_CHAIN); chain = new BlockChain(params, wallet, blockStore); // Recreate chain as sendMoneyToWallet will confuse it serverWallet = new Wallet(params); serverKey = serverWallet.freshReceiveKey(); chain.addWallet(serverWallet); halfCoin = valueOf(0, 50); broadcasts = new LinkedBlockingQueue<TxFuturePair>(); mockBroadcaster = new TransactionBroadcaster() { @Override public ListenableFuture<Transaction> broadcastTransaction(Transaction tx) { SettableFuture<Transaction> future = SettableFuture.create(); broadcasts.add(new TxFuturePair(tx, future)); return future; } }; } @After @Override public void tearDown() throws Exception { super.tearDown(); } @Test public void stateErrors() throws Exception { PaymentChannelClientState channelState = new PaymentChannelClientState(wallet, myKey, serverKey, COIN.multiply(10), 20); assertEquals(PaymentChannelClientState.State.NEW, channelState.getState()); try { channelState.getMultisigContract(); fail(); } catch (IllegalStateException e) { // Expected. } try { channelState.initiate(); fail(); } catch (InsufficientMoneyException e) { } } @Test public void basic() throws Exception { // Check it all works when things are normal (no attacks, no problems). Utils.setMockClock(); // Use mock clock final long EXPIRE_TIME = Utils.currentTimeSeconds() + 60*60*24; serverState = new PaymentChannelServerState(mockBroadcaster, serverWallet, serverKey, EXPIRE_TIME); assertEquals(PaymentChannelServerState.State.WAITING_FOR_REFUND_TRANSACTION, serverState.getState()); clientState = new PaymentChannelClientState(wallet, myKey, ECKey.fromPublicOnly(serverKey.getPubKey()), halfCoin, EXPIRE_TIME); assertEquals(PaymentChannelClientState.State.NEW, clientState.getState()); clientState.initiate(); assertEquals(PaymentChannelClientState.State.INITIATED, clientState.getState()); // Send the refund tx from client to server and get back the signature. Transaction refund = new Transaction(params, clientState.getIncompleteRefundTransaction().bitcoinSerialize()); byte[] refundSig = serverState.provideRefundTransaction(refund, myKey.getPubKey()); assertEquals(PaymentChannelServerState.State.WAITING_FOR_MULTISIG_CONTRACT, serverState.getState()); // This verifies that the refund can spend the multi-sig output when run. clientState.provideRefundSignature(refundSig); assertEquals(PaymentChannelClientState.State.SAVE_STATE_IN_WALLET, clientState.getState()); clientState.fakeSave(); assertEquals(PaymentChannelClientState.State.PROVIDE_MULTISIG_CONTRACT_TO_SERVER, clientState.getState()); // Validate the multisig contract looks right. Transaction multisigContract = new Transaction(params, clientState.getMultisigContract().bitcoinSerialize()); assertEquals(PaymentChannelClientState.State.READY, clientState.getState()); assertEquals(2, multisigContract.getOutputs().size()); // One multi-sig, one change. Script script = multisigContract.getOutput(0).getScriptPubKey(); assertTrue(script.isSentToMultiSig()); script = multisigContract.getOutput(1).getScriptPubKey(); assertTrue(script.isSentToAddress()); assertTrue(wallet.getPendingTransactions().contains(multisigContract)); // Provide the server with the multisig contract and simulate successful propagation/acceptance. serverState.provideMultiSigContract(multisigContract); assertEquals(PaymentChannelServerState.State.WAITING_FOR_MULTISIG_ACCEPTANCE, serverState.getState()); final TxFuturePair pair = broadcasts.take(); pair.future.set(pair.tx); assertEquals(PaymentChannelServerState.State.READY, serverState.getState()); // Make sure the refund transaction is not in the wallet and multisig contract's output is not connected to it assertEquals(2, wallet.getTransactions(false).size()); Iterator<Transaction> walletTransactionIterator = wallet.getTransactions(false).iterator(); Transaction clientWalletMultisigContract = walletTransactionIterator.next(); assertFalse(clientWalletMultisigContract.getHash().equals(clientState.getCompletedRefundTransaction().getHash())); if (!clientWalletMultisigContract.getHash().equals(multisigContract.getHash())) { clientWalletMultisigContract = walletTransactionIterator.next(); assertFalse(clientWalletMultisigContract.getHash().equals(clientState.getCompletedRefundTransaction().getHash())); } else assertFalse(walletTransactionIterator.next().getHash().equals(clientState.getCompletedRefundTransaction().getHash())); assertEquals(multisigContract.getHash(), clientWalletMultisigContract.getHash()); assertFalse(clientWalletMultisigContract.getInput(0).getConnectedOutput().getSpentBy().getParentTransaction().getHash().equals(refund.getHash())); // Both client and server are now in the ready state. Simulate a few micropayments of 0.005 bitcoins. Coin size = halfCoin.divide(100); Coin totalPayment = Coin.ZERO; for (int i = 0; i < 4; i++) { byte[] signature = clientState.incrementPaymentBy(size).signature.encodeToBitcoin(); totalPayment = totalPayment.add(size); serverState.incrementPayment(halfCoin.subtract(totalPayment), signature); } // Now confirm the contract transaction and make sure payments still work chain.add(makeSolvedTestBlock(blockStore.getChainHead().getHeader(), multisigContract)); byte[] signature = clientState.incrementPaymentBy(size).signature.encodeToBitcoin(); totalPayment = totalPayment.add(size); serverState.incrementPayment(halfCoin.subtract(totalPayment), signature); // And settle the channel. serverState.close(); assertEquals(PaymentChannelServerState.State.CLOSING, serverState.getState()); final TxFuturePair pair2 = broadcasts.take(); Transaction closeTx = pair2.tx; pair2.future.set(closeTx); final Transaction reserializedCloseTx = new Transaction(params, closeTx.bitcoinSerialize()); assertEquals(PaymentChannelServerState.State.CLOSED, serverState.getState()); // ... and on the client side. wallet.receivePending(reserializedCloseTx, null); assertEquals(PaymentChannelClientState.State.CLOSED, clientState.getState()); // Create a block with the payment transaction in it and give it to both wallets chain.add(makeSolvedTestBlock(blockStore.getChainHead().getHeader(), reserializedCloseTx)); assertEquals(size.multiply(5), serverWallet.getBalance()); assertEquals(0, serverWallet.getPendingTransactions().size()); assertEquals(COIN.subtract(size.multiply(5)), wallet.getBalance()); assertEquals(0, wallet.getPendingTransactions().size()); assertEquals(3, wallet.getTransactions(false).size()); walletTransactionIterator = wallet.getTransactions(false).iterator(); Transaction clientWalletCloseTransaction = walletTransactionIterator.next(); if (!clientWalletCloseTransaction.getHash().equals(closeTx.getHash())) clientWalletCloseTransaction = walletTransactionIterator.next(); if (!clientWalletCloseTransaction.getHash().equals(closeTx.getHash())) clientWalletCloseTransaction = walletTransactionIterator.next(); assertEquals(closeTx.getHash(), clientWalletCloseTransaction.getHash()); assertNotNull(clientWalletCloseTransaction.getInput(0).getConnectedOutput()); } @Test public void setupDoS() throws Exception { // Check that if the other side stops after we have provided a signed multisig contract, that after a timeout // we can broadcast the refund and get our balance back. // Spend the client wallet's one coin Transaction spendCoinTx = wallet.sendCoinsOffline(Wallet.SendRequest.to(new ECKey().toAddress(params), COIN)); assertEquals(Coin.ZERO, wallet.getBalance()); chain.add(makeSolvedTestBlock(blockStore.getChainHead().getHeader(), spendCoinTx, createFakeTx(params, CENT, myAddress))); assertEquals(CENT, wallet.getBalance()); // Set the wallet's stored states to use our real test PeerGroup StoredPaymentChannelClientStates stateStorage = new StoredPaymentChannelClientStates(wallet, mockBroadcaster); wallet.addOrUpdateExtension(stateStorage); Utils.setMockClock(); // Use mock clock final long EXPIRE_TIME = Utils.currentTimeMillis()/1000 + 60*60*24; serverState = new PaymentChannelServerState(mockBroadcaster, serverWallet, serverKey, EXPIRE_TIME); assertEquals(PaymentChannelServerState.State.WAITING_FOR_REFUND_TRANSACTION, serverState.getState()); clientState = new PaymentChannelClientState(wallet, myKey, ECKey.fromPublicOnly(serverKey.getPubKey()), CENT.divide(2), EXPIRE_TIME); assertEquals(PaymentChannelClientState.State.NEW, clientState.getState()); assertEquals(CENT.divide(2), clientState.getTotalValue()); clientState.initiate(); // We will have to pay min_tx_fee twice - both the multisig contract and the refund tx assertEquals(clientState.getRefundTxFees(), Transaction.REFERENCE_DEFAULT_MIN_TX_FEE.multiply(2)); assertEquals(PaymentChannelClientState.State.INITIATED, clientState.getState()); // Send the refund tx from client to server and get back the signature. Transaction refund = new Transaction(params, clientState.getIncompleteRefundTransaction().bitcoinSerialize()); byte[] refundSig = serverState.provideRefundTransaction(refund, myKey.getPubKey()); assertEquals(PaymentChannelServerState.State.WAITING_FOR_MULTISIG_CONTRACT, serverState.getState()); // This verifies that the refund can spend the multi-sig output when run. clientState.provideRefundSignature(refundSig); assertEquals(PaymentChannelClientState.State.SAVE_STATE_IN_WALLET, clientState.getState()); clientState.fakeSave(); assertEquals(PaymentChannelClientState.State.PROVIDE_MULTISIG_CONTRACT_TO_SERVER, clientState.getState()); // Validate the multisig contract looks right. Transaction multisigContract = new Transaction(params, clientState.getMultisigContract().bitcoinSerialize()); assertEquals(PaymentChannelClientState.State.READY, clientState.getState()); assertEquals(2, multisigContract.getOutputs().size()); // One multi-sig, one change. Script script = multisigContract.getOutput(0).getScriptPubKey(); assertTrue(script.isSentToMultiSig()); script = multisigContract.getOutput(1).getScriptPubKey(); assertTrue(script.isSentToAddress()); assertTrue(wallet.getPendingTransactions().contains(multisigContract)); // Provide the server with the multisig contract and simulate successful propagation/acceptance. serverState.provideMultiSigContract(multisigContract); assertEquals(PaymentChannelServerState.State.WAITING_FOR_MULTISIG_ACCEPTANCE, serverState.getState()); final TxFuturePair pop = broadcasts.take(); pop.future.set(pop.tx); assertEquals(PaymentChannelServerState.State.READY, serverState.getState()); // Pay a tiny bit serverState.incrementPayment(CENT.divide(2).subtract(CENT.divide(10)), clientState.incrementPaymentBy(CENT.divide(10)).signature.encodeToBitcoin()); // Advance time until our we get close enough to lock time that server should rebroadcast Utils.rollMockClock(60*60*22); // ... and store server to get it to broadcast payment transaction serverState.storeChannelInWallet(null); TxFuturePair broadcastPaymentPair = broadcasts.take(); Exception paymentException = new RuntimeException("I'm sorry, but the network really just doesn't like you"); broadcastPaymentPair.future.setException(paymentException); try { serverState.close().get(); } catch (ExecutionException e) { assertSame(e.getCause(), paymentException); } assertEquals(PaymentChannelServerState.State.ERROR, serverState.getState()); // Now advance until client should rebroadcast Utils.rollMockClock(60 * 60 * 2 + 60 * 5); // Now store the client state in a stored state object which handles the rebroadcasting clientState.doStoreChannelInWallet(Sha256Hash.create(new byte[]{})); TxFuturePair clientBroadcastedMultiSig = broadcasts.take(); TxFuturePair broadcastRefund = broadcasts.take(); assertEquals(clientBroadcastedMultiSig.tx.getHash(), multisigContract.getHash()); for (TransactionInput input : clientBroadcastedMultiSig.tx.getInputs()) input.verify(); clientBroadcastedMultiSig.future.set(clientBroadcastedMultiSig.tx); Transaction clientBroadcastedRefund = broadcastRefund.tx; assertEquals(clientBroadcastedRefund.getHash(), clientState.getCompletedRefundTransaction().getHash()); for (TransactionInput input : clientBroadcastedRefund.getInputs()) { // If the multisig output is connected, the wallet will fail to deserialize if (input.getOutpoint().getHash().equals(clientBroadcastedMultiSig.tx.getHash())) assertNull(input.getConnectedOutput().getSpentBy()); input.verify(clientBroadcastedMultiSig.tx.getOutput(0)); } broadcastRefund.future.set(clientBroadcastedRefund); // Create a block with multisig contract and refund transaction in it and give it to both wallets, // making getBalance() include the transactions chain.add(makeSolvedTestBlock(blockStore.getChainHead().getHeader(), multisigContract,clientBroadcastedRefund)); // Make sure we actually had to pay what initialize() told us we would assertEquals(wallet.getBalance(), CENT.subtract(Transaction.REFERENCE_DEFAULT_MIN_TX_FEE.multiply(2))); try { // After its expired, we cant still increment payment clientState.incrementPaymentBy(CENT); fail(); } catch (IllegalStateException e) { } } @Test public void checkBadData() throws Exception { // Check that if signatures/transactions/etc are corrupted, the protocol rejects them correctly. // We'll broadcast only one tx: multisig contract Utils.setMockClock(); // Use mock clock final long EXPIRE_TIME = Utils.currentTimeSeconds() + 60*60*24; serverState = new PaymentChannelServerState(mockBroadcaster, serverWallet, serverKey, EXPIRE_TIME); assertEquals(PaymentChannelServerState.State.WAITING_FOR_REFUND_TRANSACTION, serverState.getState()); clientState = new PaymentChannelClientState(wallet, myKey, ECKey.fromPublicOnly(serverKey.getPubKey()), halfCoin, EXPIRE_TIME); assertEquals(PaymentChannelClientState.State.NEW, clientState.getState()); clientState.initiate(); assertEquals(PaymentChannelClientState.State.INITIATED, clientState.getState()); // Test refund transaction with any number of issues byte[] refundTxBytes = clientState.getIncompleteRefundTransaction().bitcoinSerialize(); Transaction refund = new Transaction(params, refundTxBytes); refund.addOutput(Coin.ZERO, new ECKey().toAddress(params)); try { serverState.provideRefundTransaction(refund, myKey.getPubKey()); fail(); } catch (VerificationException e) {} refund = new Transaction(params, refundTxBytes); refund.addInput(new TransactionInput(params, refund, new byte[] {}, new TransactionOutPoint(params, 42, refund.getHash()))); try { serverState.provideRefundTransaction(refund, myKey.getPubKey()); fail(); } catch (VerificationException e) {} refund = new Transaction(params, refundTxBytes); refund.setLockTime(0); try { serverState.provideRefundTransaction(refund, myKey.getPubKey()); fail(); } catch (VerificationException e) {} refund = new Transaction(params, refundTxBytes); refund.getInput(0).setSequenceNumber(TransactionInput.NO_SEQUENCE); try { serverState.provideRefundTransaction(refund, myKey.getPubKey()); fail(); } catch (VerificationException e) {} refund = new Transaction(params, refundTxBytes); byte[] refundSig = serverState.provideRefundTransaction(refund, myKey.getPubKey()); try { serverState.provideRefundTransaction(refund, myKey.getPubKey()); fail(); } catch (IllegalStateException e) {} assertEquals(PaymentChannelServerState.State.WAITING_FOR_MULTISIG_CONTRACT, serverState.getState()); byte[] refundSigCopy = Arrays.copyOf(refundSig, refundSig.length); refundSigCopy[refundSigCopy.length-1] = (byte) (Transaction.SigHash.NONE.ordinal() + 1); try { clientState.provideRefundSignature(refundSigCopy); fail(); } catch (VerificationException e) { assertTrue(e.getMessage().contains("SIGHASH_NONE")); } refundSigCopy = Arrays.copyOf(refundSig, refundSig.length); refundSigCopy[3] ^= 0x42; // Make the signature fail standard checks try { clientState.provideRefundSignature(refundSigCopy); fail(); } catch (VerificationException e) { assertTrue(e.getMessage().contains("not canonical")); } refundSigCopy = Arrays.copyOf(refundSig, refundSig.length); refundSigCopy[10] ^= 0x42; // Flip some random bits in the signature (to make it invalid, not just nonstandard) try { clientState.provideRefundSignature(refundSigCopy); fail(); } catch (VerificationException e) { assertFalse(e.getMessage().contains("not canonical")); } refundSigCopy = Arrays.copyOf(refundSig, refundSig.length); try { clientState.getCompletedRefundTransaction(); fail(); } catch (IllegalStateException e) {} clientState.provideRefundSignature(refundSigCopy); try { clientState.provideRefundSignature(refundSigCopy); fail(); } catch (IllegalStateException e) {} assertEquals(PaymentChannelClientState.State.SAVE_STATE_IN_WALLET, clientState.getState()); clientState.fakeSave(); assertEquals(PaymentChannelClientState.State.PROVIDE_MULTISIG_CONTRACT_TO_SERVER, clientState.getState()); try { clientState.incrementPaymentBy(Coin.SATOSHI); fail(); } catch (IllegalStateException e) {} byte[] multisigContractSerialized = clientState.getMultisigContract().bitcoinSerialize(); Transaction multisigContract = new Transaction(params, multisigContractSerialized); multisigContract.clearOutputs(); multisigContract.addOutput(halfCoin, ScriptBuilder.createMultiSigOutputScript(2, Lists.newArrayList(serverKey, myKey))); try { serverState.provideMultiSigContract(multisigContract); fail(); } catch (VerificationException e) { assertTrue(e.getMessage().contains("client and server in that order")); } multisigContract = new Transaction(params, multisigContractSerialized); multisigContract.clearOutputs(); multisigContract.addOutput(Coin.ZERO, ScriptBuilder.createMultiSigOutputScript(2, Lists.newArrayList(myKey, serverKey))); try { serverState.provideMultiSigContract(multisigContract); fail(); } catch (VerificationException e) { assertTrue(e.getMessage().contains("zero value")); } multisigContract = new Transaction(params, multisigContractSerialized); multisigContract.clearOutputs(); multisigContract.addOutput(new TransactionOutput(params, multisigContract, halfCoin, new byte[] {0x01})); try { serverState.provideMultiSigContract(multisigContract); fail(); } catch (VerificationException e) {} multisigContract = new Transaction(params, multisigContractSerialized); ListenableFuture<PaymentChannelServerState> multisigStateFuture = serverState.provideMultiSigContract(multisigContract); try { serverState.provideMultiSigContract(multisigContract); fail(); } catch (IllegalStateException e) {} assertEquals(PaymentChannelServerState.State.WAITING_FOR_MULTISIG_ACCEPTANCE, serverState.getState()); assertFalse(multisigStateFuture.isDone()); final TxFuturePair pair = broadcasts.take(); pair.future.set(pair.tx); assertEquals(multisigStateFuture.get(), serverState); assertEquals(PaymentChannelServerState.State.READY, serverState.getState()); // Both client and server are now in the ready state. Simulate a few micropayments of 0.005 bitcoins. Coin size = halfCoin.divide(100); Coin totalPayment = Coin.ZERO; try { clientState.incrementPaymentBy(COIN); fail(); } catch (ValueOutOfRangeException e) {} byte[] signature = clientState.incrementPaymentBy(size).signature.encodeToBitcoin(); totalPayment = totalPayment.add(size); byte[] signatureCopy = Arrays.copyOf(signature, signature.length); signatureCopy[signatureCopy.length - 1] = (byte) ((Transaction.SigHash.NONE.ordinal() + 1) | 0x80); try { serverState.incrementPayment(halfCoin.subtract(totalPayment), signatureCopy); fail(); } catch (VerificationException e) {} signatureCopy = Arrays.copyOf(signature, signature.length); signatureCopy[2] ^= 0x42; // Make the signature fail standard checks try { serverState.incrementPayment(halfCoin.subtract(totalPayment), signatureCopy); fail(); } catch (VerificationException e) { assertTrue(e.getMessage().contains("not canonical")); } signatureCopy = Arrays.copyOf(signature, signature.length); signatureCopy[10] ^= 0x42; // Flip some random bits in the signature (to make it invalid, not just nonstandard) try { serverState.incrementPayment(halfCoin.subtract(totalPayment), signatureCopy); fail(); } catch (VerificationException e) { assertFalse(e.getMessage().contains("not canonical")); } serverState.incrementPayment(halfCoin.subtract(totalPayment), signature); // Pay the rest (signed with SIGHASH_NONE|SIGHASH_ANYONECANPAY) byte[] signature2 = clientState.incrementPaymentBy(halfCoin.subtract(totalPayment)).signature.encodeToBitcoin(); totalPayment = totalPayment.add(halfCoin.subtract(totalPayment)); assertEquals(totalPayment, halfCoin); signatureCopy = Arrays.copyOf(signature, signature.length); signatureCopy[signatureCopy.length - 1] = (byte) ((Transaction.SigHash.SINGLE.ordinal() + 1) | 0x80); try { serverState.incrementPayment(halfCoin.subtract(totalPayment), signatureCopy); fail(); } catch (VerificationException e) {} serverState.incrementPayment(halfCoin.subtract(totalPayment), signature2); // Trying to take reduce the refund size fails. try { serverState.incrementPayment(halfCoin.subtract(totalPayment.subtract(size)), signature); fail(); } catch (ValueOutOfRangeException e) {} assertEquals(serverState.getBestValueToMe(), totalPayment); try { clientState.incrementPaymentBy(Coin.SATOSHI.negate()); fail(); } catch (ValueOutOfRangeException e) {} try { clientState.incrementPaymentBy(halfCoin.subtract(size).add(Coin.SATOSHI)); fail(); } catch (ValueOutOfRangeException e) {} } @Test public void feesTest() throws Exception { // Test that transactions are getting the necessary fees // Spend the client wallet's one coin wallet.sendCoinsOffline(Wallet.SendRequest.to(new ECKey().toAddress(params), COIN)); assertEquals(Coin.ZERO, wallet.getBalance()); chain.add(makeSolvedTestBlock(blockStore.getChainHead().getHeader(), createFakeTx(params, CENT, myAddress))); assertEquals(CENT, wallet.getBalance()); Utils.setMockClock(); // Use mock clock final long EXPIRE_TIME = Utils.currentTimeMillis()/1000 + 60*60*24; serverState = new PaymentChannelServerState(mockBroadcaster, serverWallet, serverKey, EXPIRE_TIME); assertEquals(PaymentChannelServerState.State.WAITING_FOR_REFUND_TRANSACTION, serverState.getState()); // Clearly SATOSHI is far too small to be useful clientState = new PaymentChannelClientState(wallet, myKey, ECKey.fromPublicOnly(serverKey.getPubKey()), Coin.SATOSHI, EXPIRE_TIME); assertEquals(PaymentChannelClientState.State.NEW, clientState.getState()); try { clientState.initiate(); fail(); } catch (ValueOutOfRangeException e) {} clientState = new PaymentChannelClientState(wallet, myKey, ECKey.fromPublicOnly(serverKey.getPubKey()), Transaction.MIN_NONDUST_OUTPUT.subtract(Coin.SATOSHI).add(Transaction.REFERENCE_DEFAULT_MIN_TX_FEE), EXPIRE_TIME); assertEquals(PaymentChannelClientState.State.NEW, clientState.getState()); try { clientState.initiate(); fail(); } catch (ValueOutOfRangeException e) {} // Verify that MIN_NONDUST_OUTPUT + MIN_TX_FEE is accepted clientState = new PaymentChannelClientState(wallet, myKey, ECKey.fromPublicOnly(serverKey.getPubKey()), Transaction.MIN_NONDUST_OUTPUT.add(Transaction.REFERENCE_DEFAULT_MIN_TX_FEE), EXPIRE_TIME); assertEquals(PaymentChannelClientState.State.NEW, clientState.getState()); // We'll have to pay REFERENCE_DEFAULT_MIN_TX_FEE twice (multisig+refund), and we'll end up getting back nearly nothing... clientState.initiate(); assertEquals(clientState.getRefundTxFees(), Transaction.REFERENCE_DEFAULT_MIN_TX_FEE.multiply(2)); assertEquals(PaymentChannelClientState.State.INITIATED, clientState.getState()); // Now actually use a more useful CENT clientState = new PaymentChannelClientState(wallet, myKey, ECKey.fromPublicOnly(serverKey.getPubKey()), CENT, EXPIRE_TIME); assertEquals(PaymentChannelClientState.State.NEW, clientState.getState()); clientState.initiate(); assertEquals(clientState.getRefundTxFees(), Coin.ZERO); assertEquals(PaymentChannelClientState.State.INITIATED, clientState.getState()); // Send the refund tx from client to server and get back the signature. Transaction refund = new Transaction(params, clientState.getIncompleteRefundTransaction().bitcoinSerialize()); byte[] refundSig = serverState.provideRefundTransaction(refund, myKey.getPubKey()); assertEquals(PaymentChannelServerState.State.WAITING_FOR_MULTISIG_CONTRACT, serverState.getState()); // This verifies that the refund can spend the multi-sig output when run. clientState.provideRefundSignature(refundSig); assertEquals(PaymentChannelClientState.State.SAVE_STATE_IN_WALLET, clientState.getState()); clientState.fakeSave(); assertEquals(PaymentChannelClientState.State.PROVIDE_MULTISIG_CONTRACT_TO_SERVER, clientState.getState()); // Get the multisig contract Transaction multisigContract = new Transaction(params, clientState.getMultisigContract().bitcoinSerialize()); assertEquals(PaymentChannelClientState.State.READY, clientState.getState()); // Provide the server with the multisig contract and simulate successful propagation/acceptance. serverState.provideMultiSigContract(multisigContract); assertEquals(PaymentChannelServerState.State.WAITING_FOR_MULTISIG_ACCEPTANCE, serverState.getState()); TxFuturePair pair = broadcasts.take(); pair.future.set(pair.tx); assertEquals(PaymentChannelServerState.State.READY, serverState.getState()); // Both client and server are now in the ready state. Simulate a few micropayments Coin totalPayment = Coin.ZERO; // We can send as little as we want - its up to the server to get the fees right byte[] signature = clientState.incrementPaymentBy(Coin.SATOSHI).signature.encodeToBitcoin(); totalPayment = totalPayment.add(Coin.SATOSHI); serverState.incrementPayment(CENT.subtract(totalPayment), signature); // We can't refund more than the contract is worth... try { serverState.incrementPayment(CENT.add(SATOSHI), signature); fail(); } catch (ValueOutOfRangeException e) {} // We cannot send just under the total value - our refund would make it unspendable. So the client // will correct it for us to be larger than the requested amount, to make the change output zero. PaymentChannelClientState.IncrementedPayment payment = clientState.incrementPaymentBy(CENT.subtract(Transaction.MIN_NONDUST_OUTPUT)); assertEquals(CENT.subtract(SATOSHI), payment.amount); totalPayment = totalPayment.add(payment.amount); // The server also won't accept it if we do that. try { serverState.incrementPayment(Transaction.MIN_NONDUST_OUTPUT.subtract(Coin.SATOSHI), signature); fail(); } catch (ValueOutOfRangeException e) {} serverState.incrementPayment(CENT.subtract(totalPayment), payment.signature.encodeToBitcoin()); // And settle the channel. serverState.close(); assertEquals(PaymentChannelServerState.State.CLOSING, serverState.getState()); pair = broadcasts.take(); // settle pair.future.set(pair.tx); assertEquals(PaymentChannelServerState.State.CLOSED, serverState.getState()); serverState.close(); assertEquals(PaymentChannelServerState.State.CLOSED, serverState.getState()); } @Test public void serverAddsFeeTest() throws Exception { // Test that the server properly adds the necessary fee at the end (or just drops the payment if its not worth it) Utils.setMockClock(); // Use mock clock final long EXPIRE_TIME = Utils.currentTimeMillis()/1000 + 60*60*24; serverState = new PaymentChannelServerState(mockBroadcaster, serverWallet, serverKey, EXPIRE_TIME); assertEquals(PaymentChannelServerState.State.WAITING_FOR_REFUND_TRANSACTION, serverState.getState()); clientState = new PaymentChannelClientState(wallet, myKey, ECKey.fromPublicOnly(serverKey.getPubKey()), CENT, EXPIRE_TIME) { @Override protected void editContractSendRequest(Wallet.SendRequest req) { req.coinSelector = wallet.getCoinSelector(); } }; assertEquals(PaymentChannelClientState.State.NEW, clientState.getState()); clientState.initiate(); assertEquals(PaymentChannelClientState.State.INITIATED, clientState.getState()); // Send the refund tx from client to server and get back the signature. Transaction refund = new Transaction(params, clientState.getIncompleteRefundTransaction().bitcoinSerialize()); byte[] refundSig = serverState.provideRefundTransaction(refund, myKey.getPubKey()); assertEquals(PaymentChannelServerState.State.WAITING_FOR_MULTISIG_CONTRACT, serverState.getState()); // This verifies that the refund can spend the multi-sig output when run. clientState.provideRefundSignature(refundSig); assertEquals(PaymentChannelClientState.State.SAVE_STATE_IN_WALLET, clientState.getState()); clientState.fakeSave(); assertEquals(PaymentChannelClientState.State.PROVIDE_MULTISIG_CONTRACT_TO_SERVER, clientState.getState()); // Validate the multisig contract looks right. Transaction multisigContract = new Transaction(params, clientState.getMultisigContract().bitcoinSerialize()); assertEquals(PaymentChannelClientState.State.READY, clientState.getState()); assertEquals(2, multisigContract.getOutputs().size()); // One multi-sig, one change. Script script = multisigContract.getOutput(0).getScriptPubKey(); assertTrue(script.isSentToMultiSig()); script = multisigContract.getOutput(1).getScriptPubKey(); assertTrue(script.isSentToAddress()); assertTrue(wallet.getPendingTransactions().contains(multisigContract)); // Provide the server with the multisig contract and simulate successful propagation/acceptance. serverState.provideMultiSigContract(multisigContract); assertEquals(PaymentChannelServerState.State.WAITING_FOR_MULTISIG_ACCEPTANCE, serverState.getState()); TxFuturePair pair = broadcasts.take(); pair.future.set(pair.tx); assertEquals(PaymentChannelServerState.State.READY, serverState.getState()); // Both client and server are now in the ready state, split the channel in half byte[] signature = clientState.incrementPaymentBy(Transaction.REFERENCE_DEFAULT_MIN_TX_FEE.subtract(Coin.SATOSHI)) .signature.encodeToBitcoin(); Coin totalRefund = CENT.subtract(Transaction.REFERENCE_DEFAULT_MIN_TX_FEE.subtract(SATOSHI)); serverState.incrementPayment(totalRefund, signature); // We need to pay MIN_TX_FEE, but we only have MIN_NONDUST_OUTPUT try { serverState.close(); fail(); } catch (InsufficientMoneyException e) { } // Now give the server enough coins to pay the fee StoredBlock block = new StoredBlock(makeSolvedTestBlock(blockStore, new ECKey().toAddress(params)), BigInteger.ONE, 1); Transaction tx1 = createFakeTx(params, COIN, serverKey.toAddress(params)); serverWallet.receiveFromBlock(tx1, block, AbstractBlockChain.NewBlockType.BEST_CHAIN, 0); // The contract is still not worth redeeming - its worth less than we pay in fee try { serverState.close(); fail(); } catch (InsufficientMoneyException e) { assertTrue(e.getMessage().contains("more in fees")); } signature = clientState.incrementPaymentBy(SATOSHI.multiply(2)).signature.encodeToBitcoin(); totalRefund = totalRefund.subtract(SATOSHI.multiply(2)); serverState.incrementPayment(totalRefund, signature); // And settle the channel. serverState.close(); assertEquals(PaymentChannelServerState.State.CLOSING, serverState.getState()); pair = broadcasts.take(); pair.future.set(pair.tx); assertEquals(PaymentChannelServerState.State.CLOSED, serverState.getState()); } @Test public void doubleSpendContractTest() throws Exception { // Tests that if the client double-spends the multisig contract after it is sent, no more payments are accepted // Start with a copy of basic().... Utils.setMockClock(); // Use mock clock final long EXPIRE_TIME = Utils.currentTimeSeconds() + 60*60*24; serverState = new PaymentChannelServerState(mockBroadcaster, serverWallet, serverKey, EXPIRE_TIME); assertEquals(PaymentChannelServerState.State.WAITING_FOR_REFUND_TRANSACTION, serverState.getState()); clientState = new PaymentChannelClientState(wallet, myKey, ECKey.fromPublicOnly(serverKey.getPubKey()), halfCoin, EXPIRE_TIME); assertEquals(PaymentChannelClientState.State.NEW, clientState.getState()); clientState.initiate(); assertEquals(PaymentChannelClientState.State.INITIATED, clientState.getState()); // Send the refund tx from client to server and get back the signature. Transaction refund = new Transaction(params, clientState.getIncompleteRefundTransaction().bitcoinSerialize()); byte[] refundSig = serverState.provideRefundTransaction(refund, myKey.getPubKey()); assertEquals(PaymentChannelServerState.State.WAITING_FOR_MULTISIG_CONTRACT, serverState.getState()); // This verifies that the refund can spend the multi-sig output when run. clientState.provideRefundSignature(refundSig); assertEquals(PaymentChannelClientState.State.SAVE_STATE_IN_WALLET, clientState.getState()); clientState.fakeSave(); assertEquals(PaymentChannelClientState.State.PROVIDE_MULTISIG_CONTRACT_TO_SERVER, clientState.getState()); // Validate the multisig contract looks right. Transaction multisigContract = new Transaction(params, clientState.getMultisigContract().bitcoinSerialize()); assertEquals(PaymentChannelClientState.State.READY, clientState.getState()); assertEquals(2, multisigContract.getOutputs().size()); // One multi-sig, one change. Script script = multisigContract.getOutput(0).getScriptPubKey(); assertTrue(script.isSentToMultiSig()); script = multisigContract.getOutput(1).getScriptPubKey(); assertTrue(script.isSentToAddress()); assertTrue(wallet.getPendingTransactions().contains(multisigContract)); // Provide the server with the multisig contract and simulate successful propagation/acceptance. serverState.provideMultiSigContract(multisigContract); assertEquals(PaymentChannelServerState.State.WAITING_FOR_MULTISIG_ACCEPTANCE, serverState.getState()); final TxFuturePair pair = broadcasts.take(); pair.future.set(pair.tx); assertEquals(PaymentChannelServerState.State.READY, serverState.getState()); // Make sure the refund transaction is not in the wallet and multisig contract's output is not connected to it assertEquals(2, wallet.getTransactions(false).size()); Iterator<Transaction> walletTransactionIterator = wallet.getTransactions(false).iterator(); Transaction clientWalletMultisigContract = walletTransactionIterator.next(); assertFalse(clientWalletMultisigContract.getHash().equals(clientState.getCompletedRefundTransaction().getHash())); if (!clientWalletMultisigContract.getHash().equals(multisigContract.getHash())) { clientWalletMultisigContract = walletTransactionIterator.next(); assertFalse(clientWalletMultisigContract.getHash().equals(clientState.getCompletedRefundTransaction().getHash())); } else assertFalse(walletTransactionIterator.next().getHash().equals(clientState.getCompletedRefundTransaction().getHash())); assertEquals(multisigContract.getHash(), clientWalletMultisigContract.getHash()); assertFalse(clientWalletMultisigContract.getInput(0).getConnectedOutput().getSpentBy().getParentTransaction().getHash().equals(refund.getHash())); // Both client and server are now in the ready state. Simulate a few micropayments of 0.005 bitcoins. Coin size = halfCoin.divide(100); Coin totalPayment = Coin.ZERO; for (int i = 0; i < 5; i++) { byte[] signature = clientState.incrementPaymentBy(size).signature.encodeToBitcoin(); totalPayment = totalPayment.add(size); serverState.incrementPayment(halfCoin.subtract(totalPayment), signature); } // Now create a double-spend and send it to the server Transaction doubleSpendContract = new Transaction(params); doubleSpendContract.addInput(new TransactionInput(params, doubleSpendContract, new byte[0], multisigContract.getInput(0).getOutpoint())); doubleSpendContract.addOutput(halfCoin, myKey); doubleSpendContract = new Transaction(params, doubleSpendContract.bitcoinSerialize()); StoredBlock block = new StoredBlock(params.getGenesisBlock().createNextBlock(myKey.toAddress(params)), BigInteger.TEN, 1); serverWallet.receiveFromBlock(doubleSpendContract, block, AbstractBlockChain.NewBlockType.BEST_CHAIN, 0); // Now if we try to spend again the server will reject it since it saw a double-spend try { byte[] signature = clientState.incrementPaymentBy(size).signature.encodeToBitcoin(); totalPayment = totalPayment.add(size); serverState.incrementPayment(halfCoin.subtract(totalPayment), signature); fail(); } catch (VerificationException e) { assertTrue(e.getMessage().contains("double-spent")); } } }