Skip to content
This repository has been archived by the owner on Aug 16, 2021. It is now read-only.

Move string validation logic out of WalletManager #228

Merged
merged 2 commits into from
Jul 13, 2017
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 3 additions & 3 deletions Stratis.Bitcoin.IntegrationTests/WalletTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@ public void WalletCanReceiveAndSendCorrectly()

// send coins to the receiver
var sendto = stratisReceiver.FullNode.WalletManager.GetUnusedAddress(new WalletAccountReference("mywallet", "account 0"));
var trx = stratisSender.FullNode.WalletManager.BuildTransaction(new WalletAccountReference("mywallet", "account 0"), "123456", sendto.Address, Money.COIN * 100, FeeType.Medium, 101);
var trx = stratisSender.FullNode.WalletManager.BuildTransaction(new WalletAccountReference("mywallet", "account 0"), "123456", sendto.ScriptPubKey, Money.COIN * 100, FeeType.Medium, 101);

// broadcast to the other node
stratisSender.FullNode.WalletManager.SendTransaction(trx.hex);
Expand Down Expand Up @@ -158,7 +158,7 @@ public void WalletCanReorg()
// ====================
// send coins to the receiver
var sendto = stratisReceiver.FullNode.WalletManager.GetUnusedAddress(new WalletAccountReference("mywallet", "account 0"));
var transaction1 = stratisSender.FullNode.WalletManager.BuildTransaction(new WalletAccountReference("mywallet", "account 0"), "123456", sendto.Address, Money.COIN * 100, FeeType.Medium, 101);
var transaction1 = stratisSender.FullNode.WalletManager.BuildTransaction(new WalletAccountReference("mywallet", "account 0"), "123456", sendto.ScriptPubKey, Money.COIN * 100, FeeType.Medium, 101);

// broadcast to the other node
stratisSender.FullNode.WalletManager.SendTransaction(transaction1.hex);
Expand Down Expand Up @@ -194,7 +194,7 @@ public void WalletCanReorg()

