Telethon/telethon/_crypto/factorization.py

68 lines
1.6 KiB
Python
Raw Normal View History

2017-11-26 18:57:40 +03:00
"""
This module holds a fast Factorization class.
"""
2016-08-28 20:26:06 +03:00
from random import randint
2017-05-21 14:59:16 +03:00
class Factorization:
2017-11-26 18:57:40 +03:00
"""
Simple module to factorize large numbers really quickly.
"""
@classmethod
def factorize(cls, pq):
2017-11-26 18:57:40 +03:00
"""
Factorizes the given large integer.
Implementation from https://comeoncodeon.wordpress.com/2010/09/18/pollard-rho-brent-integer-factorization/.
2017-11-26 18:57:40 +03:00
:param pq: the prime pair pq.
:return: a tuple containing the two factors p and q.
"""
if pq % 2 == 0:
return 2, pq // 2
y, c, m = randint(1, pq - 1), randint(1, pq - 1), randint(1, pq - 1)
g = r = q = 1
x = ys = 0
while g == 1:
x = y
for i in range(r):
y = (pow(y, 2, pq) + c) % pq
k = 0
while k < r and g == 1:
ys = y
for i in range(min(m, r - k)):
y = (pow(y, 2, pq) + c) % pq
q = q * (abs(x - y)) % pq
g = cls.gcd(q, pq)
k += m
r *= 2
if g == pq:
while True:
ys = (pow(ys, 2, pq) + c) % pq
g = cls.gcd(abs(x - ys), pq)
if g > 1:
2016-08-28 20:26:06 +03:00
break
2018-12-24 16:16:50 +03:00
p, q = g, pq // g
return (p, q) if p < q else (q, p)
2016-08-28 20:26:06 +03:00
@staticmethod
def gcd(a, b):
2017-11-26 18:57:40 +03:00
"""
Calculates the Greatest Common Divisor.
:param a: the first number.
:param b: the second number.
:return: GCD(a, b)
"""
while b:
a, b = b, a % b
return a