mirror of
https://source.quilibrium.com/quilibrium/ceremonyclient.git
synced 2025-01-25 23:25:17 +00:00
221 lines
6.3 KiB
Go
221 lines
6.3 KiB
Go
|
//
|
||
|
// Copyright Coinbase, Inc. All Rights Reserved.
|
||
|
//
|
||
|
// SPDX-License-Identifier: Apache-2.0
|
||
|
//
|
||
|
|
||
|
package main
|
||
|
|
||
|
import (
|
||
|
"flag"
|
||
|
"fmt"
|
||
|
|
||
|
"source.quilibrium.com/quilibrium/monorepo/nekryptology/pkg/core/curves"
|
||
|
dkg "source.quilibrium.com/quilibrium/monorepo/nekryptology/pkg/dkg/frost"
|
||
|
"source.quilibrium.com/quilibrium/monorepo/nekryptology/pkg/sharing"
|
||
|
"source.quilibrium.com/quilibrium/monorepo/nekryptology/pkg/signatures/schnorr/mina"
|
||
|
"source.quilibrium.com/quilibrium/monorepo/nekryptology/pkg/ted25519/frost"
|
||
|
)
|
||
|
|
||
|
const LIMIT = 4
|
||
|
const THRESHOLD = 2
|
||
|
const Ctx = "string to prevent replay attack"
|
||
|
|
||
|
func main() {
|
||
|
var threshold int
|
||
|
var limit int
|
||
|
var help bool
|
||
|
flag.IntVar(&threshold, "t", THRESHOLD, "the minimum number of participants to sign")
|
||
|
flag.IntVar(&threshold, "threshold", THRESHOLD, "the minimum number of participants to sign")
|
||
|
flag.IntVar(&limit, "n", LIMIT, "the total number of participants")
|
||
|
flag.IntVar(&limit, "limit", LIMIT, "the total number of participants")
|
||
|
flag.BoolVar(&help, "h", false, "Print this menu")
|
||
|
flag.BoolVar(&help, "help", false, "Print this menu")
|
||
|
flag.Parse()
|
||
|
|
||
|
if help {
|
||
|
printHelp()
|
||
|
return
|
||
|
}
|
||
|
|
||
|
fmt.Printf("Threshold is %d\n", threshold)
|
||
|
fmt.Printf("Total participants is %d\n", limit)
|
||
|
|
||
|
// DEMO doing FROST DKG and that signers can compute a signature
|
||
|
participants := createDkgParticipants(threshold, limit)
|
||
|
|
||
|
// DKG Round 1
|
||
|
rnd1Bcast, rnd1P2p := round1(participants)
|
||
|
|
||
|
// DKG Round 2
|
||
|
verificationKey, signingShares := round2(participants, rnd1Bcast, rnd1P2p)
|
||
|
|
||
|
// Signing common setup for all participants
|
||
|
curve := curves.PALLAS()
|
||
|
scheme, _ := sharing.NewShamir(uint32(threshold), uint32(limit), curve)
|
||
|
sk, err := scheme.Combine(signingShares[1], signingShares[2])
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
skC := new(mina.SecretKey)
|
||
|
skC.SetFq(sk.(*curves.ScalarPallas).GetFq())
|
||
|
vk := skC.GetPublicKey()
|
||
|
pk := new(mina.PublicKey)
|
||
|
pk.SetPointPallas(verificationKey.(*curves.PointPallas))
|
||
|
if pk.GenerateAddress() != vk.GenerateAddress() {
|
||
|
fmt.Printf("generated key is different than expected")
|
||
|
}
|
||
|
|
||
|
feePayerPk := new(mina.PublicKey)
|
||
|
_ = feePayerPk.ParseAddress("B62qiy32p8kAKnny8ZFwoMhYpBppM1DWVCqAPBYNcXnsAHhnfAAuXgg")
|
||
|
sourcePk := new(mina.PublicKey)
|
||
|
_ = sourcePk.ParseAddress("B62qiy32p8kAKnny8ZFwoMhYpBppM1DWVCqAPBYNcXnsAHhnfAAuXgg")
|
||
|
receiverPk := new(mina.PublicKey)
|
||
|
_ = receiverPk.ParseAddress("B62qrcFstkpqXww1EkSGrqMCwCNho86kuqBd4FrAAUsPxNKdiPzAUsy")
|
||
|
txn := &mina.Transaction{
|
||
|
Fee: 3,
|
||
|
FeeToken: 1,
|
||
|
Nonce: 200,
|
||
|
ValidUntil: 10000,
|
||
|
Memo: "this is a memo",
|
||
|
FeePayerPk: feePayerPk,
|
||
|
SourcePk: sourcePk,
|
||
|
ReceiverPk: receiverPk,
|
||
|
TokenId: 1,
|
||
|
Amount: 42,
|
||
|
Locked: false,
|
||
|
Tag: [3]bool{false, false, false},
|
||
|
NetworkId: mina.MainNet,
|
||
|
}
|
||
|
sig, _ := skC.SignTransaction(txn)
|
||
|
|
||
|
ok := vk.VerifyTransaction(sig, txn)
|
||
|
fmt.Printf("Signature verification - %v\n", ok == nil)
|
||
|
|
||
|
// Test threshold signing
|
||
|
lcs, err := scheme.LagrangeCoeffs([]uint32{signingShares[1].Id, signingShares[2].Id})
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
signers := make(map[uint32]*frost.Signer, 2)
|
||
|
signers[1], err = frost.NewSigner(participants[1], 1, uint32(threshold), lcs, []uint32{1, 2}, &mina.MinaTSchnorrHandler{})
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
signers[2], err = frost.NewSigner(participants[2], 2, uint32(threshold), lcs, []uint32{1, 2}, &mina.MinaTSchnorrHandler{})
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
msg, _ := txn.MarshalBinary()
|
||
|
|
||
|
sigRnd1Bcast := make(map[uint32]*frost.Round1Bcast, 2)
|
||
|
sigRnd1Bcast[1], err = signers[1].SignRound1()
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
sigRnd1Bcast[2], err = signers[2].SignRound1()
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
sigRng2BCast := make(map[uint32]*frost.Round2Bcast, 2)
|
||
|
sigRng2BCast[1], err = signers[1].SignRound2(msg, sigRnd1Bcast)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
sigRng2BCast[2], err = signers[2].SignRound2(msg, sigRnd1Bcast)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
sigRng3BCast, err := signers[1].SignRound3(sigRng2BCast)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
secSig := &mina.Signature{
|
||
|
R: sigRng3BCast.R.(*curves.PointPallas).X(),
|
||
|
S: sigRng3BCast.Z.(*curves.ScalarPallas).GetFq(),
|
||
|
}
|
||
|
|
||
|
ok = pk.VerifyTransaction(secSig, txn)
|
||
|
fmt.Printf("Threshold Signature verification - %v\n", ok == nil)
|
||
|
ok = vk.VerifyTransaction(secSig, txn)
|
||
|
fmt.Printf("Threshold Signature verification - %v\n", ok == nil)
|
||
|
}
|
||
|
|
||
|
func round1(participants map[uint32]*dkg.DkgParticipant) (map[uint32]*dkg.Round1Bcast, map[uint32]dkg.Round1P2PSend) {
|
||
|
// DKG Round 1
|
||
|
rnd1Bcast := make(map[uint32]*dkg.Round1Bcast, len(participants))
|
||
|
rnd1P2p := make(map[uint32]dkg.Round1P2PSend, len(participants))
|
||
|
for id, p := range participants {
|
||
|
bcast, p2psend, err := p.Round1(nil)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
rnd1Bcast[id] = bcast
|
||
|
rnd1P2p[id] = p2psend
|
||
|
}
|
||
|
return rnd1Bcast, rnd1P2p
|
||
|
}
|
||
|
|
||
|
func round2(participants map[uint32]*dkg.DkgParticipant,
|
||
|
rnd1Bcast map[uint32]*dkg.Round1Bcast,
|
||
|
rnd1P2p map[uint32]dkg.Round1P2PSend,
|
||
|
) (curves.Point, map[uint32]*sharing.ShamirShare) {
|
||
|
signingShares := make(map[uint32]*sharing.ShamirShare, len(participants))
|
||
|
var verificationKey curves.Point
|
||
|
for id := range rnd1Bcast {
|
||
|
fmt.Printf("Computing DKG Round 2 for participant %d\n", id)
|
||
|
rnd1P2pForP := make(map[uint32]*sharing.ShamirShare)
|
||
|
for jid := range rnd1P2p {
|
||
|
if jid == id {
|
||
|
continue
|
||
|
}
|
||
|
rnd1P2pForP[jid] = rnd1P2p[jid][id]
|
||
|
}
|
||
|
rnd2Out, err := participants[id].Round2(rnd1Bcast, rnd1P2pForP)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
verificationKey = rnd2Out.VerificationKey
|
||
|
share := &sharing.ShamirShare{
|
||
|
Id: id,
|
||
|
Value: participants[id].SkShare.Bytes(),
|
||
|
}
|
||
|
signingShares[id] = share
|
||
|
}
|
||
|
return verificationKey, signingShares
|
||
|
}
|
||
|
|
||
|
func createDkgParticipants(thresh, limit int) map[uint32]*dkg.DkgParticipant {
|
||
|
curve := curves.PALLAS()
|
||
|
participants := make(map[uint32]*dkg.DkgParticipant, limit)
|
||
|
for i := 1; i <= limit; i++ {
|
||
|
otherIds := make([]uint32, limit-1)
|
||
|
idx := 0
|
||
|
for j := 1; j <= limit; j++ {
|
||
|
if i == j {
|
||
|
continue
|
||
|
}
|
||
|
otherIds[idx] = uint32(j)
|
||
|
idx++
|
||
|
}
|
||
|
p, err := dkg.NewDkgParticipant(uint32(i), uint32(thresh), Ctx, curve, otherIds...)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
participants[uint32(i)] = p
|
||
|
}
|
||
|
return participants
|
||
|
}
|
||
|
|
||
|
func printHelp() {
|
||
|
fmt.Printf(`
|
||
|
mina INPUT
|
||
|
Simulate a DKG using Mina keys
|
||
|
FLAGS:
|
||
|
-h, --help Show this help message and exit
|
||
|
-n, --limit The total number of participants
|
||
|
-t, --treshold The minimum number of participants needed to sign
|
||
|
`)
|
||
|
}
|