// send more coins to the wallet
sendto = stratisReceiver.FullNode.WalletManager.GetUnusedAddress(new WalletAccountReference("mywallet", "account 0"));
var transaction2 = stratisSender.FullNode.WalletManager.BuildTransaction(new WalletAccountReference("mywallet", "account 0"), "123456", sendto.Address, Money.COIN * 10, FeeType.Medium, 101);
var transaction2 = stratisSender.FullNode.WalletManager.BuildTransaction(new WalletAccountReference("mywallet", "account 0"), "123456", sendto.ScriptPubKey, Money.COIN * 10, FeeType.Medium, 101);
stratisSender.FullNode.WalletManager.SendTransaction(transaction2.hex);
// wait for the trx to arrive
TestHelper.WaitLoop(() => stratisReceiver.CreateRPCClient().GetRawMempool().Length > 0);
Expand Down
6 changes: 3 additions & 3 deletions Stratis.Bitcoin.Tests/Wallet/WalletControllerTest.cs
Original file line number Diff line number Diff line change
Expand Up @@ -1181,7 +1181,7 @@ public void BuildTransactionWithValidRequestAllowingUnconfirmedReturnsWalletBuil
{
var mockWalletWrapper = new Mock<IWalletManager>();
var key = new Key();
mockWalletWrapper.Setup(m => m.BuildTransaction(new WalletAccountReference("myWallet", "Account 1"), "test", key.PubKey.GetAddress(Network.Main).ToString(), new Money(150000), FeeType.High, 0))
mockWalletWrapper.Setup(m => m.BuildTransaction(new WalletAccountReference("myWallet", "Account 1"), "test", key.PubKey.GetAddress(Network.Main).ScriptPubKey, new Money(150000), FeeType.High, 0))
.Returns(("hexString", new uint256(15), new Money(150)));


Expand Down Expand Up @@ -1211,7 +1211,7 @@ public void BuildTransactionWithValidRequestNotAllowingUnconfirmedReturnsWalletB
{
var mockWalletWrapper = new Mock<IWalletManager>();
var key = new Key();
mockWalletWrapper.Setup(m => m.BuildTransaction(new WalletAccountReference("myWallet", "Account 1"), "test", key.PubKey.GetAddress(Network.Main).ToString(), new Money(150000), FeeType.High, 1))
mockWalletWrapper.Setup(m => m.BuildTransaction(new WalletAccountReference("myWallet", "Account 1"), "test", key.PubKey.GetAddress(Network.Main).ScriptPubKey, new Money(150000), FeeType.High, 1))
.Returns(("hexString", new uint256(15), new Money(150)));

var controller = new WalletController(mockWalletWrapper.Object, new Mock<IWalletSyncManager>().Object, It.IsAny<ConnectionManager>(), Network.Main, new Mock<ConcurrentChain>().Object, It.IsAny<DataFolder>());
Expand Down Expand Up @@ -1262,7 +1262,7 @@ public void BuildTransactionWithExceptionReturnsBadRequest()
{
var mockWalletWrapper = new Mock<IWalletManager>();
var key = new Key();
mockWalletWrapper.Setup(m => m.BuildTransaction(new WalletAccountReference("myWallet", "Account 1"), "test", key.PubKey.GetAddress(Network.Main).ToString(), new Money(150000), FeeType.High, 1))
mockWalletWrapper.Setup(m => m.BuildTransaction(new WalletAccountReference("myWallet", "Account 1"), "test", key.PubKey.GetAddress(Network.Main).ScriptPubKey, new Money(150000), FeeType.High, 1))
.Throws(new InvalidOperationException("Issue building transaction."));

var controller = new WalletController(mockWalletWrapper.Object, new Mock<IWalletSyncManager>().Object, It.IsAny<ConnectionManager>(), Network.Main, new Mock<ConcurrentChain>().Object, It.IsAny<DataFolder>());
Expand Down
5 changes: 3 additions & 2 deletions Stratis.Bitcoin/Wallet/Controllers/WalletController.cs
Original file line number Diff line number Diff line change
Expand Up @@ -418,10 +418,11 @@ public IActionResult BuildTransaction([FromBody] BuildTransactionRequest request
var errors = this.ModelState.Values.SelectMany(e => e.Errors.Select(m => m.ErrorMessage));
return ErrorHelpers.BuildErrorResponse(HttpStatusCode.BadRequest, "Formatting error", string.Join(Environment.NewLine, errors));
}

var destination = BitcoinAddress.Create(request.DestinationAddress, this.network).ScriptPubKey;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What if this fails maybe it can go inside the try catch?


try
{
var transactionResult = this.walletManager.BuildTransaction(new WalletAccountReference(request.WalletName, request.AccountName), request.Password, request.DestinationAddress, request.Amount, FeeParser.Parse(request.FeeType), request.AllowUnconfirmed ? 0 : 1);
var transactionResult = this.walletManager.BuildTransaction(new WalletAccountReference(request.WalletName, request.AccountName), request.Password, destination, request.Amount, FeeParser.Parse(request.FeeType), request.AllowUnconfirmed ? 0 : 1);
var model = new WalletBuildTransactionModel
{
Hex = transactionResult.hex,
Expand Down
4 changes: 2 additions & 2 deletions Stratis.Bitcoin/Wallet/IWalletManager.cs
Original file line number Diff line number Diff line change
Expand Up @@ -135,12 +135,12 @@ public interface IWalletManager : IDisposable
/// </summary>
/// <param name="accountReference">The name of the wallet and account</param>
/// <param name="password">The password used to decrypt the private key.</param>
/// <param name="destinationAddress">The destination address to send the funds to.</param>
/// <param name="destinationScript">The destination to send the funds to.</param>
/// <param name="amount">The amount of funds to be sent.</param>
/// <param name="feeType">The type of fee to be included.</param>
/// <param name="minConfirmations">The minimum number of confirmations we require for unspent outputs to be included.</param>
/// <returns></returns>
(string hex, uint256 transactionId, Money fee) BuildTransaction(WalletAccountReference accountReference, string password, string destinationAddress, Money amount, FeeType feeType, int minConfirmations);
(string hex, uint256 transactionId, Money fee) BuildTransaction(WalletAccountReference accountReference, string password, Script destinationScript, Money amount, FeeType feeType, int minConfirmations);

/// <summary>
/// Remove all the thransactions in the wallet that are above this block height
Expand Down
3 changes: 3 additions & 0 deletions Stratis.Bitcoin/Wallet/Models/RequestModels.cs
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@
using System.ComponentModel.DataAnnotations;
using Newtonsoft.Json;
using Newtonsoft.Json.Converters;
using NBitcoin;
using Stratis.Bitcoin.Wallet.Validations;

namespace Stratis.Bitcoin.Wallet.Models
{
Expand Down Expand Up @@ -99,6 +101,7 @@ public class BuildTransactionRequest : RequestModel
public string Password { get; set; }

[Required(ErrorMessage = "A destination address is required.")]
[IsBitcoinAddress()]
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah ok ignore my last comment

public string DestinationAddress { get; set; }

[Required(ErrorMessage = "An amount is required.")]
Expand Down
22 changes: 22 additions & 0 deletions Stratis.Bitcoin/Wallet/Validations/IsBitcoinAddressAttribute.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
using NBitcoin;
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
using System.Text;

namespace Stratis.Bitcoin.Wallet.Validations
{
public class IsBitcoinAddressAttribute : ValidationAttribute
{
protected override ValidationResult IsValid(object value, ValidationContext validationContext)
{
var network = (Network)validationContext.GetService(typeof(Network));
try
{
BitcoinAddress.Create(value as string, network);
return ValidationResult.Success;
}
catch { return new ValidationResult("Invalid address"); }
}
}
}
13 changes: 1 addition & 12 deletions Stratis.Bitcoin/Wallet/WalletManager.cs
Original file line number Diff line number Diff line change
Expand Up @@ -461,7 +461,7 @@ public ISecret GetKeyForAddress(string password, HdAddress address)
}

/// <inheritdoc />
public (string hex, uint256 transactionId, Money fee) BuildTransaction(WalletAccountReference accountReference, string password, string destinationAddress, Money amount, FeeType feeType, int minConfirmations)
public (string hex, uint256 transactionId, Money fee) BuildTransaction(WalletAccountReference accountReference, string password, Script destinationScript, Money amount, FeeType feeType, int minConfirmations)
{
if (amount == Money.Zero)
{
Expand All @@ -472,17 +472,6 @@ public ISecret GetKeyForAddress(string password, HdAddress address)
Wallet wallet = this.GetWalletByName(accountReference.WalletName);
HdAccount account = this.GetAccounts(wallet).GetAccountByName(accountReference.AccountName);

// get script destination address
Script destinationScript = null;
try
{
destinationScript = PayToPubkeyHashTemplate.Instance.GenerateScriptPubKey(new BitcoinPubKeyAddress(destinationAddress, wallet.Network));
}
catch
{
throw new WalletException("Invalid address.");
}

// get a list of transactions outputs that have not been spent
var spendableTransactions = account.GetSpendableTransactions().ToList();

Expand Down