From d08c1e7e2a634f17882fc12a468cbc0c0cc479e3 Mon Sep 17 00:00:00 2001 From: Esteban Ordano Date: Tue, 25 Nov 2014 13:54:39 -0300 Subject: [PATCH 1/4] Update .jshintrc --- .jshintrc | 55 ++++++++++++++++++++++++++++++------------------------- 1 file changed, 30 insertions(+), 25 deletions(-) diff --git a/.jshintrc b/.jshintrc index b15caaa..6251ca5 100644 --- a/.jshintrc +++ b/.jshintrc @@ -1,25 +1,34 @@ { - "node": true, // Enable globals available when code is running inside of the NodeJS runtime environment. - "browser": true, // Standard browser globals e.g. `window`, `document`. - "esnext": true, // Allow ES.next specific features such as `const` and `let`. - "bitwise": false, // Prohibit bitwise operators (&, |, ^, etc.). - "camelcase": false, // Permit only camelcase for `var` and `object indexes`. - "curly": false, // Require {} for every new block or scope. - "eqeqeq": true, // Require triple equals i.e. `===`. - "immed": true, // Require immediate invocations to be wrapped in parens e.g. `( function(){}() );` - "latedef": true, // Prohibit variable use before definition. - "newcap": false, // Require capitalization of all constructor functions e.g. `new F()`. - "noarg": true, // Prohibit use of `arguments.caller` and `arguments.callee`. + "bitwise": false, // Prohibit bitwise operators (&, |, ^, etc.). + "browser": true, // Standard browser globals e.g. `window`, `document`. + "camelcase": false, // Permit only camelcase for `var` and `object indexes`. + "curly": true, // Require {} for every new block or scope. + "devel": false, // Allow development statements e.g. `console.log();`. + "eqeqeq": true, // Require triple equals i.e. `===`. + "esnext": true, // Allow ES.next specific features such as `const` and `let`. + "freeze": true, // Forbid overwriting prototypes of native objects such as Array, Date and so on. + "immed": true, // Require immediate invocations to be wrapped in parens e.g. `( function(){}() );` + "indent": 2, // Specify indentation spacing + "latedef": true, // Prohibit variable use before definition. + "newcap": false, // Require capitalization of all constructor functions e.g. `new F()`. + "noarg": true, // Prohibit use of `arguments.caller` and `arguments.callee`. + "node": true, // Enable globals available when code is running inside of the NodeJS runtime environment. + "noempty": true, // Prohibit use of empty blocks. + "nonew": true, // Prohibits the use of constructor functions for side-effects "quotmark": "single", // Define quotes to string values. - "regexp": true, // Prohibit `.` and `[^...]` in regular expressions. - "undef": true, // Require all non-global variables be declared before they are used. - "unused": true, // Warn unused variables. - "strict": true, // Require `use strict` pragma in every file. - "trailing": true, // Prohibit trailing whitespaces. - "smarttabs": false, // Suppresses warnings about mixed tabs and spaces - "globals": { // Globals variables. - "angular": true - }, + "regexp": true, // Prohibit `.` and `[^...]` in regular expressions. + "smarttabs": false, // Supress warnings about mixed tabs and spaces + "strict": true, // Require `use strict` pragma in every file. + "trailing": true, // Prohibit trailing whitespaces. + "undef": true, // Require all non-global variables be declared before they are used. + "unused": true, // Warn unused variables. + + "maxparams": 4, // Maximum number of parameters for a function + "maxstatements": 15, // Maximum number of statements in a function + "maxcomplexity": 4, // Cyclomatic complexity (http://en.wikipedia.org/wiki/Cyclomatic_complexity) + "maxdepth": 4, // Maximum depth of nested control structures + "maxlen": 600 // Maximum number of lines of code in a file + "predef": [ // Extra globals. "define", "require", @@ -30,10 +39,6 @@ "beforeEach", "after", "afterEach", - "requirejs", "it" - ], - "indent": false, // Specify indentation spacing - "devel": true, // Allow development statements e.g. `console.log();`. - "noempty": true // Prohibit use of empty blocks. + ] } From 665b77518f79482405eeb78e41a39958fc574e19 Mon Sep 17 00:00:00 2001 From: Esteban Ordano Date: Tue, 25 Nov 2014 14:43:25 -0300 Subject: [PATCH 2/4] Update Transaction.md --- docs/Transaction.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/Transaction.md b/docs/Transaction.md index f8d6333..7c18d4b 100644 --- a/docs/Transaction.md +++ b/docs/Transaction.md @@ -49,7 +49,7 @@ You can take a look at the javadocs for the [Transaction class here](link missing). This document will go over the expected high level use cases. * from(utxo) -* fromMultisig(utxo, pubkeys, threshold) +* from(utxo, pubkeys, threshold) * change(address) * fee(amount) * usingStrategy(strategy) From 8c4301cb0a778dde39b51ac8222572ca11bafeba Mon Sep 17 00:00:00 2001 From: Braydon Fuller Date: Tue, 25 Nov 2014 14:20:43 -0500 Subject: [PATCH 3/4] Tests: Updated tests to run from sub-directories and fixed crypto and encoding related missing updates. --- browser/build | 2 +- lib/crypto/ecdsa.js | 40 ++++++++++++++++++++++------------ package.json | 4 ++-- test/crypto/bn.js | 2 +- test/crypto/ecdsa.js | 41 ++++++++++++++++------------------- test/crypto/hash.js | 2 +- test/crypto/point.js | 4 ++-- test/crypto/random.js | 2 +- test/encoding/base58.js | 2 +- test/encoding/base58check.js | 4 ++-- test/encoding/bufferreader.js | 6 ++--- test/encoding/bufferwriter.js | 6 ++--- test/encoding/varint.js | 8 +++---- 13 files changed, 66 insertions(+), 57 deletions(-) diff --git a/browser/build b/browser/build index fca4b11..fc2d706 100755 --- a/browser/build +++ b/browser/build @@ -1,4 +1,4 @@ #!/bin/bash browserify index.js -o browser/bitcore.js -ls test/*.js | xargs browserify -o browser/tests.js +find test/ -type f -name "*.js" | xargs browserify -o browser/tests.js diff --git a/lib/crypto/ecdsa.js b/lib/crypto/ecdsa.js index 886af02..24ae00c 100644 --- a/lib/crypto/ecdsa.js +++ b/lib/crypto/ecdsa.js @@ -4,6 +4,7 @@ var BN = require('./bn'); var Point = require('./point'); var Random = require('./random'); var Pubkey = require('../pubkey'); +var Privkey = require('../privkey'); var Signature = require('../signature'); var ECDSA = function ECDSA(obj) { @@ -15,24 +16,29 @@ var ECDSA = function ECDSA(obj) { ECDSA.prototype.set = function(obj) { this.hashbuf = obj.hashbuf || this.hashbuf; - this.keypair = obj.keypair || this.keypair; + this.privkey = obj.privkey || this.privkey; + this.pubkey = obj.pubkey || this.pubkey; this.sig = obj.sig || this.sig; this.k = obj.k || this.k; this.verified = obj.verified || this.verified; return this; }; +ECDSA.prototype.privkey2pubkey = function(){ + this.pubkey = Pubkey().fromPrivkey(this.privkey); +}; + ECDSA.prototype.calci = function() { for (var i = 0; i < 4; i++) { this.sig.i = i; + var Qprime; try { - var Qprime = this.sig2pubkey(); + Qprime = this.sig2pubkey(); } catch (e) { continue; } - - if (Qprime.point.eq(this.keypair.pubkey.point)) { - this.sig.compressed = this.keypair.pubkey.compressed; + if (Qprime.point.eq(this.pubkey.point)) { + this.sig.compressed = this.pubkey.compressed; return this; } } @@ -45,8 +51,10 @@ ECDSA.prototype.fromString = function(str) { var obj = JSON.parse(str); if (obj.hashbuf) this.hashbuf = new Buffer(obj.hashbuf, 'hex'); - if (obj.keypair) - this.keypair = Keypair().fromString(obj.keypair); + if (obj.pubkey) + this.pubkey = Pubkey().fromString(obj.pubkey); + if (obj.privkey) + this.privkey = Privkey().fromString(obj.privkey); if (obj.sig) this.sig = Signature().fromString(obj.sig); if (obj.k) @@ -118,7 +126,7 @@ ECDSA.prototype.sigError = function() { return 'hashbuf must be a 32 byte buffer'; try { - this.keypair.pubkey.validate(); + this.pubkey.validate(); } catch (e) { return 'Invalid pubkey: ' + e; } @@ -135,7 +143,7 @@ ECDSA.prototype.sigError = function() { var u1 = sinv.mul(e).mod(n); var u2 = sinv.mul(r).mod(n); - var p = Point.getG().mulAdd(u1, this.keypair.pubkey.point, u2); + var p = Point.getG().mulAdd(u1, this.pubkey.point, u2); if (p.isInfinity()) return 'p is infinity'; @@ -147,7 +155,7 @@ ECDSA.prototype.sigError = function() { ECDSA.prototype.sign = function() { var hashbuf = this.hashbuf; - var privkey = this.keypair.privkey; + var privkey = this.privkey; var k = this.k; var d = privkey.bn; @@ -170,7 +178,7 @@ ECDSA.prototype.sign = function() { var s = k.invm(N).mul(e.add(d.mul(r))).mod(N); } while (r.cmp(0) <= 0 || s.cmp(0) <= 0); - this.sig = new Signature({r: r, s: s, compressed: this.keypair.pubkey.compressed}); + this.sig = new Signature({r: r, s: s, compressed: this.privkey.compressed}); return this.sig; }; @@ -183,6 +191,10 @@ ECDSA.prototype.toString = function() { var obj = {}; if (this.hashbuf) obj.hashbuf = this.hashbuf.toString('hex'); + if (this.pubkey) + obj.pubkey = this.pubkey.toString(); + if (this.privkey) + obj.privkey = this.privkey.toString(); if (this.keypair) obj.keypair = this.keypair.toString(); if (this.sig) @@ -199,10 +211,10 @@ ECDSA.prototype.verify = function() { return false; }; -ECDSA.sign = function(hashbuf, keypair) { +ECDSA.sign = function(hashbuf, privkey) { return ECDSA().set({ hashbuf: hashbuf, - keypair: keypair + privkey: privkey }).signRandomK(); }; @@ -210,7 +222,7 @@ ECDSA.verify = function(hashbuf, sig, pubkey) { return ECDSA().set({ hashbuf: hashbuf, sig: sig, - keypair: Keypair().set({pubkey: pubkey}) + pubkey: pubkey }).verify(); }; diff --git a/package.json b/package.json index 2085ec4..0d0c916 100644 --- a/package.json +++ b/package.json @@ -5,8 +5,8 @@ "author": "BitPay ", "main": "index.js", "scripts": { - "test": "mocha --reporter spec", - "coverage": "istanbul cover _mocha" + "test": "mocha --recursive --reporter spec", + "coverage": "istanbul cover _mocha -- --recursive" }, "contributors": [ { diff --git a/test/crypto/bn.js b/test/crypto/bn.js index 7a72b5d..298c8d4 100644 --- a/test/crypto/bn.js +++ b/test/crypto/bn.js @@ -2,7 +2,7 @@ var should = require('chai').should(); var bitcore = require('../..'); -var BN = bitcore.BN; +var BN = bitcore.crypto.BN; describe('BN', function() { it('should create a bn', function() { diff --git a/test/crypto/ecdsa.js b/test/crypto/ecdsa.js index f9436a3..48d66da 100644 --- a/test/crypto/ecdsa.js +++ b/test/crypto/ecdsa.js @@ -2,14 +2,13 @@ var should = require('chai').should(); var bitcore = require('../..'); -var ECDSA = bitcore.ECDSA; -var Hash = bitcore.Hash; -var Keypair = bitcore.Keypair; +var ECDSA = bitcore.crypto.ECDSA; +var Hash = bitcore.crypto.Hash; var Privkey = bitcore.Privkey; var Pubkey = bitcore.Pubkey; var Signature = bitcore.Signature; -var BN = bitcore.BN; -var point = bitcore.Point; +var BN = bitcore.crypto.BN; +var point = bitcore.crypto.Point; describe('ECDSA', function() { @@ -20,9 +19,8 @@ describe('ECDSA', function() { var ecdsa = new ECDSA(); ecdsa.hashbuf = Hash.sha256(new Buffer('test data')); - ecdsa.keypair = new Keypair(); - ecdsa.keypair.privkey = new Privkey({bn: BN().fromBuffer(new Buffer('fee0a1f7afebf9d2a5a80c0c98a31c709681cce195cbcd06342b517970c0be1e', 'hex'))}); - ecdsa.keypair.pubkey = new Pubkey({ + ecdsa.privkey = new Privkey({bn: BN().fromBuffer(new Buffer('fee0a1f7afebf9d2a5a80c0c98a31c709681cce195cbcd06342b517970c0be1e', 'hex'))}); + ecdsa.pubkey = new Pubkey({ point: point(BN().fromBuffer(new Buffer('ac242d242d23be966085a2b2b893d989f824e06c9ad0395a8a52f055ba39abb2', 'hex')), BN().fromBuffer(new Buffer('4836ab292c105a711ed10fcfd30999c31ff7c02456147747e03e739ad527c380', 'hex'))) }); @@ -49,13 +47,11 @@ describe('ECDSA', function() { var r = BN('71706645040721865894779025947914615666559616020894583599959600180037551395766', 10); var s = BN('109412465507152403114191008482955798903072313614214706891149785278625167723646', 10); var ecdsa = new ECDSA(); - ecdsa.keypair = new Keypair(); - ecdsa.keypair.privkey = Privkey(); - ecdsa.keypair.privkey.bn = BN().fromBuffer(Hash.sha256(new Buffer('test'))); - ecdsa.keypair.privkey2pubkey(); + ecdsa.privkey = Privkey(); + ecdsa.privkey.bn = BN().fromBuffer(Hash.sha256(new Buffer('test'))); + ecdsa.privkey2pubkey(); ecdsa.hashbuf = hashbuf; ecdsa.sig = new Signature({r: r, s: s}); - ecdsa.calci(); ecdsa.sig.i.should.equal(1); }); @@ -69,7 +65,8 @@ describe('ECDSA', function() { var ecdsa2 = new ECDSA(); ecdsa2.fromString(str); should.exist(ecdsa.hashbuf); - should.exist(ecdsa.keypair); + should.exist(ecdsa.pubkey); + should.exist(ecdsa.privkey); }); }); @@ -100,7 +97,7 @@ describe('ECDSA', function() { ecdsa.sign(); ecdsa.sig.i = 1; var pubkey = ecdsa.sig2pubkey(); - pubkey.point.eq(ecdsa.keypair.pubkey.point).should.equal(true); + pubkey.point.eq(ecdsa.pubkey.point).should.equal(true); }); }); @@ -123,8 +120,7 @@ describe('ECDSA', function() { ecdsa.hashbuf = Hash.sha256(new Buffer('test')); var pk = new Pubkey(); pk.fromDER(new Buffer('041ff0fe0f7b15ffaa85ff9f4744d539139c252a49710fb053bb9f2b933173ff9a7baad41d04514751e6851f5304fd243751703bed21b914f6be218c0fa354a341', 'hex')); - ecdsa.keypair = new Keypair(); - ecdsa.keypair.pubkey = pk; + ecdsa.pubkey = pk; ecdsa.sig = new Signature(); ecdsa.sig.r = BN(0); ecdsa.sig.s = BN(0); @@ -151,7 +147,8 @@ describe('ECDSA', function() { it('should should throw an error if hashbuf is not 32 bytes', function() { var ecdsa2 = ECDSA().set({ hashbuf: ecdsa.hashbuf.slice(0, 31), - keypair: ecdsa.keypair + pubkey: ecdsa.pubkey, + privkey: ecdsa.privkey }); ecdsa2.randomK(); (function() { @@ -197,7 +194,7 @@ describe('ECDSA', function() { describe('@sign', function() { it('should produce a signature', function() { - var sig = ECDSA.sign(ecdsa.hashbuf, ecdsa.keypair); + var sig = ECDSA.sign(ecdsa.hashbuf, ecdsa.privkey); (sig instanceof Signature).should.equal(true); }); @@ -206,10 +203,10 @@ describe('ECDSA', function() { describe('@verify', function() { it('should verify a valid signature, and unverify an invalid signature', function() { - var sig = ECDSA.sign(ecdsa.hashbuf, ecdsa.keypair); - ECDSA.verify(ecdsa.hashbuf, sig, ecdsa.keypair.pubkey).should.equal(true); + var sig = ECDSA.sign(ecdsa.hashbuf, ecdsa.privkey); + ECDSA.verify(ecdsa.hashbuf, sig, ecdsa.pubkey).should.equal(true); var fakesig = Signature(sig.r.add(1), sig.s); - ECDSA.verify(ecdsa.hashbuf, fakesig, ecdsa.keypair.pubkey).should.equal(false); + ECDSA.verify(ecdsa.hashbuf, fakesig, ecdsa.pubkey).should.equal(false); }); }); diff --git a/test/crypto/hash.js b/test/crypto/hash.js index 816aa6b..34d0f99 100644 --- a/test/crypto/hash.js +++ b/test/crypto/hash.js @@ -2,7 +2,7 @@ require('chai').should(); var bitcore = require('../..'); -var Hash = bitcore.Hash; +var Hash = bitcore.crypto.Hash; describe('Hash', function() { var buf = new Buffer([0, 1, 2, 3, 253, 254, 255]); diff --git a/test/crypto/point.js b/test/crypto/point.js index b8d2406..43b5ed5 100644 --- a/test/crypto/point.js +++ b/test/crypto/point.js @@ -2,8 +2,8 @@ var should = require('chai').should(); var bitcore = require('../..'); -var point = bitcore.Point; -var BN = bitcore.BN; +var point = bitcore.crypto.Point; +var BN = bitcore.crypto.BN; describe('Point', function() { diff --git a/test/crypto/random.js b/test/crypto/random.js index e92afcc..1dc7eaf 100644 --- a/test/crypto/random.js +++ b/test/crypto/random.js @@ -2,7 +2,7 @@ var should = require('chai').should(); var bitcore = require('../..'); -var Random = bitcore.Random; +var Random = bitcore.crypto.Random; describe('Random', function() { diff --git a/test/encoding/base58.js b/test/encoding/base58.js index a99a2c7..adf7c83 100644 --- a/test/encoding/base58.js +++ b/test/encoding/base58.js @@ -2,7 +2,7 @@ var should = require('chai').should(); var bitcore = require('../..'); -var Base58 = bitcore.Base58; +var Base58 = bitcore.encoding.Base58; describe('Base58', function() { var buf = new Buffer([0, 1, 2, 3, 253, 254, 255]); diff --git a/test/encoding/base58check.js b/test/encoding/base58check.js index 04dd21a..5f15a01 100644 --- a/test/encoding/base58check.js +++ b/test/encoding/base58check.js @@ -2,8 +2,8 @@ var should = require('chai').should(); var bitcore = require('../..'); -var Base58Check = bitcore.Base58Check; -var base58 = bitcore.Base58; +var Base58Check = bitcore.encoding.Base58Check; +var base58 = bitcore.encoding.Base58; describe('Base58Check', function() { var buf = new Buffer([0, 1, 2, 3, 253, 254, 255]); diff --git a/test/encoding/bufferreader.js b/test/encoding/bufferreader.js index f92f59c..8372c37 100644 --- a/test/encoding/bufferreader.js +++ b/test/encoding/bufferreader.js @@ -2,9 +2,9 @@ var should = require('chai').should(); var bitcore = require('../..'); -var BufferWriter = bitcore.BufferWriter; -var BufferReader = bitcore.BufferReader; -var BN = bitcore.BN; +var BufferWriter = bitcore.encoding.BufferWriter; +var BufferReader = bitcore.encoding.BufferReader; +var BN = bitcore.crypto.BN; describe('BufferReader', function() { diff --git a/test/encoding/bufferwriter.js b/test/encoding/bufferwriter.js index e944a43..bc5bcc9 100644 --- a/test/encoding/bufferwriter.js +++ b/test/encoding/bufferwriter.js @@ -2,9 +2,9 @@ var bitcore = require('../..'); var should = require('chai').should(); -var BufferWriter = bitcore.BufferWriter; -var BufferReader = bitcore.BufferReader; -var BN = bitcore.BN; +var BufferWriter = bitcore.encoding.BufferWriter; +var BufferReader = bitcore.encoding.BufferReader; +var BN = bitcore.crypto.BN; describe('BufferWriter', function() { diff --git a/test/encoding/varint.js b/test/encoding/varint.js index dd67a50..9a8bb84 100644 --- a/test/encoding/varint.js +++ b/test/encoding/varint.js @@ -2,10 +2,10 @@ var should = require('chai').should(); var bitcore = require('../..'); -var BN = bitcore.BN; -var BufferReader = bitcore.BufferReader; -var BufferWriter = bitcore.BufferWriter; -var Varint = bitcore.Varint; +var BN = bitcore.crypto.BN; +var BufferReader = bitcore.encoding.BufferReader; +var BufferWriter = bitcore.encoding.BufferWriter; +var Varint = bitcore.encoding.Varint; describe('Varint', function() { From cd4583f9547ca551b635b5530fba1ca21819bcd9 Mon Sep 17 00:00:00 2001 From: Braydon Fuller Date: Tue, 25 Nov 2014 14:40:43 -0500 Subject: [PATCH 4/4] Encodings: Updated Base58 for the latest version. --- lib/encoding/base58.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/encoding/base58.js b/lib/encoding/base58.js index 8e7409e..6878b73 100644 --- a/lib/encoding/base58.js +++ b/lib/encoding/base58.js @@ -30,7 +30,7 @@ Base58.encode = function(buf) { Base58.decode = function(str) { if (typeof str !== 'string') throw new Error('Input should be a string'); - return bs58.decode(str); + return new Buffer(bs58.decode(str)); }; Base58.prototype.fromBuffer = function(buf) {