我需要一种计算方法:
m = a. b ^(p-1-x) mod p
在javascript中。 我发现这个算法用于计算base ^ exp%mod:
function expmod(base, exp, mod){
if (exp == 0) return 1;
if (exp % 2 == 0){
return Math.pow((this.expmod(base, (exp / 2), mod)), 2) % mod;
}
else {
return (base * (this.expmod(base, (exp - 1), mod))) % mod;
}
}
它的效果很好。但我似乎无法找到一种方法来做到这一点
m = a. b ^(p-1-x) mod p
如果这个问题不完美,我很抱歉。这是我的第一个问题。谢谢。
答案 0 :(得分:2)
我没有密码学的经验,但是,由于没有其他人在回答,我会试一试。
你的问题对我说的方式并没有多大意义,所以我决定在JavaScript中实现一个完整的Elgamal,这样我就可以在上下文中理解你的问题。这就是我想出的:
// Abstract:
var Alphabet = "!\"#$%&'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_`abcdefghijklmnopqrstuvwxyz{|}~ \n𮃩∆";
Alphabet = Alphabet.split("");
var Crypto = function (alpha, gen, C) {
var p, B, encrypt, decrypt, f, g, modInv, modPow, toAlpha, to10;
toAlpha = function (x) {
var y, p, l, n;
if (x === 0) {
return "!!!!";
}
y = [];
n = 4;
n = Math.ceil(n);
while (n--) {
p = Math.pow(alpha.length, n);
l = Math.floor(x / p);
y.push(alpha[l]);
x -= l * p;
}
y = y.join("");
return y;
};
to10 = function (x) {
var y, p, n;
y = 0;
p = 1;
x = x.split("");
n = x.length;
while (n--) {
y += alpha.indexOf(x[n]) * p;
p *= alpha.length;
}
return y;
};
modInv = function (gen, mod) {
var v, d, u, t, c, q;
v = 1;
d = gen;
t = 1;
c = mod % gen;
u = Math.floor(mod / gen);
while (d > 1) {
q = Math.floor(d / c);
d = d % c;
v = v + q * u;
if (d) {
q = Math.floor(c / d);
c = c % d;
u = u + q * v;
}
}
return d ? v : mod - u;
};
modPow = function (base, exp, mod) {
var c, x;
if (exp === 0) {
return 1;
} else if (exp < 0) {
exp = -exp;
base = modInv(base, mod);
}
c = 1;
while (exp > 0) {
if (exp % 2 === 0) {
base = (base * base) % mod;
exp /= 2;
} else {
c = (c * base) % mod;
exp--;
}
}
return c;
};
p = 91744613;
C = parseInt(C, 10);
if (isNaN(C)) {
C = Math.round(Math.sqrt(Math.random() * Math.random()) * (p - 2) + 2);
}
B = modPow(gen, C, p);
decrypt = function (a) {
var d, x, y;
x = a[1];
y = modPow(a[0], -C, p);
d = (x * y) % p;
d = Math.round(d) % p;
return alpha[d - 2];
};
encrypt = function (key, d) {
var k, a;
k = Math.ceil(Math.sqrt(Math.random() * Math.random()) * 1E10);
d = alpha.indexOf(d) + 2;
a = [];
a[0] = modPow(key[1], k, key[0]);
a[1] = (d * modPow(key[2], k, key[0])) % key[0];
return a;
};
f = function (message, key) {
var n, x, y, w;
y = [];
message = message.split("");
n = message.length;
while (n--) {
x = encrypt(key, message[n]);
y.push(toAlpha(x[0]));
y.push(toAlpha(x[1]));
}
y = y.join("");
return y;
};
g = function (message) {
var n, m, d, x;
m = [];
n = message.length / 8;
while (n--) {
x = message[8 * n + 4];
x += message[8 * n + 5];
x += message[8 * n + 6];
x += message[8 * n + 7];
m.unshift(x);
x = message[8 * n];
x += message[8 * n + 1];
x += message[8 * n + 2];
x += message[8 * n + 3];
m.unshift(x);
}
x = [];
d = [];
n = m.length / 2;
while (n--) {
x[0] = m[2 * n];
x[1] = m[2 * n + 1];
x[0] = to10(x[0]);
x[1] = to10(x[1]);
d.push(decrypt(x));
}
message = d.join("");
return message;
};
return {
pubKey: [p, gen, B],
priKey: C,
decrypt: g,
encrypt: f
};
};
// Usage:
var Alice = Crypto(Alphabet, 69);
var Bob = Crypto(Alphabet, 69);
var message = "Hello!";
console.log(message);
// "Hello!"
message = Alice.encrypt(message, Bob.pubKey);
print(message);
// "Pl)7t&rfGueuL@|)H'P,*<K\.hxw+∆d*`?Io)lg~Adz-6xrR" or something like it.
message = Bob.decrypt(message);
console.log(message);
// "Hello!"
因此,基本上,Crypto
会在需要时使用modPow
来处理所有Elgamal算法。我认为modPow
功能是你原来的功能,不是吗?您最初发布的版本使用重复的平方而不是普通的取幂,可能是出于性能目的,但它们都非常快速。
但是,我仍然不清楚为什么你需要一个不同的算法来做“m = a. b ^(p-1-x) mod p
”。在实现我的Elgamal时我从来不需要这样的东西,所以我不确定这对应于什么。我确实需要实现一个计算modular multiplicative inverse的函数,我称之为modInv
。那是你想要的吗?我使用了Extended Euclidean Algorithm的精简版来制作它。
如果有帮助,请随意复制我项目的部分或全部代码。
如果您对此有任何疑问,请问我!
编辑:请注意,此代码不适用于实际的生产级加密。它实际上只是算法概念的证明。然而,通过一些工作,它可以变得更加安全。让我知道。
编辑:要加密和解密文本,请执行以下操作:
创建一个新的Crypto
对象来加密文本,然后保存它:
var Alice=Crypto(Alphabet, 69);
此处,Alice
只是一些变量,Alphabet
是我在代码顶部定义的29个符号字母,而69
是primitive root mod 91744613。
然后,创建另一个Crypto对象来解密文本,然后保存它:
var Bob=Crypto(Alphabet, 69);
尽管Bob
的创建方式与Alice
相同,但它们是不同的对象。 Bob无法解密用于Alice的文本,而且Alice无法解密用于Bob的文本。
现在,使用Alice
的加密方法加密某些文本,并保存结果:
var codedMessage=Alice.encrypt("HELLO, WORLD.", Bob.pubKey);
此处,message
是一个空变量,"HELLO, WORLD."
是要加密的文本(或文本文件的内容)。 Bob.key是Bob的公钥。我们必须在加密中使用Bob的密钥,以便Bob(并且只有Bob)可以解密文本。生成的加密文本将如下所示:"Pl)7t&rfGueuL@|)H'P,*<K\.hxw+∆d*
?Io)lg~Adz-6xrR“`。请注意,即使使用相同的消息,每次都会生成不同的输出。它仍将始终解密为相同的值。
现在,理论上,我们可以通过我们想要的任何非安全通道发送此加密文本,并且没有人能够对其进行解码。但是,当Bob收到加密文本时,他可以对其进行解码。要做到这一点:
var plainMessage=Bob.decrypt(codedMessage);
现在,plainMessage
将包含文本"HELLO, WORLD."
,您可以阅读或执行任何操作。
所以,总之,只有这四行才能做到:
var Alice=Crypto(Alphabet, 69);
var Bob=Crypto(Alphabet, 69);
var codedMessage=Alice.encrypt("HELLO, WORLD.", Bob.pubKey);
var plainMessage=Bob.decrypt(codedMessage);
// Now, plainMessage contains "HELLO, WORLD."
如果您特别想用文本文件执行此操作,那么您可以将内容复制并粘贴到javascript中,或者您可以查看将文本文件的内容加载到javascript中。要开始使用,请参阅this SO和this HG。