RSA
RSA 的加密。
- 随机选取两个素数 $p\ne q$ , $n = p \times q$
欧拉函数$f = \phi(n) = (p-1) \times (q-1)$如果 $p,q$相等,那么,欧拉函数 $f = p (p-1)$ ,而$PQ$相等的话直接开方$N$就能获取$p,q$。所以这里,规定$p,q$不相等
- 选取一数$e$,使得 $e,f$ 互质
找到 一个数 $d \times e = 1 \pmod{f}$
e f 互质,那么 d 一定存在。可以根据扩展欧几里得算法,快速求得 $e$的模逆元$d$;
- 将$(e,n)$作为公钥 ,$(d,n)$作为私钥
RSA 公钥私钥是对称的. 可以互换。实践中,e 会很小,可以选3, 65537($2^{2^4} + 1$ ) ,方便加密 欧拉定理,如果 $a$ $n$ 互素,那么下式子成立。 其中$\phi(n)$表示 小于n的,与n互素的数的个数。 欧拉函数的性质
- 如果 n 为素数 $\phi(n) = n-1$
- $m n$互质,那么 $\phi(m \times n ) = \phi(m ) \times \phi(n )$ 将m的 互质的每个数乘以n 的互质的数每个数,新得到的肯定也是互质的。
- 如果 $n = p^k$ 那么 $\phi(n) = p^k - p^{k-1}$
因为去掉$p$的倍数的数,剩下的是和$n$互质的素,倍数个数为 $p^{k-1}$ \(a ^{\phi(n)} = 1 \pmod{n}\) 当$n$是一个素数p的时候,$\phi(p) = p-1$,上面式子退化成费马小定理。 \(a ^{p-1} = 1 \pmod{p}\)
加密
加密过程 消息为 $m$加密 ($m < n$) \(E = m ^e \pmod{n}\)
解密 \(M = E ^d \pmod{n}\)
原理证明
这里需要分类讨论
1. m 与n 互质的情况
m n 互质,直接欧拉定理即可
\[\begin{aligned} M &= E ^ e \pmod{n} \\ &= m ^{e \times d} \pmod{n} \\ &= m ^{k\times \phi(n) + 1} \pmod{n} \\ & = ((m ^{\phi(n)})^k ) \times m \pmod{n} \\ & = 1^{k} * m \quad \pmod{n} \\ & = m \end{aligned}\]2. m 与n 不互质
那么 m 肯定 $m = k \times p$ (这里 $m < n$) m 和 q肯定互质,
根据费马小定理 $m^{q-1} = 1 \pmod{q}$ (q 是素数)
继而 \(m ^{q-1} = 1 \pmod{q}\)
$ed = k_1(q-1)(p-1) + 1$
这里构造 $k_1(q-1)(p-1)$ ,两边同时 $k1(p-1)$ 次幂
在 $m^{ed - 1} = 1 \pmod{q}$ 直接展开
这样, m 与 n 不互质的情况下, $m^{ed} = m \pmod{n}$ ,RSA算法也能正常工作。
注,这里私钥e和公钥d
例子,取两个素数, 5 和 11 ,计算
n = 5 * 11 = 55$
y(n)= (5-1)(11 -1) = 40
随机取一数,私钥 e = 3
根据 ed = 1 mod n 得到
d = 27
已知消息为 7 ,
加密: 7 ^ 3 mod 55 = 13
13 ^ 27 = ((13 )^ 3 ) ^ 9 = (52 ^ 3) ^ 3 = 28 ^ 3 = 7 mod 55
消息为 22
加密:22 ^ 3 = 33 mod 55
解密:33 ^ 3 ^ 3 ^ 3 = 22 ^ 3 ^ 3 = 33 ^ 3 = 22
一个简单的js实现(原理展示)
isPrime exGcd 从后面文章中代码略做修改,改用bigInt,
const isPrime = require('./miller-rabin.js')
const exGcd = require('./exGcd.js')
function genRsaKeyPair() {
var P = 0, Q = 0, N = 0, E = 65537n, D = 0;
while (1) {
if (P != 0 && Q != 0) {
break;
}
var tmpPrime = Math.round(Math.random() * (1e10));
if (!isPrime(tmpPrime)) {
continue;
}
if (P == 0) {
P = BigInt(tmpPrime);
continue;
}
if (Q == 0 && tmpPrime != P) {
Q = BigInt(tmpPrime);
continue;
}
}
N = P * Q;
// P != Q ,if P == Q YN = P(P-1)
var YN = (P - 1n) * (Q - 1n);
console.log('P,Q,N,Yn:', P, Q, N, YN);
D = exGcd(E, YN);
/// E D not co-prime ,rechoose P Q
if (D == 0) {
return genRsaKeyPair();
}
// public key = E,N private key D,N
return {
public: [E, N],
private: [D, N]
}
}
function encyrptWithRsaKey(msg, key) {
msg = BigInt(msg);
// a * b % n
function multify(a, b, n) {
return BigInt(a) * BigInt(b) % BigInt(n);
}
// a ^ e % n
function pow(a, e, n) {
if (e == 0) {
return 1n;
}
if (e == 1) {
return a % n;
}
var e0 = e % 2n;
var e1 = (e - e0) / 2n;
var m = pow(a, e1, n) % n
var k = pow(a, e0, n)
return multify(k, multify(m, m, n), n)
}
// var z = multify(1,43046721,73783487)
var z = pow(msg, key[0], key[1]);
return z;
}
do {
var kp2 = genRsaKeyPair();
console.log('RSA Key Pair:\n', kp2);
var N = kp2.public[1];
var E = kp2.public[0];
var D = kp2.private[0];
var z = Math.floor(Math.random() * Number(N));
var e2 = encyrptWithRsaKey(z, kp2.public);
var d2 = encyrptWithRsaKey(e2, kp2.private);
var Sign = encyrptWithRsaKey(z, kp2.private);
var Verify = encyrptWithRsaKey(Sign, kp2.public);
console.log(
`
E = ${E}
D = ${D}
N = ${N}
MSG = ${z};
Enc = ${z} ^ ${E} % N= ${e2};
Dec = ${e2} ^ ${D} % N = ${d2};
Sign = ${z} ^ ${D} % N = ${Sign};
Verify = ${Sign} ^ ${E} % N = ${Verify};
Encrypt:${d2 == z ? 'Success ✅' : "Failed ❌"}: MSG ${d2 == z ? '=' : '≠'} Dec
Sign:${Verify == z ? 'Success ✅' : "Failed ❌"}: MSG ${d2 == Verify ? '=' : '≠'} Verify
`
)
if (d2 != z) {
break;
}
} while (0)
P,Q,N,Yn: 1160459n 10499113n 12183790172867n 12183778513296n
RSA Key Pair:
{
public: [ 65537n, 12183790172867n ],
private: [ 4078982626001n, 12183790172867n ]
}
E = 65537
D = 4078982626001
N = 12183790172867
MSG = 6667895871202;
Enc = 6667895871202 ^ 65537 % N= 10904750447946;
Dec = 10904750447946 ^ 4078982626001 % N = 6667895871202;
Sign = 6667895871202 ^ 4078982626001 % N = 6115510004251;
Verify = 6115510004251 ^ 65537 % N = 6667895871202;
Encrypt:Success ✅: MSG = Dec
Sign:Success ✅: MSG = Verify
一些问题
实际实现中,消息$m$ 如果太小的话很容易被暴力破解,比如0 加密一直是0 ,1 加密一直是1。 为了防止这种情况,一般会对消息进行padding 处理,保证消息不会太小,同时,增加随机化处理,使得每次加密同样的报文得到的密文不一样。
To Be Contined $\longrightarrow$
后面有空补充下,素数测试算法(Miller Rabin算法)。 扩展欧几里得算法求模逆元。(这个应该在写ECC时候就完善的)