2015-06-07 15:57:41 +02:00

130 lines
5.3 KiB
JavaScript

// test xgcd
var assert = require('assert'),
error = require('../../../lib/error/index'),
math = require('../../../index').create({matrix: 'array'}),
gcd = math.gcd,
xgcd = math.xgcd;
describe('xgcd', function() {
it('should return extended greatest common divisor of two numbers', function() {
// xgcd(36163, 21199) = 1247 => -7(36163) + 12(21199) = 1247
assert.deepEqual([1247, -7, 12], xgcd(36163, 21199));
// xgcd(120, 23) = 1 => -9(120) + 47(23) = 1
assert.deepEqual([1, -9, 47], xgcd(120, 23));
// some unit tests from: https://github.com/sjkaliski/numbers.js/blob/master/test/basic.test.js
assert.deepEqual([5, -3, 5], xgcd(65, 40));
assert.deepEqual([5, 5, -3], xgcd(40, 65));
assert.deepEqual([21, -16, 27], xgcd(1239, 735));
assert.deepEqual([21, 5, -2], xgcd(105, 252));
assert.deepEqual([21, -2, 5], xgcd(252, 105));
});
it ('should calculate xgcd for edge cases around zero', function () {
assert.deepEqual([3, 1, 0], xgcd(3, 0));
assert.deepEqual([3, -1, 0], xgcd(-3, 0));
assert.deepEqual([3, 0, 1], xgcd(0, 3));
assert.deepEqual([3, 0, -1], xgcd(0, -3));
assert.deepEqual([1, 0, 1], xgcd(1, 1));
assert.deepEqual([1, 1, 0], xgcd(1, 0));
assert.deepEqual([1, 0, -1], xgcd(1, -1));
assert.deepEqual([1, 0, 1], xgcd(-1, 1));
assert.deepEqual([1, -1, 0], xgcd(-1, 0));
assert.deepEqual([1, 0, -1], xgcd(-1, -1));
assert.deepEqual([1, 0, 1], xgcd(0, 1));
assert.deepEqual([1, 0, -1], xgcd(0, -1));
assert.deepEqual([0, 0, 0], xgcd(0, 0));
});
it('should calculate xgcd of booleans', function() {
assert.deepEqual(xgcd(true, true), [1, 0, 1]);
assert.deepEqual(xgcd(true, false), [1, 1, 0]);
assert.deepEqual(xgcd(false, true), [1, 0, 1]);
assert.deepEqual(xgcd(false, false), [0, 0, 0]);
});
it('should calculate xgcd of numbers and null', function () {
assert.deepEqual(xgcd(1, null), [1, 1, 0]);
assert.deepEqual(xgcd(null, 1), [1, 0, 1]);
assert.deepEqual(xgcd(null, null), [0, 0, 0]);
});
it('should calculate xgcd for BigNumbers', function() {
assert.deepEqual(xgcd(math.bignumber(65), math.bignumber(40)), [math.bignumber(5), math.bignumber(-3), math.bignumber(5)]);
assert.deepEqual(xgcd(math.bignumber(65), math.bignumber(40)), [math.bignumber(5), math.bignumber(-3), math.bignumber(5)]);
});
it('should calculate xgcd for mixed BigNumbers and Numbers', function() {
assert.deepEqual(xgcd(math.bignumber(65), 40), [math.bignumber(5), math.bignumber(-3), math.bignumber(5)]);
assert.deepEqual(xgcd(65, math.bignumber(40)), [math.bignumber(5), math.bignumber(-3), math.bignumber(5)]);
});
// FIXME: xgcd for negative values
it.skip('should calculate xgcd for edge cases with negative values', function () {
assert.deepEqual([1, -2, 1], xgcd(2, 5));
assert.deepEqual([1, -2, -1], xgcd(2, -5));
assert.deepEqual([1, 2, 1], xgcd(-2, 5));
assert.deepEqual([1, 2, -1], xgcd(-2, -5));
assert.deepEqual([2, 1, 0], xgcd(2, 6));
assert.deepEqual([2, 1, 0], xgcd(2, -6));
assert.deepEqual([2, -1, 0], xgcd(-2, 6));
assert.deepEqual([2, -1, 0], xgcd(-2, -6));
});
it('should find the greatest common divisor of booleans', function() {
assert.deepEqual([1, 0, 1], xgcd(true, true));
assert.deepEqual([1, 1, 0], xgcd(true, false));
assert.deepEqual([1, 0, 1], xgcd(false, true));
assert.deepEqual([0, 0, 0], xgcd(false, false));
});
it('should give same results as gcd', function() {
assert.equal(gcd(1239, 735), xgcd(1239, 735)[0]);
assert.equal(gcd(105, 252), xgcd(105, 252)[0]);
assert.equal(gcd(7, 13), xgcd(7, 13)[0]);
});
it('should return a matrix when configured to use matrices', function() {
var math1 = math.create({matrix: 'matrix'});
assert.deepEqual(math1.xgcd(65, 40), math.matrix([5, -3, 5]));
var math2 = math.create({matrix: 'array'});
assert.deepEqual(math2.xgcd(65, 40), [5, -3, 5]);
});
it('should throw an error if used with wrong number of arguments', function() {
assert.throws(function () {xgcd(1)});
assert.throws(function () {xgcd(1, 2, 3)});
});
it('should throw an error for non-integer numbers', function() {
assert.throws(function () {xgcd(2, 4.1); }, /Parameters in function xgcd must be integer numbers/);
assert.throws(function () {xgcd(2.3, 4); }, /Parameters in function xgcd must be integer numbers/);
})
it('should throw an error when used with a complex number', function() {
assert.throws(function () {xgcd(math.complex(1,3),2); }, TypeError, 'Function xgcd(complex, number) not supported');
});
it('should convert to a number when used with a string', function() {
assert.deepEqual(xgcd('65', '40'), [5, -3, 5]);
assert.throws(function () {xgcd(2, 'a'); }, /Cannot convert "a" to a number/);
});
it('should throw an error when used with a unit', function() {
assert.throws(function () { xgcd(math.unit('5cm'), 2); }, TypeError, 'Function xgcd(unit, number) not supported');
});
it('should throw an error when used with a matrix', function() {
assert.throws(function () { xgcd([5,2,3], [25,3,6]); }, TypeError, 'Function xgcd(array, array) not supported');
});
it('should LaTeX xgcd', function () {
var expression = math.parse('xgcd(2,3)');
assert.equal(expression.toTex(), '\\mathrm{xgcd}\\left(2,3\\right)');
});
});