使用 Go 通过自定义 io.Reader 确定性生成 RSA 私钥

Ben*_*ter 1 rsa go

由于可能最好不回答的原因,我需要生成无限的 RSA 公钥/私钥。请注意,这没有用于任何高度安全的事情,所以请不要告诉我不要这样做,是的,我知道它并不理想。我所说的“无限”是指我需要未知数量的它们,例如数十亿到数万亿,并且在使用之前创建它们是不可能的。

由于这会消耗无限的空间并需要无限的时间来生成,因此我需要在运行时执行此操作。

但是,对于给定的输入,我还需要具有相同的密钥对。这意味着我需要根据输入确定性地重新创建 RSA 密钥。

我正在使用 Go,通常您使用以下命令创建密钥,

k, err := rsa.GenerateKey(rand.Reader, 2048)
Run Code Online (Sandbox Code Playgroud)

当然,问题是rand.Readercrypto/rand,因此没有办法播种它。

我认为可以提供我自己的阅读器实现来实现我的目标。我查看了源代码GenerateKey并注意到它正在寻找素数,因此我实现了自己的阅读器,这样我就可以控制返回的“随机”素数,从而允许我在需要时生成相同的密钥,

type Reader struct {
    data   []byte
    sum    int
    primes []int
}

func NewReader(toRead string) *Reader {
    primes := sieveOfEratosthenes(10_000_000)
    return &Reader{[]byte(toRead), 0, primes}
}

func (r *Reader) Read(p []byte) (n int, err error) {
    r.sum = r.sum + 1

    if r.sum >= 100_000 {
        return r.primes[rand.Intn(len(r.primes))], io.EOF
    }

    return r.primes[rand.Intn(len(r.primes))], nil
}

func sieveOfEratosthenes(N int) (primes []int) {
    b := make([]bool, N)
    for i := 2; i < N; i++ {
        if b[i] == true {
            continue
        }
        primes = append(primes, i)
        for k := i * i; k < N; k += i {
            b[k] = true
        }
    }
    return
}
Run Code Online (Sandbox Code Playgroud)

然后我可以像这样调用生成密钥

k, err := rsa.GenerateKey(NewReader(""), 2048)
Run Code Online (Sandbox Code Playgroud)

它可以编译,但由于零指针而在运行时崩溃。我对 Go 相当满意,但 RSA 的实现超出了我的理解。寻找更好的方法来实现这一目标,或者也许我需要做什么才能让我的读者工作。

请注意,我在这里唯一的硬性要求是能够为给定的输入生成相同的密钥,并使用rsa.GenerateKey或删除兼容的替换。输入实际上可以是任何东西,只要我得到与输出相同的密钥即可。

这是一个 Go 游乐场链接,展示了我目前所在的位置https://go.dev/play/p/jd1nAoPR5aD

mak*_*bek 5

Read方法没有执行预期的操作。它不会p用随机字节填充输入字节片。如果您查看Unix 方法的实现crypto/rand.Read,它将输入字节切片传递给另一个读取器。所以基本上你需要用随机数填充字节片。例如:

func (r *Reader) Read(p []byte) (n int, err error) {
        i := 0
        b := p

        for i < len(b) {
                if len(b) < 4 {
                        b[0] = 7
                        b = b[1:]
                } else {
                        binary.LittleEndian.PutUint32(b, uint32(rand.Intn(len(r.primes))))
                        b = b[4:]
                }
        }

        return len(p), nil
}
Run Code Online (Sandbox Code Playgroud)

这是游乐场的链接


UPD

正如 Erwin 在回答中提到的,有一个函数称为MaybeReadRand,有 50% 的机会从 rand 读取器读取 1 个字节,使该函数具有不确定性。但是您可以通过在 Read 方法中添加 if 语句来解决:如果输入切片的长度为 1,则忽略所有内容并返回。否则,向输入切片提供素数:

func (r *Reader) Read(p []byte) (n int, err error) {
    i := 0
    b := p

    if len(p) == 1 {
        println("maybeReadRand")
        return 1, nil
    }

    for i < len(b) {
        if len(b) < 4 {
            b[0] = 7
            b = b[1:]
        } else {
            binary.LittleEndian.PutUint32(b, uint32(r.primes[r.i]))
            r.i++
            b = b[4:]
        }
    }

    return len(p), nil
}
Run Code Online (Sandbox Code Playgroud)

在此代码片段中,我创建了 2 个键,并且它们都是相等的。