BigInteger: throw if modular inverse doesn't exist (#1228)

This commit is contained in:
larabr 2021-02-10 14:23:59 +01:00 committed by GitHub
parent ca092c7cd0
commit 051f5ec8cf
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 15 additions and 6 deletions

View File

@ -28,7 +28,7 @@ export default class BigInteger {
* BigInteger increment in place
*/
iinc() {
this.value.iadd(one.value);
this.value.iadd(new BN(1));
return this;
}
@ -44,7 +44,7 @@ export default class BigInteger {
* BigInteger decrement in place
*/
idec() {
this.value.isub(one.value);
this.value.isub(new BN(1));
return this;
}
@ -151,8 +151,13 @@ export default class BigInteger {
* Note: this and and n must be relatively prime
* @param {BigInteger} n modulo
* @return {BigInteger} x such that this*x = 1 mod n
* @throws {Error} if the inverse does not exist
*/
modInv(n) {
// invm returns a wrong result if the inverse does not exist
if (!this.gcd(n).isOne()) {
throw new Error('Inverse does not exist');
}
return new BigInteger(this.value.invm(n.value));
}
@ -251,7 +256,7 @@ export default class BigInteger {
}
isOne() {
return this.equal(one);
return this.value.eq(new BN(1));
}
isNegative() {
@ -320,5 +325,3 @@ export default class BigInteger {
return this.value.toArrayLike(Uint8Array, endian, length);
}
}
const one = new BigInteger(1);

View File

@ -175,9 +175,14 @@ export default class BigInteger {
* Note: this and and n must be relatively prime
* @param {BigInteger} n modulo
* @return {BigInteger} x such that this*x = 1 mod n
* @throws {Error} if the inverse does not exist
*/
modInv(n) {
return this._egcd(n).x.add(n).mod(n);
const { gcd, x } = this._egcd(n);
if (!gcd.isOne()) {
throw new Error('Inverse does not exist');
}
return x.add(n).mod(n);
}
/**

View File

@ -149,6 +149,7 @@ module.exports = () => describe('BigInteger interface', function() {
const n = new BigInteger(moduloBN.toString());
const expected = baseBN.invm(moduloBN);
expect(a.modInv(n).toString()).to.equal(expected.toString());
expect(() => a.mul(n).modInv(n)).to.throw('Inverse does not exist');
});
it('getBit is correct', async function() {