Merge pull request #679 from matiu/feature/bits7

Feature/bits7
This commit is contained in:
Gustavo Maximiliano Cortez 2014-06-16 13:14:10 -03:00
commit 7406623ed5
25 changed files with 616 additions and 336 deletions

View file

@ -1,8 +1,12 @@
'use strict'; 'use strict';
var defaultConfig = { var defaultConfig = {
// livenet or testnet // DEFAULT network (livenet or testnet)
networkName: 'testnet', networkName: 'testnet',
// DEFAULT unit: Bit
unitName: 'bits',
unitToSatoshi: 100,
// wallet limits // wallet limits
limits: { limits: {
totalCopayers: 12, totalCopayers: 12,

View file

@ -219,6 +219,21 @@ small.has-error {
font-weight: bold; font-weight: bold;
} }
.totalAmount {
line-height: 120%;
margin-top:2px;
}
/* Turn Off Number Input Spinners */
input[type=number]::-webkit-inner-spin-button,
input[type=number]::-webkit-outer-spin-button {
-webkit-appearance: none;
margin: 0;
}
.small { .small {
font-size: 60%; font-size: 60%;
line-height: inherit; line-height: inherit;

View file

@ -21,7 +21,7 @@
</div> </div>
<div class="large-9 medium-9 small-9 columns text-center p10t" ng-show="$root.wallet"> <div class="large-9 medium-9 small-9 columns text-center p10t" ng-show="$root.wallet">
<div class="large-4 medium-4 columns line-dashed-v"> <div class="large-4 medium-4 columns line-dashed-v">
<a href="#/addresses" class="has-tip" tooltip-placement="bottom" tooltip="{{$root.wallet.id}}"> <a href="#/addresses" class="has-tip" tooltip-placement="bottom" tooltip="ID: {{$root.wallet.id}}">
<strong><span>{{$root.wallet.getName()}}</span></strong> <strong><span>{{$root.wallet.getName()}}</span></strong>
</a> </a>
<a class="button radius small-icon" title="Manual Refresh" <a class="button radius small-icon" title="Manual Refresh"
@ -31,21 +31,19 @@
ng-click="signout()"><i class="fi-power"></i></a> ng-click="signout()"><i class="fi-power"></i></a>
</div> </div>
<div class="large-4 medium-4 columns line-dashed-v"> <div class="large-4 medium-4 columns line-dashed-v">
Balance: Balance<br>
<span ng-if="$root.updatingBalance"> <span ng-if="$root.updatingBalance">
<i class="fi-bitcoin-circle icon-rotate spinner"></i> <i class="fi-bitcoin-circle icon-rotate spinner"></i>
</span> </span>
<span ng-if="!$root.updatingBalance">{{totalBalance || 0}} <span ng-if="!$root.updatingBalance" tooltip="{{totalBalanceBTC}} BTC" tooltip-trigger="mouseenter" tooltip-placement="bottom">{{totalBalance || 0 |number}} {{$root.unitName}}
<i class="fi-bitcoin"></i>
</span> </span>
</div> </div>
<div class="large-4 medium-4 columns"> <div class="large-4 medium-4 columns">
Available to Spend: Available to Spend<br>
<span ng-if="$root.updatingBalance"> <span ng-if="$root.updatingBalance">
<i class="fi-bitcoin-circle icon-rotate spinner"></i> <i class="fi-bitcoin-circle icon-rotate spinner"></i>
</span> </span>
<span ng-if="!$root.updatingBalance">{{availableBalance || 0}} <span ng-show="!$root.updatingBalance" tooltip="{{availableBalanceBTC}} BTC" tooltip-trigger="mouseenter" tooltip-placement="bottom">{{availableBalance || 0|number}} {{$root.unitName}}
<i class="fi-bitcoin"></i>
</span> </span>
</div> </div>
@ -383,13 +381,11 @@
<i class="fi-bitcoin-circle icon-rotate spinner"></i> <i class="fi-bitcoin-circle icon-rotate spinner"></i>
</span> </span>
<span ng-if="!$root.updatingBalance"> <span ng-if="!$root.updatingBalance">
{{$root.balanceByAddr[addr.address] || 0}} {{$root.balanceByAddr[addr.address] || 0|number}} {{$root.unitName}}
<i class="fi-bitcoin"></i>
</span> </span>
</span> </span>
<span ng-if="addrWithFund != addr.address"> <span ng-if="addrWithFund != addr.address">
{{addr.balance || 0}} {{addr.balance || 0|number}} {{$root.unitName}}
<i class="fi-bitcoin"></i>
</span> </span>
</span> </span>
</a> </a>
@ -407,14 +403,12 @@
<span ng-if="$root.updatingBalance"> <span ng-if="$root.updatingBalance">
<i class="fi-bitcoin-circle icon-rotate spinner"></i> <i class="fi-bitcoin-circle icon-rotate spinner"></i>
</span> </span>
<span ng-if="!$root.updatingBalance">{{balanceByAddr[selectedAddr.address] || 0}} <span ng-if="!$root.updatingBalance">{{balanceByAddr[selectedAddr.address] || 0 | number}}
<i class="fi-bitcoin"></i>
</span> </span>
</span> </span>
<span ng-if="addrWithFund != selectedAddr.address" style="word-wrap: break-word;"> <span ng-if="addrWithFund != selectedAddr.address" style="word-wrap: break-word;">
{{selectedAddr.address}}<br/> {{selectedAddr.address}}<br/>
{{selectedAddr.balance || 0}} {{selectedAddr.balance || 0|number}} {{$root.unitName}}
<i class="fi-bitcoin"></i>
</span> </span>
</strong> </strong>
</p> </p>
@ -445,7 +439,7 @@
<div class="txheader row m10"> <div class="txheader row m10">
<div class="large-8 medium-8 small-12 columns"> <div class="large-8 medium-8 small-12 columns">
<div class="row" ng-repeat="out in tx.outs"> <div class="row" ng-repeat="out in tx.outs">
<div class="large-3 medium-3 small-3 columns ellipsis"> {{out.value}} <i class="fi-bitcoin size-18"></i></div> <div class="large-3 medium-3 small-3 columns ellipsis"> {{out.value | number}} {{$root.unitName}}</div>
<div class="large-1 medium-1 small-2 columns fi-arrow-right size-24"> </div> <div class="large-1 medium-1 small-2 columns fi-arrow-right size-24"> </div>
<div class="large-8 medium-8 small-7 columns ellipsis"> {{out.address}} </div> <div class="large-8 medium-8 small-7 columns ellipsis"> {{out.address}} </div>
</div> </div>
@ -527,7 +521,7 @@
<p class="text-gray m5b" ng-show="!tx.finallyRejected && tx.missingSignatures>1"> <p class="text-gray m5b" ng-show="!tx.finallyRejected && tx.missingSignatures>1">
{{tx.missingSignatures}} signatures missing</p> {{tx.missingSignatures}} signatures missing</p>
<div class="ellipsis small text-gray"> <div class="ellipsis small text-gray">
<strong>Fee:</strong> <i class="fi-bitcoin"></i> {{tx.fee}} <strong>Fee:</strong> {{tx.fee|number}} {{$root.unitName}}
<strong>Proposal ID:</strong> {{tx.ntxid}} <strong>Proposal ID:</strong> {{tx.ntxid}}
</div> </div>
</div> </div>
@ -571,7 +565,7 @@
<div class="row"> <div class="row">
<div class="large-5 medium-5 small-5 columns"> <div class="large-5 medium-5 small-5 columns">
<div ng-repeat="vin in btx.vinSimple"> <div ng-repeat="vin in btx.vinSimple">
<small class="right m5t">{{vin.value}}</small> <small class="right m5t">{{vin.value| number}} {{$root.unitName}}</small>
<p class="ellipsis text-gray size-12"> {{vin.addr}} </p> <p class="ellipsis text-gray size-12"> {{vin.addr}} </p>
</div> </div>
</div> </div>
@ -580,7 +574,7 @@
</div> </div>
<div class="large-6 medium-6 small-6 columns"> <div class="large-6 medium-6 small-6 columns">
<div ng-repeat="vout in btx.voutSimple"> <div ng-repeat="vout in btx.voutSimple">
<small class="right m5t">{{vout.value}}</small> <small class="right m5t">{{vout.value| number}} {{$root.unitName}}</small>
<p class="ellipsis text-gray size-12"> {{vout.addr}} </p> <p class="ellipsis text-gray size-12"> {{vout.addr}} </p>
</div> </div>
</div> </div>
@ -588,9 +582,9 @@
</div> </div>
<div class="m10 size-12 text-gray"> <div class="m10 size-12 text-gray">
<div class="row"> <div class="row">
<div class="large-4 medium-4 small-4 columns">Fees: {{btx.fees}}</div> <div class="large-4 medium-4 small-4 columns">Fees: {{btx.fees | number}} {{$root.unitName}}</div>
<div class="large-4 medium-4 small-4 columns text-center">Confirmations: {{btx.confirmations || 0}}</div> <div class="large-4 medium-4 small-4 columns text-center">Confirmations: {{btx.confirmations || 0}}</div>
<div class="large-4 medium-4 small-4 columns text-right">Total: {{btx.valueOut}}</div> <div class="large-4 medium-4 small-4 columns text-right">Total: {{btx.valueOut| number}} {{$root.unitName}}</div>
</div> </div>
</div> </div>
</div> </div>
@ -652,22 +646,38 @@
<div class="row collapse"> <div class="row collapse">
<label for="amount">Amount <label for="amount">Amount
<small ng-hide="!sendForm.amount.$pristine">required</small> <small ng-hide="!sendForm.amount.$pristine">required</small>
<small class="is-valid" ng-show="!sendForm.amount.$invalid && !sendForm.amount.$pristine">valid!</small> <small class="is-valid" ng-show="!sendForm.amount.$invalid && !sendForm.amount.$pristine">Valid</small>
<small class="has-error" ng-show="sendForm.amount.$invalid && !sendForm.amount.$pristine && !notEnoughAmount"> <small class="has-error" ng-show="sendForm.amount.$invalid && !sendForm.amount.$pristine && !notEnoughAmount">
not valid Not valid
</small> </small>
<small ng-show="notEnoughAmount">Insufficient funds!</small> <small ng-show="notEnoughAmount" class="has-error">Insufficient funds</small>
</label> </label>
<div class="small-9 columns"> <div class="small-9 columns">
<input type="number" id="amount" ng-disabled="loading" <input type="number" id="amount" ng-disabled="loading"
name="amount" placeholder="Amount" ng-model="amount" name="amount" placeholder="Amount" ng-model="amount"
min="0.0001" max="10000000" enough-amount required> min="1" max="10000000000" enough-amount required
autocomplete="off"
>
</div> </div>
<div class="small-3 columns"> <div class="small-3 columns">
<span class="postfix">BTC</span> <span class="postfix">{{$root.unitName}}</span>
</div> </div>
</div> </div>
</div> </div>
<div class="large-6 medium-6 columns m10t" ng-show="amount>0">
<small>
Total amount for this transaction:
</small>
<div class="totalAmount">
<b>{{amount + defaultFee |number}}</b> {{$root.unitName}}
<small>
{{ ((amount + defaultFee) * unitToBtc) |number}} BTC
</small>
</div>
<small>
Including fee of {{defaultFee|number}} {{$root.unitName}}
</small>
</div>
</div> </div>
<div class="row"> <div class="row">
@ -680,7 +690,7 @@
</label> </label>
<div class="small-12 columns"> <div class="small-12 columns">
<textarea id="comment" ng-disabled="loading" <textarea id="comment" ng-disabled="loading"
name="comment" placeholder="Leave a private message to your copayers" ng-model="comment" ng-maxlength="100"></textarea> name="comment" placeholder="Leave a private message to your copayers" ng-model="commentText" ng-maxlength="100"></textarea>
</div> </div>
</div> </div>
</div> </div>
@ -740,6 +750,17 @@
<input id="network-name" type="checkbox" ng-model="networkName" ng-true-value="livenet" ng-false-value="testnet" class="form-control"> <input id="network-name" type="checkbox" ng-model="networkName" ng-true-value="livenet" ng-false-value="testnet" class="form-control">
<label for="network-name">Livenet</label> <label for="network-name">Livenet</label>
</fieldset> </fieldset>
<fieldset>
<legend>Wallet Unit</legend>
<select class="form-control" ng-model="selectedUnit" ng-options="o.name for o in unitOpts" required>
</select>
<label for="settingsUnit">Prefered Unit for Wallet</label>
</fieldset>
<fieldset>
<legend>Videoconferencing</legend>
<input id="disableVideo-opt" type="checkbox" ng-model="disableVideo" class="form-control">
<label for="disableVideo-opt">Enable videoconferencing (only for fast Networks)</label>
</fieldset>
<fieldset> <fieldset>
<legend>Insight API server</legend> <legend>Insight API server</legend>
<label for="insight-host">Host</label> <label for="insight-host">Host</label>
@ -758,13 +779,6 @@
<input id="peerjs-secure" type="checkbox" ng-model="networkSecure" class="form-control" disabled="disabled"> <input id="peerjs-secure" type="checkbox" ng-model="networkSecure" class="form-control" disabled="disabled">
<label for="peerjs-secure">Use SSL (disabled)</label> <label for="peerjs-secure">Use SSL (disabled)</label>
</fieldset> </fieldset>
<fieldset>
<legend>Videoconferencing</legend>
<input id="disableVideo-opt" type="checkbox" ng-model="disableVideo" class="form-control">
<label for="disableVideo-opt">Enable videoconferencing (only for fast Networks)</label>
</fieldset>
</div> </div>
<div class="row"> <div class="row">
<div class="large-12 columns line-dashed"> <div class="large-12 columns line-dashed">

View file

@ -32,6 +32,7 @@ angular.module('copayApp.controllers').controller('HeaderController',
} }
}); });
$rootScope.unitName = config.unitName;
// Initialize alert notification (not show when init wallet) // Initialize alert notification (not show when init wallet)
$rootScope.txAlertCount = 0; $rootScope.txAlertCount = 0;

