目录git
P256对应的椭圆曲线是:
\[ y^2=x^3-3x+b \]github
假设k是私钥,G是公钥(\(g^k\))
m:表示已知的公共信息,好比当前要出的块号100app
思路也很简单,将Hash(m)(注意是256位hash)做为曲线上的X,而后带入上述椭圆曲线公式,求出相应的Y便可.google
这个也很简答,就是Hash(...)%q便可.spa
\[ h=H_1(m) \\ v=VRF_k(m)=h^k \]
这就是所谓的可验证随机数,那么怎么让他可验证呢?code
随机生成一个r,而后计算
\[ s=H_2(g,h,G,v,g^r,h^r) \\ t=r-sk (mod p) \]orm
而后把(v,s,t)一块儿打包发给验证方,three
上述信息中已知的有:ip
\[ g^r =g^{t+ks} =g^t \cdot g^{ks} =g^t \cdot {g^k}^s =g^t \cdot G^s \]
\[ h^r =h^{t+ks} =h^t \cdot h^{ks} =h^t \cdot {h^k}^s =h^t \cdot v^s \]get
虽然验证人不知道k,也不知道r,可是知道h,g,G,v,s,t因此他能够计算出\(s2=H_2(g,h,G,v,g^t \cdot G^s,h^t \cdot v^s)\)
而后验证s2是否和s相等,若是相等,那就是k持有人按照规则计算出的随机数
谷歌给出的例子是针对P256的,可是不管是比特币仍是以太坊及其衍生链,采用的都是S256曲线. 那么通过简单的改造就能够在S256曲线上使用VRF
将使用的P256直接换成S256
//curve = elliptic.P256() curve=btcec.S256() params = curve.Params()
前面提到H1其实是把任意信息映射到曲线上的点,P256方案采用的曲线是
\(y^2=x^3-3x+b\),而S256曲线是\(y^2=x^3+b\),稍微有一些区别,所以计算\(y^2\)的方法要修改
// Use the curve equation to calculate y² given x. // only applies to curves of the form y² = x³ - 3x + b. func y2(curve *elliptic.CurveParams, x *big.Int) *big.Int { // y² = x³ - 3x + b x3 := new(big.Int).Mul(x, x) x3.Mul(x3, x) //threeX := new(big.Int).Lsh(x, 1) //threeX.Add(threeX, x) // //x3.Sub(x3, threeX) x3.Add(x3, curve.B) x3.Mod(x3, curve.P) return x3 }
P256代码中的ScalarMult和ScalarBaseMult都是使用的params上的方法,这个方法是在go标准库中的.标准库针对的椭圆曲线并非S256,而是\(y^2=x^3+b\),所以不能使用,要替换成curve上的想用方法.
把params.ScalarBaseMult替换成curve.ScalarBaseMult
把params.ScalarMult替换成curve.ScalarMult