View file

@ -20,15 +20,7 @@ angular.module('copayApp.controllers').controller('ImportController',
} }
$rootScope.wallet = w; $rootScope.wallet = w;
controllerUtils.startNetwork($rootScope.wallet); controllerUtils.startNetwork($rootScope.wallet, $scope);
$rootScope.wallet.on('connectionError', function() {
var message = "Looks like you are already connected to this wallet, please logout from it and try importing it again.";
$rootScope.$flashMessage = { message: message, type: 'error'};
});
$rootScope.wallet.on('serverError', function() {
$rootScope.$flashMessage = { message: 'The PeerJS server is not responding, please try again', type: 'error'};
controllerUtils.onErrorDigest();
});
}); });
}; };

View file

@ -1,10 +1,15 @@
'use strict'; 'use strict';
var bitcore = require('bitcore');
angular.module('copayApp.controllers').controller('SendController', angular.module('copayApp.controllers').controller('SendController',
function($scope, $rootScope, $window, $location, $timeout) { function($scope, $rootScope, $window, $location, $timeout) {
$scope.title = 'Send'; $scope.title = 'Send';
$scope.loading = false; $scope.loading = false;
var satToUnit = 1 / config.unitToSatoshi;
$scope.defaultFee = bitcore.TransactionBuilder.FEE_PER_1000B_SAT * satToUnit;
$scope.unitToBtc = config.unitToSatoshi / bitcore.util.COIN;
// TODO this shouldnt be on a particular controller.
// Detect mobile devices // Detect mobile devices
var isMobile = { var isMobile = {
Android: function() { Android: function() {
@ -32,27 +37,31 @@ angular.module('copayApp.controllers').controller('SendController',
navigator.getUserMedia = navigator.getUserMedia || navigator.webkitGetUserMedia || navigator.mozGetUserMedia || navigator.msGetUserMedia; navigator.getUserMedia = navigator.getUserMedia || navigator.webkitGetUserMedia || navigator.mozGetUserMedia || navigator.msGetUserMedia;
window.URL = window.URL || window.webkitURL || window.mozURL || window.msURL; window.URL = window.URL || window.webkitURL || window.mozURL || window.msURL;
$scope.isMobile = isMobile.any(); $scope.isMobile = isMobile.any();
$scope.unitIds = ['BTC','mBTC'];
$scope.selectedUnit = $scope.unitIds[0];
$scope.submitForm = function(form) { $scope.submitForm = function(form) {
if (form.$invalid) { if (form.$invalid) {
$rootScope.$flashMessage = { message: 'You can not send a proposal transaction. Please, try again', type: 'error'}; $rootScope.$flashMessage = {
message: 'You can not send a proposal transaction. Please, try again',
type: 'error'
};
return; return;
} }
$scope.loading = true; $scope.loading = true;
var address = form.address.$modelValue; var address = form.address.$modelValue;
var amount = (form.amount.$modelValue * 100000000).toFixed(); // satoshi to string var amount = (form.amount.$modelValue * config.unitToSatoshi) | 0;
var comment = form.comment.$modelValue; var commentText = form.comment.$modelValue;
var w = $rootScope.wallet; var w = $rootScope.wallet;
w.createTx(address, amount, comment, function() {
w.createTx(address, amount, commentText, function() {
$scope.loading = false; $scope.loading = false;
$rootScope.$flashMessage = { message: 'The transaction proposal has been created', type: 'success'}; $rootScope.$flashMessage = {
message: 'The transaction proposal has been created',
type: 'success'
};
$rootScope.$digest(); $rootScope.$digest();
}); });
@ -81,7 +90,11 @@ angular.module('copayApp.controllers').controller('SendController',
reader.onload = (function(theFile) { reader.onload = (function(theFile) {
return function(e) { return function(e) {
var mpImg = new MegaPixImage(file); var mpImg = new MegaPixImage(file);
mpImg.render(canvas, { maxWidth: 200, maxHeight: 200, orientation: 6 }); mpImg.render(canvas, {
maxWidth: 200,
maxHeight: 200,
orientation: 6
});
$timeout(function() { $timeout(function() {
qrcode.width = canvas.width; qrcode.width = canvas.width;
@ -168,7 +181,9 @@ angular.module('copayApp.controllers').controller('SendController',
canvas.height = 225; canvas.height = 225;
context.clearRect(0, 0, 300, 225); context.clearRect(0, 0, 300, 225);
navigator.getUserMedia({video: true}, _successCallback, _videoError); navigator.getUserMedia({
video: true
}, _successCallback, _videoError);
} }
}, 500); }, 500);
}; };

View file

@ -13,6 +13,31 @@ angular.module('copayApp.controllers').controller('SettingsController',
$scope.networkSecure = config.network.secure || false; $scope.networkSecure = config.network.secure || false;
$scope.disableVideo = config.disableVideo || true; $scope.disableVideo = config.disableVideo || true;
$scope.unitOpts = [{
name: 'Satoshis (100,000,000 bits = 1BTC)',
shortName: 'SAT',
value: 1
}, {
name: 'bits (1,000,000 bits = 1BTC)',
shortName: 'bits',
value: 100
}, {
name: 'mBTC (1,000 mBTC = 1BTC)',
shortName: 'mBTC',
value: 100000
}, {
name: 'BTC',
shortName: 'BTC',
value: 100000000
}];
for (var ii in $scope.unitOpts) {
if (config.unitName === $scope.unitOpts[ii].shortName) {
$scope.selectedUnit = $scope.unitOpts[ii];
break;
}
}
$scope.$watch('networkName', function(net) { $scope.$watch('networkName', function(net) {
$scope.insightHost = net === 'testnet' ? 'test.insight.is' : 'live.insight.is'; $scope.insightHost = net === 'testnet' ? 'test.insight.is' : 'live.insight.is';
}); });
@ -36,8 +61,9 @@ angular.module('copayApp.controllers').controller('SettingsController',
}, },
network: network, network: network,
disableVideo: $scope.disableVideo, disableVideo: $scope.disableVideo,
}) unitName: $scope.selectedUnit.shortName,
); unitToSatoshi: $scope.selectedUnit.value,
}));
$window.location.href = $window.location.origin + $window.location.pathname; $window.location.href = $window.location.origin + $window.location.pathname;
}; };

View file

@ -84,7 +84,7 @@ angular.module('copayApp.controllers').controller('SetupController',
passphrase: passphrase, passphrase: passphrase,
}; };
var w = walletFactory.create(opts); var w = walletFactory.create(opts);
controllerUtils.startNetwork(w); controllerUtils.startNetwork(w, $scope);
}); });
}; };

View file

@ -34,8 +34,7 @@ angular.module('copayApp.controllers').controller('SigninController',
$rootScope.$digest(); $rootScope.$digest();
return; return;
} }
installStartupHandlers(w); controllerUtils.startNetwork(w, $scope);
controllerUtils.startNetwork(w);
}); });
}; };
@ -65,25 +64,9 @@ angular.module('copayApp.controllers').controller('SigninController',
$rootScope.$flashMessage = { message: 'Unknown error', type: 'error'}; $rootScope.$flashMessage = { message: 'Unknown error', type: 'error'};
controllerUtils.onErrorDigest(); controllerUtils.onErrorDigest();
} else { } else {
controllerUtils.startNetwork(w); controllerUtils.startNetwork(w, $scope);
installStartupHandlers(w);
} }
}); });
}); });
};
function installStartupHandlers(wallet) {
wallet.on('serverError', function(msg) {
$rootScope.$flashMessage = {
message: 'There was an error connecting to the PeerJS server.'
+(msg||'Check you settings and Internet connection.'),
type: 'error',
};
controllerUtils.onErrorDigest($scope);
});
wallet.on('ready', function() {
$scope.loading = false;
});
} }
}); });

View file

@ -1,4 +1,5 @@
'use strict'; 'use strict';
var bitcore = require('bitcore');
angular.module('copayApp.controllers').controller('TransactionsController', angular.module('copayApp.controllers').controller('TransactionsController',
function($scope, $rootScope, $timeout, controllerUtils) { function($scope, $rootScope, $timeout, controllerUtils) {
@ -10,10 +11,10 @@ angular.module('copayApp.controllers').controller('TransactionsController',
$scope.txpCurrentPage = 1; $scope.txpCurrentPage = 1;
$scope.txpItemsPerPage = 4; $scope.txpItemsPerPage = 4;
var COIN = 100000000;
$scope.blockchain_txs = []; $scope.blockchain_txs = [];
var satToUnit = 1 / config.unitToSatoshi;
$scope.update = function() { $scope.update = function() {
$scope.loading = false; $scope.loading = false;
var from = ($scope.txpCurrentPage - 1) * $scope.txpItemsPerPage; var from = ($scope.txpCurrentPage - 1) * $scope.txpItemsPerPage;
@ -81,14 +82,14 @@ angular.module('copayApp.controllers').controller('TransactionsController',
tmp[addr].doubleSpentIndex = tmp[addr].doubleSpentIndex || items[i].doubleSpentIndex; tmp[addr].doubleSpentIndex = tmp[addr].doubleSpentIndex || items[i].doubleSpentIndex;
tmp[addr].unconfirmedInput += items[i].unconfirmedInput; tmp[addr].unconfirmedInput += items[i].unconfirmedInput;
tmp[addr].dbError = tmp[addr].dbError || items[i].dbError; tmp[addr].dbError = tmp[addr].dbError || items[i].dbError;
tmp[addr].valueSat += Math.round(items[i].value * COIN); tmp[addr].valueSat += parseInt((items[i].value * bitcore.util.COIN).toFixed(0));
tmp[addr].items.push(items[i]); tmp[addr].items.push(items[i]);
tmp[addr].notAddr = notAddr; tmp[addr].notAddr = notAddr;
tmp[addr].count++; tmp[addr].count++;
} }
angular.forEach(tmp, function(v) { angular.forEach(tmp, function(v) {
v.value = v.value || parseInt(v.valueSat) / COIN; v.value = (parseInt(v.valueSat || 0).toFixed(0)) * satToUnit;
ret.push(v); ret.push(v);
}); });
return ret; return ret;
@ -106,10 +107,13 @@ angular.module('copayApp.controllers').controller('TransactionsController',
$rootScope.txAlertCount = 0; $rootScope.txAlertCount = 0;
var w = $rootScope.wallet; var w = $rootScope.wallet;
w.sendTx(ntxid, function(txid) { w.sendTx(ntxid, function(txid) {
$rootScope.$flashMessage = txid $rootScope.$flashMessage = txid ? {
? {type:'success', message: 'Transaction broadcasted. txid: ' + txid} type: 'success',
: {type:'error', message: 'There was an error sending the Transaction'} message: 'Transaction broadcasted. txid: ' + txid
; } : {
type: 'error',
message: 'There was an error sending the Transaction'
};
if (cb) return cb(); if (cb) return cb();
else $scope.update(); else $scope.update();
}); });
@ -131,8 +135,7 @@ angular.module('copayApp.controllers').controller('TransactionsController',
$scope.send(ntxid, function() { $scope.send(ntxid, function() {
$scope.update(); $scope.update();
}); });
} } else
else
$scope.update(); $scope.update();
} }
}); });
@ -150,13 +153,14 @@ angular.module('copayApp.controllers').controller('TransactionsController',
for (var i = 0; i < txs.length; i++) { for (var i = 0; i < txs.length; i++) {
txs[i].vinSimple = _aggregateItems(txs[i].vin); txs[i].vinSimple = _aggregateItems(txs[i].vin);
txs[i].voutSimple = _aggregateItems(txs[i].vout); txs[i].voutSimple = _aggregateItems(txs[i].vout);
txs[i].valueOut = ((txs[i].valueOut * bitcore.util.COIN).toFixed(0)) * satToUnit;
txs[i].fees = ((txs[i].fees * bitcore.util.COIN).toFixed(0)) * satToUnit;
$scope.blockchain_txs.push(txs[i]); $scope.blockchain_txs.push(txs[i]);
} }
$scope.loading = false; $scope.loading = false;
}, 10); }, 10);
}); });
} } else {
else {
$timeout(function() { $timeout(function() {
$scope.loading = false; $scope.loading = false;
$scope.lastShowed = false; $scope.lastShowed = false;
@ -174,7 +178,10 @@ angular.module('copayApp.controllers').controller('TransactionsController',
$rootScope.txAlertCount = 0; $rootScope.txAlertCount = 0;
var w = $rootScope.wallet; var w = $rootScope.wallet;
w.reject(ntxid); w.reject(ntxid);
$rootScope.$flashMessage = {type:'warning', message: 'Transaction rejected by you'}; $rootScope.$flashMessage = {
type: 'warning',
message: 'Transaction rejected by you'
};
$scope.loading = false; $scope.loading = false;
}; };

View file

@ -45,8 +45,8 @@ angular.module('copayApp.directives')
require: 'ngModel', require: 'ngModel',
link: function(scope, element, attrs, ctrl) { link: function(scope, element, attrs, ctrl) {
var val = function(value) { var val = function(value) {
var availableBalanceNum = ($rootScope.availableBalance * bitcore.util.COIN).toFixed(0); var availableBalanceNum = ($rootScope.availableBalance * config.unitToSatoshi).toFixed(0);
var vNum = Number((value * bitcore.util.COIN).toFixed(0)) + feeSat; var vNum = Number((value * config.unitToSatoshi).toFixed(0)) + feeSat;
if (typeof vNum == "number" && vNum > 0) { if (typeof vNum == "number" && vNum > 0) {
if (availableBalanceNum < vNum) { if (availableBalanceNum < vNum) {
ctrl.$setValidity('enoughAmount', false); ctrl.$setValidity('enoughAmount', false);
@ -128,7 +128,9 @@ angular.module('copayApp.directives')
link: function(scope, element, attrs) { link: function(scope, element, attrs) {
scope.$watch(attrs.highlightOnChange, function(newValue, oldValue) { scope.$watch(attrs.highlightOnChange, function(newValue, oldValue) {
element.addClass('highlight'); element.addClass('highlight');
setTimeout(function() { element.removeClass('highlight'); }, 500); setTimeout(function() {
element.removeClass('highlight');
}, 500);
}); });
} }
} }
@ -150,7 +152,9 @@ angular.module('copayApp.directives')
var numbers = /[0-9]+/.test(p); var numbers = /[0-9]+/.test(p);
var symbols = regex.test(p); var symbols = regex.test(p);
var flags = [lowerLetters, upperLetters, numbers, symbols]; var flags = [lowerLetters, upperLetters, numbers, symbols];
var passedMatches = flags.filter(function (el) { return !!el; }).length; var passedMatches = flags.filter(function(el) {
return !!el;
}).length;
force = 2 * p.length + (p.length >= 10 ? 1 : 0); force = 2 * p.length + (p.length >= 10 ? 1 : 0);
force += passedMatches * 10; force += passedMatches * 10;
@ -167,20 +171,32 @@ angular.module('copayApp.directives')
getColor: function(s) { getColor: function(s) {
var idx = 0; var idx = 0;
if (s <= 10) { idx = 0; } if (s <= 10) {
else if (s <= 20) { idx = 1; } idx = 0;
else if (s <= 30) { idx = 2; } } else if (s <= 20) {
else if (s <= 40) { idx = 3; } idx = 1;
else { idx = 4; } } else if (s <= 30) {
idx = 2;
} else if (s <= 40) {
idx = 3;
} else {
idx = 4;
}
return { idx: idx + 1, col: this.colors[idx], message: this.messages[idx] }; return {
idx: idx + 1,
col: this.colors[idx],
message: this.messages[idx]
};
} }
}; };
scope.$watch(attrs.ngModel, function(newValue, oldValue) { scope.$watch(attrs.ngModel, function(newValue, oldValue) {
if (newValue && newValue !== '') { if (newValue && newValue !== '') {
var c = strength.getColor(strength.mesureStrength(newValue)); var c = strength.getColor(strength.mesureStrength(newValue));
element.css({ 'border-color': c.col }); element.css({
'border-color': c.col
});
scope[attrs.checkStrength] = c.message; scope[attrs.checkStrength] = c.message;
} }
}); });

View file

@ -189,28 +189,29 @@ Insight.prototype._request = function(options, callback) {
}; };
request.onreadystatechange = function() { request.onreadystatechange = function() {
if (request.readyState === 4) { if (request.readyState !== 4) return;
var ret, errTxt, e;
if (request.status === 200 || request.status === 304) { if (request.status === 200 || request.status === 304) {
try { try {
var ret = JSON.parse(request.responseText); ret = JSON.parse(request.responseText);
return callback(null, ret); } catch (e2) {
} catch (e) { errTxt = 'CRITICAL: Wrong response from insight' + e2;
return callback(new Error('CRITICAL: Wrong response from insight'));
} }
} } else if (request.status >= 400 && request.status < 499) {
// User error errTxt = 'CRITICAL: Bad request to insight. Probably wrong transaction to broadcast?.';
else if (request.status >= 400 && request.status < 499) { } else {
return callback(new Error('CRITICAL: Bad request to insight. Probably wrong transaction to broadcast?.')); errTxt = 'Error code: ' + request.status + ' - Status: ' + request.statusText + ' - Description: ' + request.responseText;
}
else {
var err= 'Error code: ' + request.status + ' - Status: ' + request.statusText
+ ' - Description: ' + request.responseText;
setTimeout(function() { setTimeout(function() {
console.log('### Retrying Insight Request....');
return self._request(options, callback); return self._request(options, callback);
}, self.retryDelay); }, self.retryDelay);
return callback(new Error(err));
} }
if (errTxt) {
console.log("INSIGHT ERROR:", e);
e = new Error(errTxt);
} }
return callback(e, ret);
}; };
if (options.method === 'POST') { if (options.method === 'POST') {
@ -218,9 +219,7 @@ Insight.prototype._request = function(options, callback) {
} }
request.send(options.data || null); request.send(options.data || null);
} } else {
else {
var http = require('http'); var http = require('http');
var req = http.request(options, function(response) { var req = http.request(options, function(response) {
var ret; var ret;

View file

@ -1,4 +1,3 @@
'use strict'; 'use strict';
@ -94,15 +93,17 @@ TxProposals.prototype.toObj = function(onlyThisNtxid) {
}; };
TxProposals.prototype._startMerge = function(myTxps, theirTxps) { TxProposals.prototype._startMerge = function(myTxps, theirTxps) {
var fromUs=0, fromTheirs=0, merged =0; var fromUs = 0,
var toMerge = {}, ready={}; fromTheirs = 0,
merged = 0;
var toMerge = {},
ready = {};
for (var hash in theirTxps) { for (var hash in theirTxps) {
if (!myTxps[hash]) { if (!myTxps[hash]) {
ready[hash] = theirTxps[hash]; // only in theirs; ready[hash] = theirTxps[hash]; // only in theirs;
fromTheirs++; fromTheirs++;
} } else {
else {
toMerge[hash] = theirTxps[hash]; // need Merging toMerge[hash] = theirTxps[hash]; // need Merging
merged++; merged++;
} }
@ -212,7 +213,9 @@ TxProposals.prototype.getTxProposal = function(ntxid, copayers) {
} }
for (var p in txp.seenBy) { for (var p in txp.seenBy) {
i.peerActions[p]={seen: txp.seenBy[p]}; i.peerActions[p] = {
seen: txp.seenBy[p]
};
} }
for (var p in txp.signedBy) { for (var p in txp.signedBy) {
i.peerActions[p] = i.peerActions[p] || {}; i.peerActions[p] = i.peerActions[p] || {};

View file

@ -562,11 +562,12 @@ Wallet.prototype.addressIsOwn = function(addrStr, opts) {
return ret; return ret;
}; };
//retunrs values in SATOSHIs
Wallet.prototype.getBalance = function(cb) { Wallet.prototype.getBalance = function(cb) {
var balance = 0; var balance = 0;
var safeBalance = 0; var safeBalance = 0;
var balanceByAddr = {}; var balanceByAddr = {};
var COIN = bitcore.util.COIN; var COIN = coinUtil.COIN;
this.getUnspent(function(err, safeUnspent, unspent) { this.getUnspent(function(err, safeUnspent, unspent) {
if (err) { if (err) {
@ -580,11 +581,12 @@ Wallet.prototype.getBalance = function(cb) {
balanceByAddr[u.address] = (balanceByAddr[u.address] || 0) + amt; balanceByAddr[u.address] = (balanceByAddr[u.address] || 0) + amt;
} }
// we multiply and divide by COIN to avoid rounding errors when adding // we multiply and divide by BIT to avoid rounding errors when adding
for (var a in balanceByAddr) { for (var a in balanceByAddr) {
balanceByAddr[a] = balanceByAddr[a].toFixed(0) / COIN; balanceByAddr[a] = parseInt(balanceByAddr[a].toFixed(0));
} }
balance = balance / COIN;
balance = parseInt(balance.toFixed(0));
for (var i = 0; i < safeUnspent.length; i++) { for (var i = 0; i < safeUnspent.length; i++) {
var u = safeUnspent[i]; var u = safeUnspent[i];
@ -592,7 +594,7 @@ Wallet.prototype.getBalance = function(cb) {
safeBalance += amt; safeBalance += amt;
} }
safeBalance = safeBalance.toFixed(0) / COIN; safeBalance = parseInt(safeBalance.toFixed(0));
return cb(null, balance, balanceByAddr, safeBalance); return cb(null, balance, balanceByAddr, safeBalance);
}); });
}; };

View file

@ -1,9 +1,9 @@
'use strict'; 'use strict';
var bitcore = require('bitcore');
angular.module('copayApp.services') angular.module('copayApp.services')
.factory('controllerUtils', function($rootScope, $sce, $location, $notification, Socket, video) { .factory('controllerUtils', function($rootScope, $sce, $location, $notification, $timeout, Socket, video) {
var root = {}; var root = {};
var bitcore = require('bitcore');
root.getVideoMutedStatus = function(copayer) { root.getVideoMutedStatus = function(copayer) {
var vi = $rootScope.videoInfo[copayer] var vi = $rootScope.videoInfo[copayer]
@ -39,9 +39,42 @@ angular.module('copayApp.services')
message: msg message: msg
}; };
$rootScope.$digest(); $rootScope.$digest();
} };
root.installStartupHandlers = function(wallet, $scope) {
wallet.on('serverError', function(msg) {
$rootScope.$flashMessage = {
message: 'There was an error connecting to the PeerJS server.' + (msg || 'Check you settings and Internet connection.'),
type: 'error',
};
root.onErrorDigest($scope);
$location.path('addresses');
});
wallet.on('connectionError', function() {
var message = "Looks like you are already connected to this wallet, please logout from it and try importing it again.";
$rootScope.$flashMessage = {
message: message,
type: 'error'
};
root.onErrorDigest($scope);
});
wallet.on('serverError', function() {
$rootScope.$flashMessage = {
message: 'The PeerJS server is not responding, please try again',
type: 'error'
};
root.onErrorDigest($scope);
});
wallet.on('ready', function() {
$scope.loading = false;
});
};
root.startNetwork = function(w, $scope) {
root.installStartupHandlers(w, $scope);
root.startNetwork = function(w) {
var handlePeerVideo = function(err, peerID, url) { var handlePeerVideo = function(err, peerID, url) {
if (err) { if (err) {
delete $rootScope.videoInfo[peerID]; delete $rootScope.videoInfo[peerID];
@ -77,12 +110,17 @@ angular.module('copayApp.services')
} }
}); });
w.on('txProposalsUpdated', function(dontDigest) { w.on('txProposalsUpdated', function(dontDigest) {
root.updateTxs({onlyPending:true}); root.updateTxs({
onlyPending: true
});
// give sometime to the tx to propagate.
$timeout(function() {
root.updateBalance(function() { root.updateBalance(function() {
if (!dontDigest) { if (!dontDigest) {
$rootScope.$digest(); $rootScope.$digest();
} }
}); });
}, 3000);
}); });
w.on('connectionError', function(msg) { w.on('connectionError', function(msg) {
root.onErrorDigest(null, msg); root.onErrorDigest(null, msg);
@ -109,22 +147,30 @@ angular.module('copayApp.services')
var w = $rootScope.wallet; var w = $rootScope.wallet;
if (!w) return root.onErrorDigest(); if (!w) return root.onErrorDigest();
$rootScope.balanceByAddr = {}; $rootScope.balanceByAddr = {};
$rootScope.updatingBalance = true; $rootScope.updatingBalance = true;
w.getBalance(function(err, balance, balanceByAddr, safeBalance) {
w.getBalance(function(err, balanceSat, balanceByAddrSat, safeBalanceSat) {
if (err) { if (err) {
console.error('Error: ' + err.message); //TODO console.error('Error: ' + err.message); //TODO
root._setCommError(); root._setCommError();
return null; return null;
} } else {
else {
root._clearCommError(); root._clearCommError();
} }
$rootScope.totalBalance = balance; var satToUnit = 1 / config.unitToSatoshi;
var COIN = bitcore.util.COIN;
$rootScope.totalBalance = balanceSat * satToUnit;
$rootScope.totalBalanceBTC = (balanceSat / COIN).toFixed(4);
$rootScope.availableBalance = safeBalanceSat * satToUnit;
$rootScope.availableBalanceBTC = (safeBalanceSat / COIN).toFixed(4);
var balanceByAddr = {};
for (var ii in balanceByAddrSat) {
balanceByAddr[ii] = balanceByAddrSat[ii] * satToUnit;
}
$rootScope.balanceByAddr = balanceByAddr; $rootScope.balanceByAddr = balanceByAddr;
$rootScope.availableBalance = safeBalance;
root.updateAddressList(); root.updateAddressList();
$rootScope.updatingBalance = false; $rootScope.updatingBalance = false;
return cb ? cb() : null; return cb ? cb() : null;
@ -136,9 +182,12 @@ angular.module('copayApp.services')
if (!w) return; if (!w) return;
opts = opts || {}; opts = opts || {};
var satToUnit = 1 / config.unitToSatoshi;
var myCopayerId = w.getMyCopayerId(); var myCopayerId = w.getMyCopayerId();
var pendingForUs = 0; var pendingForUs = 0;
var inT = w.getTxProposals().sort(function(t1, t2) { return t2.createdTs - t1.createdTs }); var inT = w.getTxProposals().sort(function(t1, t2) {
return t2.createdTs - t1.createdTs
});
var txs = []; var txs = [];
inT.forEach(function(i, index) { inT.forEach(function(i, index) {
@ -157,16 +206,18 @@ angular.module('copayApp.services')
var outs = []; var outs = [];
tx.outs.forEach(function(o) { tx.outs.forEach(function(o) {
var addr = bitcore.Address.fromScriptPubKey(o.getScript(), config.networkName)[0].toString(); var addr = bitcore.Address.fromScriptPubKey(o.getScript(), config.networkName)[0].toString();
if (!w.addressIsOwn(addr, {excludeMain:true})) { if (!w.addressIsOwn(addr, {
excludeMain: true
})) {
outs.push({ outs.push({
address: addr, address: addr,
value: bitcore.util.valueToBigInt(o.getValue())/bitcore.util.COIN, value: bitcore.util.valueToBigInt(o.getValue()) * satToUnit,
}); });
} }
}); });
// extra fields // extra fields
i.outs = outs; i.outs = outs;
i.fee = i.builder.feeSat/bitcore.util.COIN; i.fee = i.builder.feeSat * satToUnit;
i.missingSignatures = tx.countInputMissingSignatures(0); i.missingSignatures = tx.countInputMissingSignatures(0);
txs.push(i); txs.push(i);
} }

View file

@ -43,11 +43,11 @@ module.exports = function(config) {
'lib/chai/chai.js', 'lib/chai/chai.js',
'test/lib/chai-should.js', 'test/lib/chai-should.js',
'test/lib/chai-expect.js', 'test/lib/chai-expect.js',
'test/mocks/FakeWallet.js',
//Mocha stuff //Mocha stuff
'test/mocha.conf.js', 'test/mocha.conf.js',
//App-specific Code //App-specific Code
'js/app.js', 'js/app.js',
'js/routes.js', 'js/routes.js',

View file

@ -18,7 +18,6 @@
var copay = require('copay'); var copay = require('copay');
</script> </script>
<script src="test.blockchain.Insight.js"></script> <script src="test.blockchain.Insight.js"></script>
<script src="test.performance.js"></script>
<script src="test.PrivateKey.js"></script> <script src="test.PrivateKey.js"></script>
<script src="test.PublicKeyRing.js"></script> <script src="test.PublicKeyRing.js"></script>
<script src="test.storage.LocalEncrypted.js"></script> <script src="test.storage.LocalEncrypted.js"></script>
@ -26,6 +25,7 @@
<script src="test.TxProposals.js"></script> <script src="test.TxProposals.js"></script>
<script src="test.Wallet.js"></script> <script src="test.Wallet.js"></script>
<script src="test.Walletfactory.js"></script> <script src="test.Walletfactory.js"></script>
<script src="test.performance.js"></script>
<!-- <!--
--> -->
<script> <script>

View file

@ -14,7 +14,7 @@ FakeBlockchain.prototype.getTransactions = function(addresses, cb) {
FakeBlockchain.prototype.fixUnspent = function(u) { FakeBlockchain.prototype.fixUnspent = function(u) {
this.u = u; this.u = u;
} };
FakeBlockchain.prototype.getUnspent = function(addresses, cb) { FakeBlockchain.prototype.getUnspent = function(addresses, cb) {
if (!addresses || !addresses.length) return cb(null, []); if (!addresses || !addresses.length) return cb(null, []);

34
test/mocks/FakeWallet.js Normal file
View file

@ -0,0 +1,34 @@
var FakeWallet = function(){
this.balance=10000;
this.safeBalance=1000;
this.balanceByAddr={'1CjPR7Z5ZSyWk6WtXvSFgkptmpoi4UM9BC': 1000};
};
FakeWallet.prototype.set = function(balance, safeBalance, balanceByAddr){
this.balance=balance;
this.safeBalance = safeBalance;
this.balanceByAddr = balanceByAddr;
};
FakeWallet.prototype.getAddressesInfo=function(){
var ret = [];
for(var ii in this.balanceByAddr){
ret.push({
address: ii,
isChange: false,
});
}
return ret;
};
FakeWallet.prototype.getBalance=function(cb){
return cb(null, this.balance, this.balanceByAddr, this.safeBalance);
};
// This mock is meant for karma, module.exports is not necesary.
try {
module.exports = require('soop')(FakeWallet);
} catch (e) {}

View file

@ -308,15 +308,21 @@ describe('TxProposals model', function() {
it('#merge, merge signatures case 2', function() { it('#merge, merge signatures case 2', function() {
var o1 ={ extendedPrivateKeyString: 'tprv8ZgxMBicQKsPdSF1avR6mXyDj5Uv1XY2UyUHSDpAXQ5TvPN7prGeDppjy4562rBB9gMMAhRfFdJrNDpQ4t69kkqHNEEen3PX1zBJqSehJDH', var o1 = {
extendedPrivateKeyString: 'tprv8ZgxMBicQKsPdSF1avR6mXyDj5Uv1XY2UyUHSDpAXQ5TvPN7prGeDppjy4562rBB9gMMAhRfFdJrNDpQ4t69kkqHNEEen3PX1zBJqSehJDH',
networkName: 'testnet', networkName: 'testnet',
privateKeyCache: {} }; privateKeyCache: {}
var o2 ={ extendedPrivateKeyString: 'tprv8ZgxMBicQKsPdVeB5RzuxS9JQcACueZYgUaM5eWzaEBkHjW5Pg6Mqez1APSqoUP1jUdbT8WVG7ZJYTXvUL7XtPzFYBXjmdKuwSor1dcNQ8j', };
var o2 = {
extendedPrivateKeyString: 'tprv8ZgxMBicQKsPdVeB5RzuxS9JQcACueZYgUaM5eWzaEBkHjW5Pg6Mqez1APSqoUP1jUdbT8WVG7ZJYTXvUL7XtPzFYBXjmdKuwSor1dcNQ8j',
networkName: 'testnet', networkName: 'testnet',
privateKeyCache: {} }; privateKeyCache: {}
var o3 ={ extendedPrivateKeyString: 'tprv8ZgxMBicQKsPeHWNrPVZtQVgcCtXBr5TACNbDQ56rwqNJce9MEc64US6DJKxpWsrebEomxxWZFDtkvkZGkzA43uLvdF4XHiWqoNaL6Dq2Gd', };
var o3 = {
extendedPrivateKeyString: 'tprv8ZgxMBicQKsPeHWNrPVZtQVgcCtXBr5TACNbDQ56rwqNJce9MEc64US6DJKxpWsrebEomxxWZFDtkvkZGkzA43uLvdF4XHiWqoNaL6Dq2Gd',
networkName: 'testnet', networkName: 'testnet',
privateKeyCache: {} }; privateKeyCache: {}
};
var priv = PrivateKey.fromObj(o1); var priv = PrivateKey.fromObj(o1);

View file

@ -440,9 +440,69 @@ describe('Wallet model', function() {
r.length.should.equal(2); r.length.should.equal(2);
r[0].should.not.equal(r[1]); r[0].should.not.equal(r[1]);
}); });
it('#getBalance should call #getUnspent', function(done) {
var w = createW2();
var spy = sinon.spy(w.blockchain, 'getUnspent');
w.generateAddress();
w.getBalance(function(err, balance, balanceByAddr, safeBalance) {
sinon.assert.callCount(spy, 1);
done();
});
});
it('#getBalance should return values in satoshis', function(done) {
var w = createW2();
w.generateAddress();
w.getBalance(function(err, balance, balanceByAddr, safeBalance) {
balance.should.equal(2500010000);
safeBalance.should.equal(2500010000);
balanceByAddr.mji7zocy8QzYywQakwWf99w9bCT6orY1C1.should.equal(2500010000);
Object.keys(balanceByAddr).length.should.equal(1);
done();
});
});
var roundErrorChecks = [{
unspent: [1.0001],
balance: 100010000
}, {
unspent: [1.0002, 1.0003, 1.0004],
balance: 300090000
}, {
unspent: [0.000002, 1.000003, 2.000004],
balance: 300000900
}, {
unspent: [0.0001, 0.0003],
balance: 40000
}, {
unspent: [0.0001, 0.0003, 0.0001, 0.0001, 0.0001, 0.0001, 0.0001, 0.0002],
balance: 110000
},
];
roundErrorChecks.forEach(function(c) {
it('check rounding errors ' + c.unspent[0], function(done) {
var w = createW2();
w.generateAddress();
w.blockchain.fixUnspent(c.unspent.map(function(u) {
return {
amount: u
}
}));
w.getBalance(function(err, balance, balanceByAddr, safeBalance) {
balance.should.equal(c.balance);
done();
});
});
});
it('should get balance', function(done) { it('should get balance', function(done) {
var w = createW(); var w = createW();
var spy = sinon.spy(w.blockchain, 'getUnspent');
w.getBalance(function(err, balance, balanceByAddr, safeBalance) { w.getBalance(function(err, balance, balanceByAddr, safeBalance) {
sinon.assert.callCount(spy, 1);
balance.should.equal(0); balance.should.equal(0);
done(); done();
}); });

View file

@ -7,7 +7,7 @@ if (typeof process === 'undefined' || !process.version) {
var copay = copay || require('../copay'); var copay = copay || require('../copay');
var LocalPlain = copay.StorageLocalPlain; var LocalPlain = copay.StorageLocalPlain;
describe('Storage/LocalPlain model', function() { describe.skip('Storage/LocalPlain model', function() {
it('should create an instance', function() { it('should create an instance', function() {
var s = new LocalPlain(); var s = new LocalPlain();
@ -17,7 +17,12 @@ if (typeof process === 'undefined' || !process.version) {
describe('#setFromObj', function() { describe('#setFromObj', function() {
it('should set keys from an object', function() { it('should set keys from an object', function() {
localStorage.clear(); localStorage.clear();
var obj = {test:'testval', opts: {name: 'testname'}}; var obj = {
test: 'testval',
opts: {
name: 'testname'
}
};
var storage = new LocalPlain(); var storage = new LocalPlain();
storage.setFromObj('walletId', obj); storage.setFromObj('walletId', obj);
storage.get('walletId', 'test').should.equal('testval'); storage.get('walletId', 'test').should.equal('testval');

View file

@ -110,6 +110,7 @@ describe("Unit: Controllers", function() {
expect(rootScope.insightError).equal(1); expect(rootScope.insightError).equal(1);
scope.$apply(); scope.$apply();
}); });
}); });
}); });

View file

@ -8,6 +8,14 @@ describe("Unit: Testing Directives", function() {
beforeEach(module('copayApp.directives')); beforeEach(module('copayApp.directives'));
describe('Check config', function() {
it('unit should be set to BITS in config.js', function() {
expect(config.unitToSatoshi).to.equal(100);
expect(config.unitName).to.equal('bits');
});
});
describe('Validate Address', function() { describe('Validate Address', function() {
beforeEach(inject(function($compile, $rootScope) { beforeEach(inject(function($compile, $rootScope) {
$scope = $rootScope; $scope = $rootScope;
@ -16,7 +24,9 @@ describe("Unit: Testing Directives", function() {
'<input type="text" id="address" name="address" placeholder="Send to" ng-model="address" valid-address required>' + '<input type="text" id="address" name="address" placeholder="Send to" ng-model="address" valid-address required>' +
'</form>' '</form>'
); );
$scope.model = { address: null }; $scope.model = {
address: null
};
$compile(element)($scope); $compile(element)($scope);
$scope.$digest(); $scope.$digest();
form = $scope.form; form = $scope.form;
@ -35,32 +45,37 @@ describe("Unit: Testing Directives", function() {
describe('Validate Amount', function() { describe('Validate Amount', function() {
beforeEach(inject(function($compile, $rootScope) { beforeEach(inject(function($compile, $rootScope) {
$scope = $rootScope; $scope = $rootScope;
$rootScope.availableBalance = 0.101; $rootScope.availableBalance = 1000;
var element = angular.element( var element = angular.element(
'<form name="form">' + '<form name="form">' +
'<input type="number" id="amount" name="amount" placeholder="Amount" ng-model="amount" min="0.0001" max="10000000" enough-amount required>' + '<input type="number" id="amount" name="amount" placeholder="Amount" ng-model="amount" min="0.0001" max="10000000" enough-amount required>' +
'</form>' '</form>'
); );
$scope.model = { amount: null }; $scope.model = {
amount: null
};
$compile(element)($scope); $compile(element)($scope);
$scope.$digest(); $scope.$digest();
form = $scope.form; form = $scope.form;
})); }));
it('should validate', function() { it('should validate', function() {
form.amount.$setViewValue(0.1); form.amount.$setViewValue(100);
expect(form.amount.$invalid).to.equal(false); expect(form.amount.$invalid).to.equal(false);
form.amount.$setViewValue(0.1009); form.amount.$setViewValue(900);
expect(form.amount.$invalid).to.equal(false); expect(form.amount.$invalid).to.equal(false);
}); });
it('should not validate', function() { it('should not validate', function() {
form.amount.$setViewValue(0); form.amount.$setViewValue(0);
expect(form.amount.$invalid).to.equal(true); expect(form.amount.$invalid).to.equal(true);
form.amount.$setViewValue(9999999999); form.amount.$setViewValue(9999999999);
expect(form.amount.$invalid).to.equal(true); expect(form.amount.$invalid).to.equal(true);
form.amount.$setViewValue(2.1); form.amount.$setViewValue(901);
expect(form.amount.$invalid).to.equal(true); expect(form.amount.$invalid).to.equal(true);
form.amount.$setViewValue(0.10091); form.amount.$setViewValue(1000);
expect(form.amount.$invalid).to.equal(true); expect(form.amount.$invalid).to.equal(true);
}); });
}); });

View file

@ -1,8 +1,17 @@
// //
// test/unit/services/servicesSpec.js // test/unit/services/servicesSpec.js
// //
describe("Unit: Testing Services", function() { //
//
describe('Check config', function() {
it('unit should be set to BITS in config.js', function() {
expect(config.unitToSatoshi).to.equal(100);
expect(config.unitName).to.equal('bits');
});
});
describe("Unit: Socket Service", function() {
beforeEach(angular.mock.module('copayApp.services')); beforeEach(angular.mock.module('copayApp.services'));
it('should contain a Socket service', inject(function(Socket) { it('should contain a Socket service', inject(function(Socket) {
@ -17,7 +26,7 @@ describe("Unit: Testing Services", function() {
it('Socket should support #sysOn', inject(function(Socket) { it('Socket should support #sysOn', inject(function(Socket) {
expect(Socket.sysOn).to.be.a('function'); expect(Socket.sysOn).to.be.a('function');
})) }));
it('Socket should add handlers with #on', inject(function(Socket) { it('Socket should add handlers with #on', inject(function(Socket) {
@ -40,19 +49,41 @@ describe("Unit: Testing Services", function() {
ret = Socket.getListeners(); ret = Socket.getListeners();
expect(Object.keys(ret)).to.have.length(0); expect(Object.keys(ret)).to.have.length(0);
})); }));
});
describe("Unit: Walletfactory Service", function() {
beforeEach(angular.mock.module('copayApp.services'));
it('should contain a walletFactory service', inject(function(walletFactory) { it('should contain a walletFactory service', inject(function(walletFactory) {
expect(walletFactory).not.to.equal(null); expect(walletFactory).not.to.equal(null);
})); }));
});
// TODO describe("Unit: controllerUtils", function() {
/*beforeEach(angular.mock.module('copayApp.controllerUtils')); beforeEach(angular.mock.module('copayApp.services'));
beforeEach(angular.mock.module('notifications'));
it('should contain a controllerUtils service', inject(function(controllerUtils) {
expect(controllerUtils).not.to.equal(null); it('should updateBalance in bits', inject(function(controllerUtils, $rootScope) {
})); expect(controllerUtils.updateBalance).not.to.equal(null);
*/ scope = $rootScope.$new();
$rootScope.wallet = new FakeWallet();
var addr = '1CjPR7Z5ZSyWk6WtXvSFgkptmpoi4UM9BC';
var a = {};
a[addr] = 100;
//SATs
$rootScope.wallet.set(100000001, 90000002, a);
//retuns values in DEFAULT UNIT(bits)
controllerUtils.updateBalance(function() {
expect($rootScope.totalBalanceBTC).to.be.equal('1.0000');
expect($rootScope.availableBalanceBTC).to.be.equal('0.9000');
expect($rootScope.totalBalance).to.be.equal(1000000.01);
expect($rootScope.availableBalance).to.be.equal(900000.02);
expect($rootScope.addrInfos).not.to.equal(null);
expect($rootScope.addrInfos[0].address).to.equal(addr);
});
}));
}); });