From 6f4918fd9e9cbe357a54e3757ffaaf67692955d6 Mon Sep 17 00:00:00 2001 From: Andrew Gouin Date: Thu, 18 May 2023 00:15:55 -0600 Subject: [PATCH 1/6] dkg tests passing using frost-ed25519. wip dkg command --- cmd/horcrux/cmd/dkg.go | 110 +++++++++++++++++++++++++++++ cmd/horcrux/cmd/root.go | 1 + go.mod | 1 + go.sum | 3 + signer/keygen/cosigner.go | 76 ++++++++++++++++++++ signer/keygen/dkg.go | 47 ++++++++++++ signer/keygen/keygen_test.go | 97 +++++++++++++++++++++++++ signer/threshold_validator_test.go | 25 +++++++ 8 files changed, 360 insertions(+) create mode 100644 cmd/horcrux/cmd/dkg.go create mode 100644 signer/keygen/cosigner.go create mode 100644 signer/keygen/dkg.go create mode 100644 signer/keygen/keygen_test.go diff --git a/cmd/horcrux/cmd/dkg.go b/cmd/horcrux/cmd/dkg.go new file mode 100644 index 00000000..7741303a --- /dev/null +++ b/cmd/horcrux/cmd/dkg.go @@ -0,0 +1,110 @@ +package cmd + +import ( + "errors" + "fmt" + "os" + "path/filepath" + + "github.com/spf13/cobra" +) + +const flagID = "id" + +// dkgCmd is a cobra command for performing +// a DKG key ceremony as a participating cosigner. +func dkgCmd() *cobra.Command { + cmd := &cobra.Command{ + Use: "dkg", + Args: cobra.NoArgs, + Short: `Command to perform DKG key sharding ceremony (no trusted "dealer")`, + RunE: func(cmd *cobra.Command, args []string) (err error) { + flags := cmd.Flags() + + id, _ := flags.GetUint8(flagID) + chainID, _ := flags.GetString(flagChainID) + threshold, _ := flags.GetUint8(flagThreshold) + shards, _ := flags.GetUint8(flagShards) + + var errs []error + + if id == 0 { + errs = append(errs, fmt.Errorf("id must not be zero")) + } + + if id > shards { + errs = append(errs, fmt.Errorf("id must not be greater than total shards")) + } + + if chainID == "" { + errs = append(errs, fmt.Errorf("chain-id flag must not be empty")) + } + + if threshold == 0 { + errs = append(errs, fmt.Errorf("threshold flag must be > 0, <= --shards, and > --shards/2")) + } + + if shards == 0 { + errs = append(errs, fmt.Errorf("shards flag must be greater than zero")) + } + + if threshold > shards { + errs = append(errs, fmt.Errorf( + "threshold cannot be greater than total shards, got [threshold](%d) > [shards](%d)", + threshold, shards, + )) + } + + if threshold <= shards/2 { + errs = append(errs, fmt.Errorf("threshold must be greater than total shards "+ + "divided by 2, got [threshold](%d) <= [shards](%d) / 2", threshold, shards)) + } + + if len(errs) > 0 { + return errors.Join(errs...) + } + + out, _ := cmd.Flags().GetString(flagOutputDir) + if out != "" { + if err := os.MkdirAll(out, 0700); err != nil { + return err + } + } + + // silence usage after all input has been validated + cmd.SilenceUsage = true + + // cosigner, err := keygen.NewCosigner(id, threshold, shards) + // if err != nil { + // return err + // } + + // TODO network routine for connecting to other cosigners and progressing through the rounds + // maybe libp2p, use cosigners from config on same port? this would ensure that after DKG process, + // networking is established between cosigners. + + filename := filepath.Join(out, fmt.Sprintf("%s_shard.json", chainID)) + + // TODO write shard after DKG + + fmt.Fprintf(cmd.OutOrStdout(), "Created Ed25519 Shard %s\n", filename) + + return nil + }, + } + + addOutputDirFlag(cmd) + addTotalShardsFlag(cmd) + + f := cmd.Flags() + f.Uint8(flagThreshold, 0, "threshold number of shards required to successfully sign") + _ = cmd.MarkFlagRequired(flagThreshold) + f.Uint8(flagID, 0, "cosigner shard ID as participant in DKG ceremony") + _ = cmd.MarkFlagRequired(flagID) + f.String(flagKeyFile, "", "priv_validator_key.json file to shard") + _ = cmd.MarkFlagRequired(flagKeyFile) + f.String(flagChainID, "", "key shards will sign for this chain ID") + _ = cmd.MarkFlagRequired(flagChainID) + + return cmd +} diff --git a/cmd/horcrux/cmd/root.go b/cmd/horcrux/cmd/root.go index eff962d6..e74d5461 100644 --- a/cmd/horcrux/cmd/root.go +++ b/cmd/horcrux/cmd/root.go @@ -25,6 +25,7 @@ func rootCmd() *cobra.Command { cmd.AddCommand(addressCmd()) cmd.AddCommand(createCosignerEd25519ShardsCmd()) cmd.AddCommand(createCosignerRSAShardsCmd()) + cmd.AddCommand(dkgCmd()) cmd.AddCommand(leaderElectionCmd()) cmd.AddCommand(getLeaderCmd()) cmd.AddCommand(stateCmd()) diff --git a/go.mod b/go.mod index 82e5e87c..0eca39d8 100644 --- a/go.mod +++ b/go.mod @@ -21,6 +21,7 @@ require ( github.com/spf13/cobra v1.6.1 github.com/spf13/viper v1.14.0 github.com/stretchr/testify v1.8.2 + github.com/taurusgroup/frost-ed25519 v0.0.0-20210707140332-5abc84a4dba7 github.com/tendermint/go-amino v0.16.0 gitlab.com/unit410/edwards25519 v0.0.0-20220725154547-61980033348e gitlab.com/unit410/threshold-ed25519 v0.0.0-20220725172740-6ee731f539ac diff --git a/go.sum b/go.sum index 2ade6aa5..8eab71eb 100644 --- a/go.sum +++ b/go.sum @@ -47,6 +47,7 @@ cosmossdk.io/math v1.0.0 h1:ro9w7eKx23om2tZz/VM2Pf+z2WAbGX1yDQQOJ6iGeJw= cosmossdk.io/math v1.0.0/go.mod h1:Ygz4wBHrgc7g0N+8+MrnTfS9LLn9aaTGa9hKopuym5k= cosmossdk.io/tools/rosetta v0.2.1 h1:ddOMatOH+pbxWbrGJKRAawdBkPYLfKXutK9IETnjYxw= dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= +filippo.io/edwards25519 v1.0.0-rc.1/go.mod h1:N1IkdkCkiLB6tki+MYJoSx2JTY9NUlxZE7eHn5EwJns= filippo.io/edwards25519 v1.0.0 h1:0wAIcmJUqRdI8IJ/3eGi5/HwXZWPujYXXlkrQogz0Ek= filippo.io/edwards25519 v1.0.0/go.mod h1:N1IkdkCkiLB6tki+MYJoSx2JTY9NUlxZE7eHn5EwJns= github.com/99designs/go-keychain v0.0.0-20191008050251-8e49817e8af4 h1:/vQbFIOMbk2FiG/kXiLl8BRyzTWDw7gX/Hz7Dd5eDMs= @@ -582,6 +583,8 @@ github.com/subosito/gotenv v1.4.1/go.mod h1:ayKnFf/c6rvx/2iiLrJUk1e6plDbT3edrFNG github.com/syndtr/gocapability v0.0.0-20200815063812-42c35b437635/go.mod h1:hkRG7XYTFWNJGYcbNJQlaLq0fg1yr4J4t/NcTQtrfww= github.com/syndtr/goleveldb v1.0.1-0.20210819022825-2ae1ddf74ef7 h1:epCh84lMvA70Z7CTTCmYQn2CKbY8j86K7/FAIr141uY= github.com/syndtr/goleveldb v1.0.1-0.20210819022825-2ae1ddf74ef7/go.mod h1:q4W45IWZaF22tdD+VEXcAWRA037jwmWEB5VWYORlTpc= +github.com/taurusgroup/frost-ed25519 v0.0.0-20210707140332-5abc84a4dba7 h1:hUvSA2LcuXFrzcU/vDnomemN+trNt+KfrA3s28dfXDw= +github.com/taurusgroup/frost-ed25519 v0.0.0-20210707140332-5abc84a4dba7/go.mod h1:HvWEpeV7Nptyy1OAzxcb/DvYOhrFfzMUx0CFrzUgQbg= github.com/tecbot/gorocksdb v0.0.0-20191217155057-f0fad39f321c h1:g+WoO5jjkqGAzHWCjJB1zZfXPIAaDpzXIEJ0eS6B5Ok= github.com/tecbot/gorocksdb v0.0.0-20191217155057-f0fad39f321c/go.mod h1:ahpPrc7HpcfEWDQRZEmnXMzHY03mLDYMCxeDzy46i+8= github.com/tendermint/go-amino v0.16.0 h1:GyhmgQKvqF82e2oZeuMSp9JTN0N09emoSZlb2lyGa2E= diff --git a/signer/keygen/cosigner.go b/signer/keygen/cosigner.go new file mode 100644 index 00000000..bc440093 --- /dev/null +++ b/signer/keygen/cosigner.go @@ -0,0 +1,76 @@ +package keygen + +import ( + "fmt" + + "github.com/taurusgroup/frost-ed25519/pkg/eddsa" + "github.com/taurusgroup/frost-ed25519/pkg/frost" + "github.com/taurusgroup/frost-ed25519/pkg/frost/keygen" + "github.com/taurusgroup/frost-ed25519/pkg/frost/party" + "github.com/taurusgroup/frost-ed25519/pkg/helpers" + "github.com/taurusgroup/frost-ed25519/pkg/state" +) + +// Cosigner corresponds to a horcrux cosigner, but in this case strictly for the purposes of +// DKG key combination (sharding without a trusted "dealer"). +type Cosigner struct { + ID party.ID + allCosigners party.IDSlice + threshold uint16 + + state *state.State // private, do not share with other cosigners + output *keygen.Output // private, do not share with other cosigners +} + +// NewCosigner builds a Cosigner ready to participate in the DKG rounds. +func NewCosigner(id uint8, threshold uint8, total uint8) (Cosigner, error) { + c := Cosigner{ + ID: party.ID(id), + allCosigners: helpers.GenerateSet(party.ID(total)), + threshold: uint16(threshold - 1), + } + + var err error + c.state, c.output, err = frost.NewKeygenState(c.ID, c.allCosigners, party.ID(c.threshold), 0) + return c, err +} + +// Round1 is the first DKG round. It produces messages that must be shared with all of the other Cosigners before the second round. +func (c Cosigner) Round1() ([][]byte, error) { + return helpers.PartyRoutine(nil, c.state) +} + +// Round2 is the second DKG round. It requires all of the Cosigners messages from the first round, and produces messages for the third round. +func (c Cosigner) Round2(round1Msgs [][]byte) ([][]byte, error) { + if len(round1Msgs) != len(c.allCosigners) { + return nil, fmt.Errorf("length of messages (%d) must match number of cosigners (%d) for round 2", len(round1Msgs), len(c.allCosigners)) + } + return helpers.PartyRoutine(round1Msgs, c.state) +} + +// Round3 is the final DKG round. It requires all of the Cosigners messages from the second round. If successful, the shamir secret share is ready for signing. +func (c Cosigner) Round3(round2Msgs [][]byte) error { + total := len(c.allCosigners) + expectedLen := total * (total - 1) + if len(round2Msgs) != expectedLen { + return fmt.Errorf("length of messages (%d) must match N*(N-1) (%d where N = total number of cosigners %d) for round 3", len(round2Msgs), expectedLen, total) + } + _, err := helpers.PartyRoutine(round2Msgs, c.state) + return err +} + +// WaitForCompletion makes sure the protocol is done processing the DKG rounds. +func (c Cosigner) WaitForCompletion() error { + return c.state.WaitForError() +} + +// Public returns the public information such as the combined public key and individual public keys. +func (c Cosigner) Public() *eddsa.Public { + return c.output.Public +} + +// Secret returns the private shard. +// BE VERY CAREFUL WITH THIS METHOD. +func (c Cosigner) Secret() *eddsa.SecretShare { + return c.output.SecretKey +} diff --git a/signer/keygen/dkg.go b/signer/keygen/dkg.go new file mode 100644 index 00000000..9f238668 --- /dev/null +++ b/signer/keygen/dkg.go @@ -0,0 +1,47 @@ +package keygen + +// LocalDKG simulates a DKG key combination ceremony. TEST USE ONLY. +func LocalDKG(threshold, total uint8) (map[uint8]Cosigner, error) { + cosigners := make(map[uint8]Cosigner) + + var err error + for i := uint8(1); i <= total; i++ { + cosigners[i], err = NewCosigner(i, threshold, total) + if err != nil { + return nil, err + } + } + + msgsOut1 := make([][]byte, 0, total) + + for _, c := range cosigners { + msgs1, err := c.Round1() + if err != nil { + return nil, err + } + + msgsOut1 = append(msgsOut1, msgs1...) + } + + msgsOut2 := make([][]byte, 0, total*(total-1)/2) + + for _, c := range cosigners { + msgs2, err := c.Round2(msgsOut1) + if err != nil { + return nil, err + } + + msgsOut2 = append(msgsOut2, msgs2...) + } + + for _, c := range cosigners { + if err := c.Round3(msgsOut2); err != nil { + return nil, err + } + if err := c.WaitForCompletion(); err != nil { + return nil, err + } + } + + return cosigners, nil +} diff --git a/signer/keygen/keygen_test.go b/signer/keygen/keygen_test.go new file mode 100644 index 00000000..4ff92b73 --- /dev/null +++ b/signer/keygen/keygen_test.go @@ -0,0 +1,97 @@ +package keygen_test + +import ( + "errors" + "testing" + + "github.com/strangelove-ventures/horcrux/signer/keygen" + "github.com/stretchr/testify/require" + "github.com/taurusgroup/frost-ed25519/pkg/eddsa" + "github.com/taurusgroup/frost-ed25519/pkg/frost/party" + "github.com/taurusgroup/frost-ed25519/pkg/ristretto" +) + +func TestKeygenCosigner(t *testing.T) { + const ( + threshold = 2 + total = 3 + ) + + cosigners, err := keygen.LocalDKG(threshold, total) + require.NoError(t, err) + + var pubCmp *eddsa.Public + secrets := make(map[party.ID]*eddsa.SecretShare) + + for _, c := range cosigners { + p := c.Public() + secrets[c.ID] = c.Secret() + if pubCmp == nil { + pubCmp = p + continue + } + err := CompareOutput(pubCmp, p) + require.NoError(t, err) + } + + err = ValidateSecrets(secrets, pubCmp) + require.NoError(t, err) +} + +func CompareOutput(public1, public2 *eddsa.Public) error { + if !public1.Equal(public2) { + return errors.New("shares not equal") + } + partyIDs1 := public1.PartyIDs + partyIDs2 := public2.PartyIDs + if len(partyIDs1) != len(partyIDs2) { + return errors.New("partyIDs are not the same length") + } + + for i, id1 := range partyIDs1 { + if id1 != partyIDs2[i] { + return errors.New("partyIDs are not the same") + } + + publicShare1 := public1.Shares[partyIDs1[i]] + publicShare2 := public2.Shares[partyIDs2[i]] + if publicShare1.Equal(publicShare2) != 1 { + return errors.New("different public keys") + } + } + + if !public1.GroupKey.Equal(public2.GroupKey) { + return errors.New("groupKeys not computed the same way") + } + + return nil +} + +func ValidateSecrets(secrets map[party.ID]*eddsa.SecretShare, public *eddsa.Public) error { + fullSecret := ristretto.NewScalar() + + for id, secret := range secrets { + pk1 := &secret.Public + pk2, ok := public.Shares[id] + if !ok { + return errors.New("party %d has no share") + } + + if pk1.Equal(pk2) != 1 { + return errors.New("pk not the same") + } + + lagrange, err := id.Lagrange(public.PartyIDs) + if err != nil { + return err + } + fullSecret.MultiplyAdd(lagrange, &secret.Secret, fullSecret) + } + + fullPk := eddsa.NewPublicKeyFromPoint(new(ristretto.Element).ScalarBaseMult(fullSecret)) + if !public.GroupKey.Equal(fullPk) { + return errors.New("computed groupKey does not match") + } + + return nil +} diff --git a/signer/threshold_validator_test.go b/signer/threshold_validator_test.go index 22f7424b..141d532c 100644 --- a/signer/threshold_validator_test.go +++ b/signer/threshold_validator_test.go @@ -18,6 +18,7 @@ import ( cometrand "github.com/cometbft/cometbft/libs/rand" cometproto "github.com/cometbft/cometbft/proto/tendermint/types" comet "github.com/cometbft/cometbft/types" + "github.com/strangelove-ventures/horcrux/signer/keygen" "github.com/stretchr/testify/require" tsed25519 "gitlab.com/unit410/threshold-ed25519/pkg" ) @@ -34,6 +35,10 @@ func TestThresholdValidator2of3(t *testing.T) { testThresholdValidator(t, 2, 3) } +func TestThresholdValidator2of3DKG(t *testing.T) { + testThresholdValidatorDKG(t, 2, 3) +} + func TestThresholdValidator3of5(t *testing.T) { testThresholdValidator(t, 3, 5) } @@ -70,7 +75,27 @@ func loadKeyForLocalCosigner( return os.WriteFile(cosigner.config.KeyFilePathCosigner(chainID), keyBz, 0600) } +func testThresholdValidatorDKG(t *testing.T, threshold, total uint8) { + cosigners, err := keygen.LocalDKG(threshold, total) + require.NoError(t, err) + + privShards := make([]tsed25519.Scalar, 0, total) + + for _, c := range cosigners { + privShards = append(privShards, c.Secret().Secret.Bytes()) + } + + testThresholdValidatorWithShards(t, threshold, total, privShards) +} + func testThresholdValidator(t *testing.T, threshold, total uint8) { + privateKey := cometcryptoed25519.GenPrivKey() + privKeyBytes := privateKey[:] + privShards := tsed25519.DealShares(tsed25519.ExpandSecret(privKeyBytes[:32]), threshold, total) + testThresholdValidatorWithShards(t, threshold, total, privShards) +} + +func testThresholdValidatorWithShards(t *testing.T, threshold, total uint8, shards []tsed25519.Scalar) { rsaKeys := make([]*rsa.PrivateKey, total) pubKeys := make([]CosignerRSAPubKey, total) cosigners := make([]*LocalCosigner, total) From 514df7d6ca034b5bca9aa7856fc78b10d2b2edd3 Mon Sep 17 00:00:00 2001 From: Andrew Gouin Date: Thu, 18 May 2023 12:56:49 -0600 Subject: [PATCH 2/6] fix lint --- signer/keygen/cosigner.go | 22 ++++++++++++++++----- signer/threshold_validator_test.go | 31 +++++++++++++++++------------- 2 files changed, 35 insertions(+), 18 deletions(-) diff --git a/signer/keygen/cosigner.go b/signer/keygen/cosigner.go index bc440093..674d514e 100644 --- a/signer/keygen/cosigner.go +++ b/signer/keygen/cosigner.go @@ -35,25 +35,37 @@ func NewCosigner(id uint8, threshold uint8, total uint8) (Cosigner, error) { return c, err } -// Round1 is the first DKG round. It produces messages that must be shared with all of the other Cosigners before the second round. +// Round1 is the first DKG round. It produces messages that must be +// shared with all of the other Cosigners before the second round. func (c Cosigner) Round1() ([][]byte, error) { return helpers.PartyRoutine(nil, c.state) } -// Round2 is the second DKG round. It requires all of the Cosigners messages from the first round, and produces messages for the third round. +// Round2 is the second DKG round. It requires all of the Cosigners messages +// from the first round, and produces messages for the third round. func (c Cosigner) Round2(round1Msgs [][]byte) ([][]byte, error) { if len(round1Msgs) != len(c.allCosigners) { - return nil, fmt.Errorf("length of messages (%d) must match number of cosigners (%d) for round 2", len(round1Msgs), len(c.allCosigners)) + return nil, fmt.Errorf( + "length of messages (%d) must match number of cosigners (%d) for round 2", + len(round1Msgs), + len(c.allCosigners), + ) } return helpers.PartyRoutine(round1Msgs, c.state) } -// Round3 is the final DKG round. It requires all of the Cosigners messages from the second round. If successful, the shamir secret share is ready for signing. +// Round3 is the final DKG round. It requires all of the Cosigners messages +// from the second round. If successful, the shamir secret share is ready for signing. func (c Cosigner) Round3(round2Msgs [][]byte) error { total := len(c.allCosigners) expectedLen := total * (total - 1) if len(round2Msgs) != expectedLen { - return fmt.Errorf("length of messages (%d) must match N*(N-1) (%d where N = total number of cosigners %d) for round 3", len(round2Msgs), expectedLen, total) + return fmt.Errorf( + "length of messages (%d) must match N*(N-1) (%d where N = total number of cosigners %d) for round 3", + len(round2Msgs), + expectedLen, + total, + ) } _, err := helpers.PartyRoutine(round2Msgs, c.state) return err diff --git a/signer/threshold_validator_test.go b/signer/threshold_validator_test.go index 141d532c..65729c5f 100644 --- a/signer/threshold_validator_test.go +++ b/signer/threshold_validator_test.go @@ -79,23 +79,32 @@ func testThresholdValidatorDKG(t *testing.T, threshold, total uint8) { cosigners, err := keygen.LocalDKG(threshold, total) require.NoError(t, err) - privShards := make([]tsed25519.Scalar, 0, total) + privShards := make([]tsed25519.Scalar, total) - for _, c := range cosigners { - privShards = append(privShards, c.Secret().Secret.Bytes()) + for i := uint8(1); i <= total; i++ { + c := cosigners[i] + + privShards[c.ID-1] = c.Secret().Secret.Bytes() } - testThresholdValidatorWithShards(t, threshold, total, privShards) + testThresholdValidatorWithShards(t, threshold, total, + cometcryptoed25519.PubKey(cosigners[1].Public().GroupKey.ToEd25519()), privShards) } func testThresholdValidator(t *testing.T, threshold, total uint8) { privateKey := cometcryptoed25519.GenPrivKey() privKeyBytes := privateKey[:] privShards := tsed25519.DealShares(tsed25519.ExpandSecret(privKeyBytes[:32]), threshold, total) - testThresholdValidatorWithShards(t, threshold, total, privShards) + testThresholdValidatorWithShards(t, threshold, total, privateKey.PubKey(), privShards) } -func testThresholdValidatorWithShards(t *testing.T, threshold, total uint8, shards []tsed25519.Scalar) { +func testThresholdValidatorWithShards( + t *testing.T, + threshold uint8, + total uint8, + publicKey cometcrypto.PubKey, + shards []tsed25519.Scalar, +) { rsaKeys := make([]*rsa.PrivateKey, total) pubKeys := make([]CosignerRSAPubKey, total) cosigners := make([]*LocalCosigner, total) @@ -112,10 +121,6 @@ func testThresholdValidatorWithShards(t *testing.T, threshold, total uint8, shar } } - privateKey := cometcryptoed25519.GenPrivKey() - privKeyBytes := privateKey[:] - privShards := tsed25519.DealShares(tsed25519.ExpandSecret(privKeyBytes[:32]), threshold, total) - tmpDir := t.TempDir() for i, pubKey := range pubKeys { @@ -140,10 +145,10 @@ func testThresholdValidatorWithShards(t *testing.T, threshold, total uint8, shar cosigners[i] = cosigner - err = loadKeyForLocalCosigner(cosigner, privateKey.PubKey(), testChainID, privShards[i]) + err = loadKeyForLocalCosigner(cosigner, publicKey, testChainID, shards[i]) require.NoError(t, err) - err = loadKeyForLocalCosigner(cosigner, privateKey.PubKey(), testChainID2, privShards[i]) + err = loadKeyForLocalCosigner(cosigner, publicKey, testChainID2, shards[i]) require.NoError(t, err) } @@ -191,7 +196,7 @@ func testThresholdValidatorWithShards(t *testing.T, threshold, total uint8, shar err = validator.SignProposal(testChainID, &proposal) require.NoError(t, err) - require.True(t, privateKey.PubKey().VerifySignature(signBytes, proposal.Signature)) + require.True(t, publicKey.VerifySignature(signBytes, proposal.Signature)) firstSignature := proposal.Signature From 056a8fdfd20167b7cea6e6b9e2e89c5993a425a0 Mon Sep 17 00:00:00 2001 From: Andrew Gouin Date: Fri, 19 May 2023 00:56:25 -0600 Subject: [PATCH 3/6] get network dkg test working with ed25519 libp2p keys --- .golangci.yml | 2 + cmd/horcrux/cmd/config.go | 4 +- cmd/horcrux/cmd/dkg.go | 124 ++++++++-- cmd/horcrux/cmd/migrate.go | 6 +- cmd/horcrux/cmd/shards.go | 2 +- cmd/horcrux/cmd/threshold.go | 2 +- go.mod | 81 +++++- go.sum | 379 +++++++++++++++++++++++++---- signer/config.go | 76 ++++-- signer/cosigner.go | 12 +- signer/cosigner_key.go | 4 +- signer/cosigner_key_shares.go | 4 +- signer/dkg.go | 315 ++++++++++++++++++++++++ signer/dkg_test.go | 73 ++++++ signer/keygen/cosigner.go | 2 +- signer/keygen/dkg.go | 47 ---- signer/{keygen => }/keygen_test.go | 6 +- signer/local_cosigner.go | 10 +- signer/remote_cosigner.go | 6 +- signer/threshold_signer.go | 6 +- signer/threshold_signer_soft.go | 6 +- signer/threshold_validator.go | 14 +- signer/threshold_validator_test.go | 11 +- 23 files changed, 1003 insertions(+), 189 deletions(-) create mode 100644 signer/dkg.go create mode 100644 signer/dkg_test.go delete mode 100644 signer/keygen/dkg.go rename signer/{keygen => }/keygen_test.go (94%) diff --git a/.golangci.yml b/.golangci.yml index 8065b598..a5e472c6 100644 --- a/.golangci.yml +++ b/.golangci.yml @@ -86,6 +86,8 @@ linters-settings: alias: grpcretry - pkg : github.com/kraken-hpc/go-fork alias: fork + - pkg: github.com/libp2p/go-libp2p-pubsub + alias: pubsub - pkg: github.com/armon/go-metrics/prometheus alias: gmprometheus - pkg: github.com/mitchellh/go-homedir diff --git a/cmd/horcrux/cmd/config.go b/cmd/horcrux/cmd/config.go index 4c93ffb2..6bd3cdc1 100644 --- a/cmd/horcrux/cmd/config.go +++ b/cmd/horcrux/cmd/config.go @@ -71,7 +71,7 @@ for threshold signer mode, --cosigner flags and --threshold flag are required. if signMode == string(signer.SignModeThreshold) { // Threshold Mode Config cosignersFlag, _ := cmdFlags.GetStringSlice(flagCosigner) - threshold, _ := cmdFlags.GetInt(flagThreshold) + threshold, _ := cmdFlags.GetUint8(flagThreshold) raftTimeout, _ := cmdFlags.GetString(flagRaftTimeout) grpcTimeout, _ := cmdFlags.GetString(flagGRPCTimeout) cosigners, err := signer.CosignersFromFlag(cosignersFlag) @@ -141,7 +141,7 @@ for threshold signer mode, --cosigner flags and --threshold flag are required. `cosigners in format tcp://{cosigner-addr}:{p2p-port} (e.g. --cosigner tcp://horcrux-1:2222 --cosigner tcp://horcrux-2:2222 --cosigner tcp://horcrux-3:2222)`) - f.IntP(flagThreshold, "t", 0, "number of shards required for threshold signature") + f.Uint8P(flagThreshold, "t", 0, "number of shards required for threshold signature") f.StringP( flagDebugAddr, "d", "", diff --git a/cmd/horcrux/cmd/dkg.go b/cmd/horcrux/cmd/dkg.go index 7741303a..3f2050dd 100644 --- a/cmd/horcrux/cmd/dkg.go +++ b/cmd/horcrux/cmd/dkg.go @@ -1,30 +1,109 @@ package cmd import ( + "crypto/rand" + "encoding/json" "errors" "fmt" "os" "path/filepath" + "github.com/libp2p/go-libp2p/core/crypto" + "github.com/libp2p/go-libp2p/core/peer" "github.com/spf13/cobra" + "github.com/strangelove-ventures/horcrux/signer" ) const flagID = "id" -// dkgCmd is a cobra command for performing -// a DKG key ceremony as a participating cosigner. func dkgCmd() *cobra.Command { cmd := &cobra.Command{ Use: "dkg", + Short: "Commands for DKG trustless key generation", + } + + cmd.AddCommand(dkgInitCmd()) + cmd.AddCommand(dkgRunCmd()) + + return cmd +} + +func dkgInitCmd() *cobra.Command { + cmd := &cobra.Command{ + Use: "init", Args: cobra.NoArgs, - Short: `Command to perform DKG key sharding ceremony (no trusted "dealer")`, + Short: `Command to initialize peer private key for libp2p communication during DKG`, + RunE: func(cmd *cobra.Command, args []string) (err error) { + flags := cmd.Flags() + + id, _ := flags.GetUint8(flagID) + + // This ed25519 key is used for libp2p communication during DKG. + // i.e. this key is not used for consensus. + privKey, _, err := crypto.GenerateEd25519Key(rand.Reader) + if err != nil { + return err + } + + rawKey, err := privKey.Raw() + if err != nil { + return err + } + + filename := filepath.Join(config.HomeDir, "libp2p.key") + + if err := os.WriteFile(filename, rawKey, 0600); err != nil { + return err + } + + p2pID, err := peer.IDFromPrivateKey(privKey) + if err != nil { + return err + } + + for i, cosigner := range config.Config.ThresholdModeConfig.Cosigners { + if cosigner.ShardID == id { + config.Config.ThresholdModeConfig.Cosigners[i].DKGID = p2pID.String() + } else if cosigner.DKGID == "" { + config.Config.ThresholdModeConfig.Cosigners[i].DKGID = "REPLACE ME" + } + } + + if err := config.WriteConfigFile(); err != nil { + return err + } + + fmt.Fprintf( + cmd.OutOrStdout(), + "libp2p key generated for DKG. `dkgID` to share with other members:\n%s\n", + p2pID.String(), + ) + + return nil + }, + } + + f := cmd.Flags() + f.Uint8(flagID, 0, "cosigner shard ID as participant in DKG ceremony") + _ = cmd.MarkFlagRequired(flagID) + + return cmd +} + +// dkgCmd is a cobra command for performing +// a DKG key ceremony as a participating cosigner. +func dkgRunCmd() *cobra.Command { + cmd := &cobra.Command{ + Use: "run", + Args: cobra.NoArgs, + Short: `Perform DKG key sharding ceremony (no trusted "dealer")`, RunE: func(cmd *cobra.Command, args []string) (err error) { flags := cmd.Flags() id, _ := flags.GetUint8(flagID) chainID, _ := flags.GetString(flagChainID) - threshold, _ := flags.GetUint8(flagThreshold) - shards, _ := flags.GetUint8(flagShards) + threshold := config.Config.ThresholdModeConfig.Threshold + shards := uint8(len(config.Config.ThresholdModeConfig.Cosigners)) var errs []error @@ -71,21 +150,35 @@ func dkgCmd() *cobra.Command { } } + libp2pKeyFile := filepath.Join(config.HomeDir, "libp2p.key") + p2pKeyBz, err := os.ReadFile(libp2pKeyFile) + if err != nil { + return fmt.Errorf("failed to read libp2p.key. Did you run horcrux dkg init first?: %w", err) + } + + p2pKey, err := crypto.UnmarshalEd25519PrivateKey(p2pKeyBz) + if err != nil { + return err + } + // silence usage after all input has been validated cmd.SilenceUsage = true - // cosigner, err := keygen.NewCosigner(id, threshold, shards) - // if err != nil { - // return err - // } + shard, err := signer.NetworkDKG(cmd.Context(), config.Config.ThresholdModeConfig.Cosigners, id, p2pKey, threshold) + if err != nil { + return err + } - // TODO network routine for connecting to other cosigners and progressing through the rounds - // maybe libp2p, use cosigners from config on same port? this would ensure that after DKG process, - // networking is established between cosigners. + shardBz, err := json.Marshal(shard) + if err != nil { + return err + } filename := filepath.Join(out, fmt.Sprintf("%s_shard.json", chainID)) - // TODO write shard after DKG + if err := os.WriteFile(filename, shardBz, 0600); err != nil { + return err + } fmt.Fprintf(cmd.OutOrStdout(), "Created Ed25519 Shard %s\n", filename) @@ -94,15 +187,10 @@ func dkgCmd() *cobra.Command { } addOutputDirFlag(cmd) - addTotalShardsFlag(cmd) f := cmd.Flags() - f.Uint8(flagThreshold, 0, "threshold number of shards required to successfully sign") - _ = cmd.MarkFlagRequired(flagThreshold) f.Uint8(flagID, 0, "cosigner shard ID as participant in DKG ceremony") _ = cmd.MarkFlagRequired(flagID) - f.String(flagKeyFile, "", "priv_validator_key.json file to shard") - _ = cmd.MarkFlagRequired(flagKeyFile) f.String(flagChainID, "", "key shards will sign for this chain ID") _ = cmd.MarkFlagRequired(flagChainID) diff --git a/cmd/horcrux/cmd/migrate.go b/cmd/horcrux/cmd/migrate.go index e9e9de0c..03cf83b1 100644 --- a/cmd/horcrux/cmd/migrate.go +++ b/cmd/horcrux/cmd/migrate.go @@ -48,11 +48,11 @@ type ( } v2CosignerConfig struct { - Threshold int `json:"threshold" yaml:"threshold"` + Threshold uint8 `json:"threshold" yaml:"threshold"` Shares int `json:"shares" yaml:"shares"` P2PListen string `json:"p2p-listen" yaml:"p2p-listen"` Peers []struct { - ShareID int `json:"share-id" yaml:"share-id"` + ShareID uint8 `json:"share-id" yaml:"share-id"` P2PAddr string `json:"p2p-addr" yaml:"p2p-addr"` } `json:"peers" yaml:"peers"` Timeout string `json:"rpc-timeout" yaml:"rpc-timeout"` @@ -66,7 +66,7 @@ type ( PubKey cometcrypto.PubKey `json:"pub_key"` ShareKey []byte `json:"secret_share"` RSAKey rsa.PrivateKey `json:"rsa_key"` - ID int `json:"id"` + ID uint8 `json:"id"` RSAPubs []*rsa.PublicKey `json:"rsa_pubs"` } ) diff --git a/cmd/horcrux/cmd/shards.go b/cmd/horcrux/cmd/shards.go index 9f53701a..ab461556 100644 --- a/cmd/horcrux/cmd/shards.go +++ b/cmd/horcrux/cmd/shards.go @@ -25,7 +25,7 @@ import ( "github.com/strangelove-ventures/horcrux/signer" ) -func createCosignerDirectoryIfNecessary(out string, id int) (string, error) { +func createCosignerDirectoryIfNecessary(out string, id uint8) (string, error) { dir := filepath.Join(out, fmt.Sprintf("cosigner_%d", id)) dirStat, err := os.Stat(dir) if err != nil { diff --git a/cmd/horcrux/cmd/threshold.go b/cmd/horcrux/cmd/threshold.go index de708b15..fa844793 100644 --- a/cmd/horcrux/cmd/threshold.go +++ b/cmd/horcrux/cmd/threshold.go @@ -60,7 +60,7 @@ func NewThresholdValidator( key, pubKeys, p2pListen, - uint8(thresholdCfg.Threshold), + thresholdCfg.Threshold, ) // Validated prior in ValidateThresholdModeConfig diff --git a/go.mod b/go.mod index 0eca39d8..8781850d 100644 --- a/go.mod +++ b/go.mod @@ -8,6 +8,7 @@ require ( github.com/Jille/raftadmin v1.2.0 github.com/armon/go-metrics v0.4.1 github.com/avast/retry-go v3.0.0+incompatible + github.com/avast/retry-go/v4 v4.3.4 github.com/cometbft/cometbft v0.37.0 github.com/cosmos/cosmos-sdk v0.47.1 github.com/gogo/protobuf v1.3.2 @@ -15,6 +16,8 @@ require ( github.com/hashicorp/raft v1.5.0 github.com/hashicorp/raft-boltdb/v2 v2.2.2 github.com/kraken-hpc/go-fork v0.1.1 + github.com/libp2p/go-libp2p v0.27.3 + github.com/libp2p/go-libp2p-pubsub v0.9.3 github.com/mitchellh/go-homedir v1.1.0 github.com/ory/dockertest v3.3.5+incompatible github.com/prometheus/client_golang v1.14.0 @@ -44,6 +47,7 @@ require ( github.com/ChainSafe/go-schnorrkel v0.0.0-20200405005733-88cbf1b4c40d // indirect github.com/Microsoft/go-winio v0.6.0 // indirect github.com/Nvveen/Gotty v0.0.0-20120604004816-cd527374f1e5 // indirect + github.com/benbjohnson/clock v1.3.0 // indirect github.com/beorn7/perks v1.0.1 // indirect github.com/bgentry/speakeasy v0.1.1-0.20220910012023-760eaf8b6816 // indirect github.com/boltdb/bolt v1.3.1 // indirect @@ -53,7 +57,9 @@ require ( github.com/cespare/xxhash/v2 v2.2.0 // indirect github.com/cometbft/cometbft-db v0.7.0 // indirect github.com/confio/ics23/go v0.9.0 // indirect + github.com/containerd/cgroups v1.1.0 // indirect github.com/containerd/continuity v0.3.0 // indirect + github.com/coreos/go-systemd/v22 v22.5.0 // indirect github.com/cosmos/btcutil v1.0.5 // indirect github.com/cosmos/cosmos-proto v1.0.0-beta.2 // indirect github.com/cosmos/go-bip39 v1.0.0 // indirect @@ -62,6 +68,7 @@ require ( github.com/cosmos/ledger-cosmos-go v0.12.2 // indirect github.com/danieljoos/wincred v1.1.2 // indirect github.com/davecgh/go-spew v1.1.1 // indirect + github.com/davidlazar/go-crypto v0.0.0-20200604182044-b73af7476f6c // indirect github.com/decred/dcrd/dcrec/secp256k1/v4 v4.1.0 // indirect github.com/dgraph-io/badger/v2 v2.2007.4 // indirect github.com/dgraph-io/ristretto v0.1.1 // indirect @@ -70,17 +77,26 @@ require ( github.com/docker/go-units v0.5.0 // indirect github.com/dustin/go-humanize v1.0.1 // indirect github.com/dvsekhvalnov/jose2go v1.5.0 // indirect + github.com/elastic/gosigar v0.14.2 // indirect github.com/fatih/color v1.13.0 // indirect + github.com/flynn/noise v1.0.0 // indirect + github.com/francoispqt/gojay v1.2.13 // indirect + github.com/frankban/quicktest v1.14.4 // indirect github.com/fsnotify/fsnotify v1.6.0 // indirect github.com/go-kit/kit v0.12.0 // indirect github.com/go-kit/log v0.2.1 // indirect github.com/go-logfmt/logfmt v0.5.1 // indirect + github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 // indirect github.com/godbus/dbus v0.0.0-20190726142602-4481cbc300e2 // indirect + github.com/godbus/dbus/v5 v5.1.0 // indirect github.com/golang/glog v1.1.0 // indirect + github.com/golang/mock v1.6.0 // indirect github.com/golang/protobuf v1.5.3 // indirect github.com/golang/snappy v0.0.4 // indirect github.com/google/btree v1.1.2 // indirect github.com/google/go-cmp v0.5.9 // indirect + github.com/google/gopacket v1.1.19 // indirect + github.com/google/pprof v0.0.0-20230405160723-4a4c7d95572b // indirect github.com/gorilla/websocket v1.5.0 // indirect github.com/grpc-ecosystem/grpc-gateway v1.16.0 // indirect github.com/gsterjov/go-libsecret v0.0.0-20161001094733-a6f4afe4910c // indirect @@ -93,34 +109,76 @@ require ( github.com/hashicorp/go-multierror v1.1.1 // indirect github.com/hashicorp/go-uuid v1.0.1 // indirect github.com/hashicorp/golang-lru v0.5.5-0.20210104140557-80c98217689d // indirect + github.com/hashicorp/golang-lru/v2 v2.0.2 // indirect github.com/hashicorp/hcl v1.0.0 // indirect github.com/hdevalence/ed25519consensus v0.1.0 // indirect github.com/huandu/skiplist v1.2.0 // indirect + github.com/huin/goupnp v1.1.0 // indirect github.com/inconshreveable/mousetrap v1.0.1 // indirect + github.com/ipfs/go-cid v0.4.1 // indirect + github.com/ipfs/go-log/v2 v2.5.1 // indirect + github.com/jackpal/go-nat-pmp v1.0.2 // indirect + github.com/jbenet/go-temp-err-catcher v0.1.0 // indirect github.com/jmhodges/levigo v1.0.0 // indirect - github.com/klauspost/compress v1.16.3 // indirect + github.com/klauspost/compress v1.16.4 // indirect + github.com/klauspost/cpuid/v2 v2.2.4 // indirect + github.com/koron/go-ssdp v0.0.4 // indirect github.com/libp2p/go-buffer-pool v0.1.0 // indirect + github.com/libp2p/go-cidranger v1.1.0 // indirect + github.com/libp2p/go-flow-metrics v0.1.0 // indirect + github.com/libp2p/go-libp2p-asn-util v0.3.0 // indirect + github.com/libp2p/go-msgio v0.3.0 // indirect + github.com/libp2p/go-nat v0.1.0 // indirect + github.com/libp2p/go-netroute v0.2.1 // indirect + github.com/libp2p/go-reuseport v0.2.0 // indirect + github.com/libp2p/go-yamux/v4 v4.0.0 // indirect github.com/magiconair/properties v1.8.6 // indirect + github.com/marten-seemann/tcp v0.0.0-20210406111302-dfbc87cc63fd // indirect github.com/mattn/go-colorable v0.1.13 // indirect - github.com/mattn/go-isatty v0.0.16 // indirect - github.com/matttproud/golang_protobuf_extensions v1.0.2-0.20181231171920-c182affec369 // indirect + github.com/mattn/go-isatty v0.0.18 // indirect + github.com/matttproud/golang_protobuf_extensions v1.0.4 // indirect + github.com/miekg/dns v1.1.53 // indirect + github.com/mikioh/tcpinfo v0.0.0-20190314235526-30a79bb1804b // indirect + github.com/mikioh/tcpopt v0.0.0-20190314235656-172688c1accc // indirect github.com/mimoo/StrobeGo v0.0.0-20210601165009-122bf33a46e0 // indirect + github.com/minio/sha256-simd v1.0.0 // indirect github.com/mitchellh/mapstructure v1.5.0 // indirect + github.com/mr-tron/base58 v1.2.0 // indirect github.com/mtibben/percent v0.2.1 // indirect + github.com/multiformats/go-base32 v0.1.0 // indirect + github.com/multiformats/go-base36 v0.2.0 // indirect + github.com/multiformats/go-multiaddr v0.9.0 // indirect + github.com/multiformats/go-multiaddr-dns v0.3.1 // indirect + github.com/multiformats/go-multiaddr-fmt v0.1.0 // indirect + github.com/multiformats/go-multibase v0.2.0 // indirect + github.com/multiformats/go-multicodec v0.8.1 // indirect + github.com/multiformats/go-multihash v0.2.1 // indirect + github.com/multiformats/go-multistream v0.4.1 // indirect + github.com/multiformats/go-varint v0.0.7 // indirect + github.com/onsi/ginkgo/v2 v2.9.2 // indirect github.com/opencontainers/go-digest v1.0.0 // indirect github.com/opencontainers/image-spec v1.1.0-rc2 // indirect github.com/opencontainers/runc v1.1.3 // indirect + github.com/opencontainers/runtime-spec v1.0.3-0.20210326190908-1c3f411f0417 // indirect + github.com/pbnjay/memory v0.0.0-20210728143218-7b4eea64cf58 // indirect github.com/pelletier/go-toml v1.9.5 // indirect github.com/pelletier/go-toml/v2 v2.0.5 // indirect github.com/petermattis/goid v0.0.0-20180202154549-b0b1615b78e5 // indirect github.com/pkg/errors v0.9.1 // indirect github.com/pmezard/go-difflib v1.0.0 // indirect github.com/prometheus/client_model v0.3.0 // indirect - github.com/prometheus/common v0.37.0 // indirect - github.com/prometheus/procfs v0.8.0 // indirect + github.com/prometheus/common v0.42.0 // indirect + github.com/prometheus/procfs v0.9.0 // indirect + github.com/quic-go/qpack v0.4.0 // indirect + github.com/quic-go/qtls-go1-19 v0.3.2 // indirect + github.com/quic-go/qtls-go1-20 v0.2.2 // indirect + github.com/quic-go/quic-go v0.33.0 // indirect + github.com/quic-go/webtransport-go v0.5.2 // indirect + github.com/raulk/go-watchdog v1.3.0 // indirect github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475 // indirect github.com/sasha-s/go-deadlock v0.3.1 // indirect github.com/sirupsen/logrus v1.9.0 // indirect + github.com/spaolacci/murmur3 v1.1.0 // indirect github.com/spf13/afero v1.9.2 // indirect github.com/spf13/cast v1.5.0 // indirect github.com/spf13/jwalterweatherman v1.1.0 // indirect @@ -132,17 +190,24 @@ require ( github.com/zondax/hid v0.9.1 // indirect github.com/zondax/ledger-go v0.14.1 // indirect go.etcd.io/bbolt v1.3.7 // indirect + go.uber.org/atomic v1.10.0 // indirect + go.uber.org/dig v1.16.1 // indirect + go.uber.org/fx v1.19.2 // indirect + go.uber.org/multierr v1.11.0 // indirect + go.uber.org/zap v1.24.0 // indirect golang.org/x/crypto v0.7.0 // indirect golang.org/x/exp v0.0.0-20230321023759-10a507213a29 // indirect - golang.org/x/mod v0.8.0 // indirect + golang.org/x/mod v0.10.0 // indirect golang.org/x/net v0.8.0 // indirect - golang.org/x/sys v0.6.0 // indirect + golang.org/x/sys v0.7.0 // indirect golang.org/x/term v0.6.0 // indirect golang.org/x/text v0.8.0 // indirect - golang.org/x/tools v0.6.0 // indirect + golang.org/x/tools v0.7.0 // indirect google.golang.org/genproto v0.0.0-20230306155012-7f2fa6fef1f4 // indirect gopkg.in/ini.v1 v1.67.0 // indirect gopkg.in/yaml.v3 v3.0.1 // indirect + lukechampine.com/blake3 v1.1.7 // indirect + nhooyr.io/websocket v1.8.7 // indirect pgregory.net/rapid v0.5.5 // indirect sigs.k8s.io/yaml v1.3.0 // indirect ) diff --git a/go.sum b/go.sum index 8eab71eb..6cd3173e 100644 --- a/go.sum +++ b/go.sum @@ -1,5 +1,7 @@ cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= +cloud.google.com/go v0.31.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= +cloud.google.com/go v0.37.0/go.mod h1:TS1dMSSfndXH133OKGwekG838Om/cQT0BUHV3HcBgoo= cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU= cloud.google.com/go v0.44.1/go.mod h1:iSa0KzasP4Uvy3f1mN/7PiObzGgflwredwwASm/v6AU= cloud.google.com/go v0.44.2/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY= @@ -46,10 +48,15 @@ cosmossdk.io/errors v1.0.0-beta.7/go.mod h1:mz6FQMJRku4bY7aqS/Gwfcmr/ue91roMEKAm cosmossdk.io/math v1.0.0 h1:ro9w7eKx23om2tZz/VM2Pf+z2WAbGX1yDQQOJ6iGeJw= cosmossdk.io/math v1.0.0/go.mod h1:Ygz4wBHrgc7g0N+8+MrnTfS9LLn9aaTGa9hKopuym5k= cosmossdk.io/tools/rosetta v0.2.1 h1:ddOMatOH+pbxWbrGJKRAawdBkPYLfKXutK9IETnjYxw= +dmitri.shuralyov.com/app/changes v0.0.0-20180602232624-0a106ad413e3/go.mod h1:Yl+fi1br7+Rr3LqpNJf1/uxUdtRUV+Tnj0o93V2B9MU= dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= +dmitri.shuralyov.com/html/belt v0.0.0-20180602232347-f7d459c86be0/go.mod h1:JLBrvjyP0v+ecvNYvCpyZgu5/xkfAUhi6wJj28eUfSU= +dmitri.shuralyov.com/service/change v0.0.0-20181023043359-a85b471d5412/go.mod h1:a1inKt/atXimZ4Mv927x+r7UpyzRUf4emIoiiSC2TN4= +dmitri.shuralyov.com/state v0.0.0-20180228185332-28bcc343414c/go.mod h1:0PRwlb0D6DFvNNtx+9ybjezNCa8XF0xaYcETyp6rHWU= filippo.io/edwards25519 v1.0.0-rc.1/go.mod h1:N1IkdkCkiLB6tki+MYJoSx2JTY9NUlxZE7eHn5EwJns= filippo.io/edwards25519 v1.0.0 h1:0wAIcmJUqRdI8IJ/3eGi5/HwXZWPujYXXlkrQogz0Ek= filippo.io/edwards25519 v1.0.0/go.mod h1:N1IkdkCkiLB6tki+MYJoSx2JTY9NUlxZE7eHn5EwJns= +git.apache.org/thrift.git v0.0.0-20180902110319-2566ecd5d999/go.mod h1:fPE2ZNJGynbRyZ4dJvy6G277gSllfV2HJqblrnkyeyg= github.com/99designs/go-keychain v0.0.0-20191008050251-8e49817e8af4 h1:/vQbFIOMbk2FiG/kXiLl8BRyzTWDw7gX/Hz7Dd5eDMs= github.com/99designs/go-keychain v0.0.0-20191008050251-8e49817e8af4/go.mod h1:hN7oaIRCjzsZ2dE+yG5k+rsdt3qcwykqK6HVGcKwsw4= github.com/99designs/keyring v1.2.1 h1:tYLp1ULvO7i3fI5vE21ReQuj99QFSs7lGm0xWyJo87o= @@ -80,7 +87,7 @@ github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuy github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= github.com/alecthomas/units v0.0.0-20190717042225-c3de453c63f4/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= -github.com/alecthomas/units v0.0.0-20190924025748-f65c72e2690d/go.mod h1:rBZYJk541a8SKzHPHnH3zbiI+7dagKZ0cgpgrD7Fyho= +github.com/anmitsu/go-shlex v0.0.0-20161002113705-648efa622239/go.mod h1:2FmKhYUyUczH0OGQWaF5ceTx0UBShxjsH6f8oGKYe2c= github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY= github.com/armon/consul-api v0.0.0-20180202201655-eb2c6b5be1b6/go.mod h1:grANhF5doyWs3UAsr3K4I6qtAmlQcZDesFNEHPZAzj8= github.com/armon/go-metrics v0.0.0-20190430140413-ec5e00d3c878/go.mod h1:3AMJUQhVx52RsWOnlkpikZr01T/yAVN2gn0861vByNg= @@ -89,6 +96,11 @@ github.com/armon/go-metrics v0.4.1 h1:hR91U9KYmb6bLBYLQjyM+3j+rcd/UhE+G78SFnF8gJ github.com/armon/go-metrics v0.4.1/go.mod h1:E6amYzXo6aW1tqzoZGT755KkbgrJsSdpwZ+3JqfkOG4= github.com/avast/retry-go v3.0.0+incompatible h1:4SOWQ7Qs+oroOTQOYnAHqelpCO0biHSxpiH9JdtuBj0= github.com/avast/retry-go v3.0.0+incompatible/go.mod h1:XtSnn+n/sHqQIpZ10K1qAevBhOOCWBLXXy3hyiqqBrY= +github.com/avast/retry-go/v4 v4.3.4 h1:pHLkL7jvCvP317I8Ge+Km2Yhntv3SdkJm7uekkqbKhM= +github.com/avast/retry-go/v4 v4.3.4/go.mod h1:rv+Nla6Vk3/ilU0H51VHddWHiwimzX66yZ0JT6T+UvE= +github.com/benbjohnson/clock v1.1.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA= +github.com/benbjohnson/clock v1.3.0 h1:ip6w0uFQkncKQ979AypyG0ER7mqUSBdKLOgAle/AT8A= +github.com/benbjohnson/clock v1.3.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA= github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= @@ -97,10 +109,12 @@ github.com/bgentry/speakeasy v0.1.1-0.20220910012023-760eaf8b6816 h1:41iFGWnSlI2 github.com/bgentry/speakeasy v0.1.1-0.20220910012023-760eaf8b6816/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs= github.com/boltdb/bolt v1.3.1 h1:JQmyP4ZBrce+ZQu0dY660FMfatumYDLun9hBCUVIkF4= github.com/boltdb/bolt v1.3.1/go.mod h1:clJnj/oiGkjum5o1McbSZDSLxVThjynRyGBgiAx27Ps= +github.com/bradfitz/go-smtpd v0.0.0-20170404230938-deb6d6237625/go.mod h1:HYsPBTaaSFSlLx/70C2HPIMNZpVV8+vt/A+FMnYP11g= github.com/btcsuite/btcd/btcec/v2 v2.3.2 h1:5n0X6hX0Zk+6omWcihdYvdAlGf2DfasC0GMf7DClJ3U= github.com/btcsuite/btcd/btcec/v2 v2.3.2/go.mod h1:zYzJ8etWJQIv1Ogk7OzpWjowwOdXY1W/17j2MW85J04= github.com/btcsuite/btcd/btcutil v1.1.2 h1:XLMbX8JQEiwMcYft2EGi8zPUkoa0abKIU6/BJSRsjzQ= github.com/btcsuite/btcd/chaincfg/chainhash v1.0.1 h1:q0rUy8C/TYNBQS1+CGKw68tLOFYSNEs0TFnxxnS9+4U= +github.com/buger/jsonparser v0.0.0-20181115193947-bf1c66bbce23/go.mod h1:bbYlZJ7hK1yFx9hf58LP0zeX7UjIGs20ufpu3evjr+s= github.com/cenkalti/backoff v2.2.1+incompatible h1:tNowT99t7UNflLxfYYSlKYsBpXdEet03Pg2g16Swow4= github.com/cenkalti/backoff v2.2.1+incompatible/go.mod h1:90ReRw6GdpyfrHakVjL/QHaoyV4aDUVVkXQJJJ3NXXM= github.com/cenkalti/backoff/v4 v4.1.3 h1:cFAlzYUlVYDysBEH2T5hyJZMh3+5+WCBvSnK6Q8UtC4= @@ -108,7 +122,6 @@ github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA github.com/cespare/xxhash v1.1.0 h1:a6HrQnmkObjyL+Gs60czilIUGqrzKutQD6XZog3p+ko= github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc= github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= -github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= github.com/cespare/xxhash/v2 v2.2.0 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44= github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= github.com/checkpoint-restore/go-criu/v5 v5.3.0/go.mod h1:E/eQpaFtUKGOOSEBZgmKAcn+zUUwWxqcaKZlF54wK8E= @@ -116,6 +129,7 @@ github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWR github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI= github.com/chzyer/readline v1.5.1 h1:upd/6fQk4src78LMRzh5vItIt361/o4uq553V8B5sGI= github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU= +github.com/cilium/ebpf v0.2.0/go.mod h1:To2CFviqOWL/M0gIMsvSMlqe7em/l1ALkX1PyjrX2Qs= github.com/cilium/ebpf v0.7.0/go.mod h1:/oI2+1shJiTGAMgl6/RgJr36Eo1jzrRcAWbcXO2usCA= github.com/circonus-labs/circonus-gometrics v2.3.1+incompatible/go.mod h1:nmEj6Dob7S7YxXgwXpfOuvO54S+tGdZdw9fuRZt25Ag= github.com/circonus-labs/circonusllhist v0.1.3/go.mod h1:kMXHVDlOchFAehlya5ePtbp5jckzBHf4XRpQvBOLI+I= @@ -132,13 +146,20 @@ github.com/cometbft/cometbft-db v0.7.0 h1:uBjbrBx4QzU0zOEnU8KxoDl18dMNgDh+zZRUE0 github.com/cometbft/cometbft-db v0.7.0/go.mod h1:yiKJIm2WKrt6x8Cyxtq9YTEcIMPcEe4XPxhgX59Fzf0= github.com/confio/ics23/go v0.9.0 h1:cWs+wdbS2KRPZezoaaj+qBleXgUk5WOQFMP3CQFGTr4= github.com/confio/ics23/go v0.9.0/go.mod h1:4LPZ2NYqnYIVRklaozjNR1FScgDJ2s5Xrp+e/mYVRak= +github.com/containerd/cgroups v0.0.0-20201119153540-4cbc285b3327/go.mod h1:ZJeTFisyysqgcCdecO57Dj79RfL0LNeGiFUqLYQRYLE= +github.com/containerd/cgroups v1.1.0 h1:v8rEWFl6EoqHB+swVNjVoCJE8o3jX7e8nqBGPLaDFBM= +github.com/containerd/cgroups v1.1.0/go.mod h1:6ppBcbh/NOOUU+dMKrykgaBnK9lCIBxHqJDGwsa1mIw= github.com/containerd/console v1.0.3/go.mod h1:7LqA/THxQ86k76b8c/EMSiaJ3h1eZkMkXar0TQ1gf3U= github.com/containerd/continuity v0.3.0 h1:nisirsYROK15TAMVukJOUyGJjz4BNQJBVsNvAXZJ/eg= github.com/containerd/continuity v0.3.0/go.mod h1:wJEAIwKOm/pBZuBd0JmeTvnLquTB1Ag8espWhkykbPM= github.com/coreos/etcd v3.3.10+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE= github.com/coreos/go-etcd v2.0.0+incompatible/go.mod h1:Jez6KQU2B/sWsbdaef3ED8NzMklzPG4d5KIOhIy30Tk= github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= +github.com/coreos/go-systemd v0.0.0-20181012123002-c6f51f82210d/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= +github.com/coreos/go-systemd/v22 v22.1.0/go.mod h1:xO0FLkIi5MaZafQlIrOotqXZ90ih+1atmu1JpKERPPk= github.com/coreos/go-systemd/v22 v22.3.2/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc= +github.com/coreos/go-systemd/v22 v22.5.0 h1:RrqgGjYQKalulkV8NGVIfkXQf6YYmOyiJKk8iXXhfZs= +github.com/coreos/go-systemd/v22 v22.5.0/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc= github.com/cosmos/btcutil v1.0.5 h1:t+ZFcX77LpKtDBhjucvnOH8C2l2ioGsBNEQ3jef8xFk= github.com/cosmos/btcutil v1.0.5/go.mod h1:IyB7iuqZMJlthe2tkIFL33xPyzbFYP0XVdS8P5lUPis= github.com/cosmos/cosmos-proto v1.0.0-beta.2 h1:X3OKvWgK9Gsejo0F1qs5l8Qn6xJV/AzgIWR2wZ8Nua8= @@ -158,8 +179,10 @@ github.com/cosmos/ledger-cosmos-go v0.12.2/go.mod h1:ZcqYgnfNJ6lAXe4HPtWgarNEY+B github.com/cosmos/rosetta-sdk-go v0.10.0 h1:E5RhTruuoA7KTIXUcMicL76cffyeoyvNybzUGSKFTcM= github.com/cpuguy83/go-md2man v1.0.10/go.mod h1:SmD6nW6nTyfqj6ABTjUi3V3JVMnlJmwcJI5acqYI6dE= github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= +github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= github.com/cpuguy83/go-md2man/v2 v2.0.2/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o= github.com/creachadair/taskgroup v0.3.2 h1:zlfutDS+5XG40AOxcHDSThxKzns8Tnr9jnr6VqkYlkM= +github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= github.com/cucumber/common/gherkin/go/v22 v22.0.0 h1:4K8NqptbvdOrjL9DEea6HFjSpbdT9+Q5kgLpmmsHYl0= github.com/cucumber/common/messages/go/v17 v17.1.1 h1:RNqopvIFyLWnKv0LfATh34SWBhXeoFTJnSrgm9cT/Ts= github.com/cyphar/filepath-securejoin v0.2.3/go.mod h1:aPGpWjXOXUn2NCNjFvBE6aRxGGx79pTxQpKOJNYHHl4= @@ -168,6 +191,8 @@ github.com/danieljoos/wincred v1.1.2/go.mod h1:GijpziifJoIBfYh+S7BbkdUTU4LfM+QnG github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davidlazar/go-crypto v0.0.0-20200604182044-b73af7476f6c h1:pFUpOrbxDR6AkioZ1ySsx5yxlDQZ8stG2b88gTPxgJU= +github.com/davidlazar/go-crypto v0.0.0-20200604182044-b73af7476f6c/go.mod h1:6UhI8N9EjYm1c2odKpFpAYeR8dsBeM7PtzQhRgxRr9U= github.com/decred/dcrd/crypto/blake256 v1.0.0 h1:/8DMNYp9SGi5f0w7uCm6d6M4OU2rGFK09Y2A4Xv7EE0= github.com/decred/dcrd/dcrec/secp256k1/v4 v4.1.0 h1:HbphB4TFFXpv7MNrT52FGrrgVXF1owhMVTHFZIlnvd4= github.com/decred/dcrd/dcrec/secp256k1/v4 v4.1.0/go.mod h1:DZGJHZMqrU4JJqFAWUS2UO1+lbSKsdiOoYi9Zzey7Fc= @@ -190,6 +215,9 @@ github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkp github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto= github.com/dvsekhvalnov/jose2go v1.5.0 h1:3j8ya4Z4kMCwT5nXIKFSV84YS+HdqSSO0VsTQxaLAeM= github.com/dvsekhvalnov/jose2go v1.5.0/go.mod h1:QsHjhyTlD/lAVqn/NSbVZmSCGeDehTB/mPZadG+mhXU= +github.com/elastic/gosigar v0.12.0/go.mod h1:iXRIGg2tLnu7LBdpqzyQfGDEidKCfWcCMS0WKyPWoMs= +github.com/elastic/gosigar v0.14.2 h1:Dg80n8cr90OZ7x+bAax/QjoW/XqTI11RmA79ZwIm9/4= +github.com/elastic/gosigar v0.14.2/go.mod h1:iXRIGg2tLnu7LBdpqzyQfGDEidKCfWcCMS0WKyPWoMs= github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= @@ -205,14 +233,26 @@ github.com/fatih/color v1.12.0/go.mod h1:ELkj/draVOlAH/xkhN6mQ50Qd0MPOk5AAr3maGE github.com/fatih/color v1.13.0 h1:8LOYc1KYPPmyKMuN8QV2DNRWNbLo6LZ0iLs8+mlH53w= github.com/fatih/color v1.13.0/go.mod h1:kLAiJbzzSOZDVNGyDpeOxJ47H46qBXwg5ILebYFFOfk= github.com/felixge/httpsnoop v1.0.2 h1:+nS9g82KMXccJ/wp0zyRW9ZBHFETmMGtkk+2CTTrW4o= +github.com/flynn/go-shlex v0.0.0-20150515145356-3f9db97f8568/go.mod h1:xEzjJPgXI435gkrCt3MPfRiAkVrwSbHsst4LCFVfpJc= +github.com/flynn/noise v1.0.0 h1:DlTHqmzmvcEiKj+4RYo/imoswx/4r6iBlCMfVtrMXpQ= +github.com/flynn/noise v1.0.0/go.mod h1:xbMo+0i6+IGbYdJhF31t2eR1BIU0CYc12+BNAKwUTag= github.com/fortytw2/leaktest v1.3.0 h1:u8491cBMTQ8ft8aeV+adlcytMZylmA5nnwwkRZjI8vw= +github.com/francoispqt/gojay v1.2.13 h1:d2m3sFjloqoIUQU3TsHBgj6qg/BVGlTBeHDUmyJnXKk= +github.com/francoispqt/gojay v1.2.13/go.mod h1:ehT5mTG4ua4581f1++1WLG0vPdaA9HaiDsoyrBGkyDY= github.com/frankban/quicktest v1.11.3/go.mod h1:wRf/ReqHper53s+kmmSZizM8NamnL3IM0I9ntUbOk+k= -github.com/frankban/quicktest v1.14.3 h1:FJKSZTDHjyhriyC81FLQ0LY93eSai0ZyR/ZIkd3ZUKE= +github.com/frankban/quicktest v1.14.4 h1:g2rn0vABPOOXmZUj+vbmUp0lPoXEMuhTpIluN0XL9UY= +github.com/frankban/quicktest v1.14.4/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0= github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= github.com/fsnotify/fsnotify v1.6.0 h1:n+5WquG0fcWoWp6xPWfHdbskMCQaFnG6PfBrh1Ky4HY= github.com/fsnotify/fsnotify v1.6.0/go.mod h1:sl3t1tCWJFWoRz9R8WJCbQihKKwmorjAbSClcnxKAGw= github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= +github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE= +github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI= +github.com/gin-gonic/gin v1.6.3 h1:ahKqKTFpO5KTPHxWZjEdPScmYaGtLo8Y4DMHoEsnp14= +github.com/gin-gonic/gin v1.6.3/go.mod h1:75u5sXoLsGZoRN5Sgbi1eraJ4GU3++wFwWzhwvtwp4M= +github.com/gliderlabs/ssh v0.1.1/go.mod h1:U7qILu1NlMHj9FlMhZLlkCdDnU1DBEAqr0aevW3Awn0= +github.com/go-errors/errors v1.0.1/go.mod h1:f4zRHt4oKfwPJE5k8C9vpYG+aDHdBFUsgrm6/TyX73Q= github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= @@ -220,23 +260,40 @@ github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2 github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= github.com/go-kit/kit v0.12.0 h1:e4o3o3IsBfAKQh5Qbbiqyfu97Ku7jrO/JbohvztANh4= github.com/go-kit/kit v0.12.0/go.mod h1:lHd+EkCZPIwYItmGDDRdhinkzX2A1sj+M9biaEaizzs= -github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY= -github.com/go-kit/log v0.2.0/go.mod h1:NwTd00d/i8cPZ3xOwwiv2PO5MOcx78fFErGNcVmBjv0= github.com/go-kit/log v0.2.1 h1:MRVx0/zhvdseW+Gza6N9rVzU/IVzaeE1SFI4raAhmBU= github.com/go-kit/log v0.2.1/go.mod h1:NwTd00d/i8cPZ3xOwwiv2PO5MOcx78fFErGNcVmBjv0= github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE= github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= -github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A= github.com/go-logfmt/logfmt v0.5.1 h1:otpy5pqBCBZ1ng9RQ0dPu4PN7ba75Y/aA+UpowDyNVA= github.com/go-logfmt/logfmt v0.5.1/go.mod h1:WYhtIu8zTZfxdn5+rREduYbwxfcBr/Vr6KEVveWlfTs= +github.com/go-logr/logr v1.2.3 h1:2DntVwHkVopvECVRSlL5PSo9eG+cAkDCuckLubN+rq0= +github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4= +github.com/go-playground/locales v0.13.0 h1:HyWk6mgj5qFqCT5fjGBuRArbVDfE4hi8+e8ceBS/t7Q= +github.com/go-playground/locales v0.13.0/go.mod h1:taPMhCMXrRLJO55olJkUXHZBHCxTMfnGwq/HNwmWNS8= +github.com/go-playground/universal-translator v0.17.0 h1:icxd5fm+REJzpZx7ZfpaD876Lmtgy7VtROAbHHXk8no= +github.com/go-playground/universal-translator v0.17.0/go.mod h1:UkSxE5sNxxRwHyU+Scu5vgOQjsIJAF8j9muTVoKLVtA= +github.com/go-playground/validator/v10 v10.2.0 h1:KgJ0snyC2R9VXYN2rneOtQcw5aHQB1Vv0sFl1UcHBOY= +github.com/go-playground/validator/v10 v10.2.0/go.mod h1:uOYAAleCW8F/7oMFd6aG0GOhaH6EGOAJShg8Id5JGkI= github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= +github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 h1:tfuBGBXKqDEevZMzYi5KSi8KkcZtzBcTgAUUtapy0OI= +github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572/go.mod h1:9Pwr4B2jHnOSGXyyzV8ROjYa2ojvAY6HCGYYfMoC3Ls= +github.com/gobwas/httphead v0.0.0-20180130184737-2c6c146eadee h1:s+21KNqlpePfkah2I+gwHF8xmJWRjooY+5248k6m4A0= +github.com/gobwas/httphead v0.0.0-20180130184737-2c6c146eadee/go.mod h1:L0fX3K22YWvt/FAX9NnzrNzcI4wNYi9Yku4O0LKYflo= +github.com/gobwas/pool v0.2.0 h1:QEmUOlnSjWtnpRGHF3SauEiOsy82Cup83Vf2LcMlnc8= +github.com/gobwas/pool v0.2.0/go.mod h1:q8bcK0KcYlCgd9e7WYLm9LpyS+YeLd8JVDW6WezmKEw= +github.com/gobwas/ws v1.0.2 h1:CoAavW/wd/kulfZmSIBt6p24n4j7tHgNVCjsfHVNUbo= +github.com/gobwas/ws v1.0.2/go.mod h1:szmBTxLgaFppYjEmNtny/v3w89xOydFnnZMcgRRu/EM= github.com/godbus/dbus v0.0.0-20190726142602-4481cbc300e2 h1:ZpnhV/YsD2/4cESfV5+Hoeu/iUR3ruzNvZ+yQfO03a0= github.com/godbus/dbus v0.0.0-20190726142602-4481cbc300e2/go.mod h1:bBOAhwG1umN6/6ZUMtDFBMQR8jRg9O75tm9K00oMsK4= +github.com/godbus/dbus/v5 v5.0.3/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= github.com/godbus/dbus/v5 v5.0.6/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= +github.com/godbus/dbus/v5 v5.1.0 h1:4KLkAxT3aOY8Li4FRJe/KvhoNFFxo0m6fNuFUO8QJUk= +github.com/godbus/dbus/v5 v5.1.0/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= github.com/gofrs/uuid v4.3.0+incompatible h1:CaSVZxm5B+7o45rtab4jC2G37WGYX1zQfuU2i6DSvnc= github.com/gogo/googleapis v1.4.1 h1:1Yx4Myt7BxzvUr5ldGSbwYiZG6t9wGBZ+8/fX3Wvtq0= github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= +github.com/gogo/protobuf v1.3.1/go.mod h1:SlYgWuQ5SjCEi6WLHjHCa1yvBfUnHcTbrrZtXPKa29o= github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q= github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= @@ -245,6 +302,7 @@ github.com/golang/glog v1.1.0/go.mod h1:pfYeQZ3JWZoXTV5sFc986z3HTpwQs9At6P4ImfuP github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= +github.com/golang/lint v0.0.0-20180702182130-06c8688daad7/go.mod h1:tluoj9z5200jBnyusfRPU2LqT6J+DAorxEvtC7LHB+E= github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= github.com/golang/mock v1.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y= @@ -253,6 +311,7 @@ github.com/golang/mock v1.4.1/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt github.com/golang/mock v1.4.3/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= github.com/golang/mock v1.4.4/go.mod h1:l3mdAwkq5BuhzHwde/uurv3sEJeZMXNpwsxVWU71h+4= github.com/golang/mock v1.6.0 h1:ErTB+efbowRARo13NNdxyJji2egdxLGQhRaY+DUumQc= +github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs= github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= github.com/golang/protobuf v1.3.0/go.mod h1:Qd/q+1AKNOZr9uGQzbzCmRO6sUih6GTPZv6a1/R87v0= github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= @@ -291,10 +350,15 @@ github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/ github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38= github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= +github.com/google/go-github v17.0.0+incompatible/go.mod h1:zLgOLi98H3fifZn+44m+umXrS52loVEgC2AApnigrVQ= +github.com/google/go-querystring v1.0.0/go.mod h1:odCYkC5MyYFN7vkCjXpyrEuKhc/BUO6wN/zVPAxq5ck= github.com/google/gofuzz v0.0.0-20170612174753-24818f796faf/go.mod h1:HP5RmnzzSNb993RKQDq4+1A4ia9nllfqcQFTQJedwGI= github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= github.com/google/gofuzz v1.2.0 h1:xRy4A+RhZaiKjJ1bPfwQ8sedCA+YS2YcCHW6ec7JMi0= github.com/google/gofuzz v1.2.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= +github.com/google/gopacket v1.1.17/go.mod h1:UdDNZ1OO62aGYVnPhxT1U6aI7ukYtA/kB8vaU0diBUM= +github.com/google/gopacket v1.1.19 h1:ves8RnFZPGiFnTS0uPQStjwru6uO6h+nlr9j6fL7kF8= +github.com/google/gopacket v1.1.19/go.mod h1:iJ8V8n6KS+z2U1A8pUwu8bW5SyEMkXJB8Yo/Vo+TKTo= github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= github.com/google/martian/v3 v3.0.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= github.com/google/martian/v3 v3.1.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= @@ -309,18 +373,26 @@ github.com/google/pprof v0.0.0-20200708004538-1a94d8640e99/go.mod h1:ZgVRPoUq/hf github.com/google/pprof v0.0.0-20201023163331-3e6fc7fc9c4c/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= github.com/google/pprof v0.0.0-20201203190320-1bf35d6f28c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= github.com/google/pprof v0.0.0-20201218002935-b9804c9f04c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= +github.com/google/pprof v0.0.0-20230405160723-4a4c7d95572b h1:Qcx5LM0fSiks9uCyFZwDBUasd3lxd1RM0GYpL+Li5o4= +github.com/google/pprof v0.0.0-20230405160723-4a4c7d95572b/go.mod h1:79YE0hCXdHag9sBkw2o+N/YnZtTkXi0UT9Nnixa5eYk= github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI= github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/googleapis/gax-go v2.0.0+incompatible/go.mod h1:SFVmujtThgffbyetf+mdk2eWhX2bMyUtNHzFKcPA9HY= +github.com/googleapis/gax-go/v2 v2.0.3/go.mod h1:LLvjysVCY1JZeum8Z6l8qUty8fiNwE08qbEPm1M08qg= github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg= github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk= github.com/googleapis/google-cloud-go-testing v0.0.0-20200911160855-bcd43fbb19e8/go.mod h1:dvDLG8qkwmyD9a/MJJN3XJcT3xFxOKAvTZGvuZmac9g= +github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY= github.com/gorilla/handlers v1.5.1 h1:9lRY6j8DEeeBT10CvO9hGW0gmky0BprnvDI5vfhUHH4= github.com/gorilla/mux v1.8.0 h1:i40aqfkR1h2SlN9hojwV5ZA91wcXFOvkdNIeFDP5koI= +github.com/gorilla/websocket v1.4.1/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= github.com/gorilla/websocket v1.5.0 h1:PPwGk2jz7EePpoHN/+ClbZu8SPxiqlu12wZP/3sWmnc= github.com/gorilla/websocket v1.5.0/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= github.com/gotestyourself/gotestyourself v2.2.0+incompatible h1:AQwinXlbQR2HvPjQZOmDhRqsv5mZf+Jb1RnSLxcqZcI= +github.com/gregjones/httpcache v0.0.0-20180305231024-9cad4c3443a7/go.mod h1:FecbI9+v66THATjSRHfNgh1IVFe/9kFxbXtjV0ctIMA= github.com/grpc-ecosystem/go-grpc-middleware v1.3.0 h1:+9834+KizmvFV7pXQGSXQTsaWhq2GjuNUt0aUU0YBYw= github.com/grpc-ecosystem/go-grpc-middleware v1.3.0/go.mod h1:z0ButlSOZa5vEBq9m2m2hlwIgKw+rp3sdCBRoJY+30Y= +github.com/grpc-ecosystem/grpc-gateway v1.5.0/go.mod h1:RSKVYQBd5MCa4OVpNdGskqpgL2+G+NZTnrVHpWWfpdw= github.com/grpc-ecosystem/grpc-gateway v1.16.0 h1:gmcG1KaJ57LophUzW0Hy8NmPhnMZb4M0+kPpLofRdBo= github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw= github.com/gsterjov/go-libsecret v0.0.0-20161001094733-a6f4afe4910c h1:6rhixN/i8ZofjG1Y75iExal34USq5p+wiN1tpie8IrU= @@ -355,6 +427,8 @@ github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ github.com/hashicorp/golang-lru v0.5.4/go.mod h1:iADmTwqILo4mZ8BN3D2Q6+9jd8WM5uGBxy+E8yxSoD4= github.com/hashicorp/golang-lru v0.5.5-0.20210104140557-80c98217689d h1:dg1dEPuWpEqDnvIw251EVy4zlP8gWbsGj4BsUKCRpYs= github.com/hashicorp/golang-lru v0.5.5-0.20210104140557-80c98217689d/go.mod h1:iADmTwqILo4mZ8BN3D2Q6+9jd8WM5uGBxy+E8yxSoD4= +github.com/hashicorp/golang-lru/v2 v2.0.2 h1:Dwmkdr5Nc/oBiXgJS3CDHNhJtIHkuZ3DZF5twqnfBdU= +github.com/hashicorp/golang-lru/v2 v2.0.2/go.mod h1:QeFd9opnmA6QUJc5vARoKUSoFhyfM2/ZepoAG6RGpeM= github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4= github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ= github.com/hashicorp/raft v1.1.0/go.mod h1:4Ak7FSPnuvmb0GV6vgIAJ4vYT4bek9bb6Q+7HVbyzqM= @@ -374,48 +448,99 @@ github.com/huandu/go-assert v1.1.5 h1:fjemmA7sSfYHJD7CUqs9qTwwfdNAx7/j2/ZlHXzNB3 github.com/huandu/go-assert v1.1.5/go.mod h1:yOLvuqZwmcHIC5rIzrBhT7D3Q9c3GFnd0JrPVhn/06U= github.com/huandu/skiplist v1.2.0 h1:gox56QD77HzSC0w+Ws3MH3iie755GBJU1OER3h5VsYw= github.com/huandu/skiplist v1.2.0/go.mod h1:7v3iFjLcSAzO4fN5B8dvebvo/qsfumiLiDXMrPiHF9w= +github.com/huin/goupnp v1.0.0/go.mod h1:n9v9KO1tAxYH82qOn+UTIFQDmx5n1Zxd/ClZDMX7Bnc= +github.com/huin/goupnp v1.1.0 h1:gEe0Dp/lZmPZiDFzJJaOfUpOvv2MKUkoBX8lDrn9vKU= +github.com/huin/goupnp v1.1.0/go.mod h1:gnGPsThkYa7bFi/KWmEysQRf48l2dvR5bxr2OFckNX8= +github.com/huin/goutil v0.0.0-20170803182201-1ca381bf3150/go.mod h1:PpLOETDnJ0o3iZrZfqZzyLl6l7F3c6L1oWn7OICBi6o= github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= github.com/improbable-eng/grpc-web v0.15.0 h1:BN+7z6uNXZ1tQGcNAuaU1YjsLTApzkjt2tzCixLaUPQ= github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= github.com/inconshreveable/mousetrap v1.0.1 h1:U3uMjPSQEBMNp1lFxmllqCPM6P5u/Xq7Pgzkat/bFNc= github.com/inconshreveable/mousetrap v1.0.1/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw= +github.com/ipfs/go-cid v0.4.1 h1:A/T3qGvxi4kpKWWcPC/PgbvDA2bjVLO7n4UeVwnbs/s= +github.com/ipfs/go-cid v0.4.1/go.mod h1:uQHwDeX4c6CtyrFwdqyhpNcxVewur1M7l7fNU7LKwZk= +github.com/ipfs/go-detect-race v0.0.1 h1:qX/xay2W3E4Q1U7d9lNs1sU9nvguX0a7319XbyQ6cOk= +github.com/ipfs/go-detect-race v0.0.1/go.mod h1:8BNT7shDZPo99Q74BpGMK+4D8Mn4j46UU0LZ723meps= +github.com/ipfs/go-log/v2 v2.5.1 h1:1XdUzF7048prq4aBjDQQ4SL5RxftpRGdXhNRwKSAlcY= +github.com/ipfs/go-log/v2 v2.5.1/go.mod h1:prSpmC1Gpllc9UYWxDiZDreBYw7zp4Iqp1kOLU9U5UI= +github.com/jackpal/go-nat-pmp v1.0.2 h1:KzKSgb7qkJvOUTqYl9/Hg/me3pWgBmERKrTGD7BdWus= +github.com/jackpal/go-nat-pmp v1.0.2/go.mod h1:QPH045xvCAeXUZOxsnwmrtiCoxIr9eob+4orBN1SBKc= +github.com/jbenet/go-temp-err-catcher v0.1.0 h1:zpb3ZH6wIE8Shj2sKS+khgRvf7T7RABoLk/+KKHggpk= +github.com/jbenet/go-temp-err-catcher v0.1.0/go.mod h1:0kJRvmDZXNMIiJirNPEYfhpPwbGVtZVWC34vc5WLsDk= +github.com/jellevandenhooff/dkim v0.0.0-20150330215556-f50fe3d243e1/go.mod h1:E0B/fFc00Y+Rasa88328GlI/XbtyysCtTHZS8h7IrBU= github.com/jmhodges/levigo v1.0.0 h1:q5EC36kV79HWeTBWsod3mG11EgStG3qArTKcvlksN1U= github.com/jmhodges/levigo v1.0.0/go.mod h1:Q6Qx+uH3RAqyK4rFQroq9RL7mdkABMcfhEI+nNuzMJQ= -github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX5e0EB2j4= github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= -github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= -github.com/json-iterator/go v1.1.11/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= -github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= +github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM= github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk= github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= -github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM= +github.com/kisielk/errcheck v1.2.0/go.mod h1:/BMXB+zMLi60iA8Vv6Ksmxu/1UDYcXs4uQLJ+jE2L00= github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= +github.com/klauspost/compress v1.10.3/go.mod h1:aoV0uJVorq1K+umq18yTdKaF57EivdYsUV+/s2qKfXs= github.com/klauspost/compress v1.12.3/go.mod h1:8dP1Hq4DHOhN9w426knH3Rhby4rFm6D8eO+e+Dq5Gzg= -github.com/klauspost/compress v1.16.3 h1:XuJt9zzcnaz6a16/OU53ZjWp/v7/42WcR5t2a0PcNQY= -github.com/klauspost/compress v1.16.3/go.mod h1:ntbaceVETuRiXiv4DpjP66DpAtAGkEQskQzEyD//IeE= +github.com/klauspost/compress v1.16.4 h1:91KN02FnsOYhuunwU4ssRe8lc2JosWmizWa91B5v1PU= +github.com/klauspost/compress v1.16.4/go.mod h1:ntbaceVETuRiXiv4DpjP66DpAtAGkEQskQzEyD//IeE= +github.com/klauspost/cpuid/v2 v2.0.4/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= +github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= +github.com/klauspost/cpuid/v2 v2.2.4 h1:acbojRNwl3o09bUq+yDCtZFc1aiwaAAxtcn8YkZXnvk= +github.com/klauspost/cpuid/v2 v2.2.4/go.mod h1:RVVoqg1df56z8g3pUjL/3lE5UfnlrJX8tyFgg4nqhuY= github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= -github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= +github.com/koron/go-ssdp v0.0.0-20191105050749-2e1c40ed0b5d/go.mod h1:5Ky9EC2xfoUKUor0Hjgi2BJhCSXJfMOFlmyYrVKGQMk= +github.com/koron/go-ssdp v0.0.4 h1:1IDwrghSKYM7yLf7XCzbByg2sJ/JcNOZRXS2jczTwz0= +github.com/koron/go-ssdp v0.0.4/go.mod h1:oDXq+E5IL5q0U8uSBcoAXzTzInwy5lEgC91HoKtbmZk= github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg= github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= +github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= +github.com/kr/pty v1.1.3/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= +github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/kraken-hpc/go-fork v0.1.1 h1:O3X/ynoNy/eS7UIcZYef8ndFq2RXEIOue9kZqyzF0Sk= github.com/kraken-hpc/go-fork v0.1.1/go.mod h1:uu0e5h+V4ONH5Qk/xuVlyNXJXy/swhqGIEMK7w+9dNc= +github.com/leodido/go-urn v1.2.0 h1:hpXL4XnriNwQ/ABnpepYM/1vCLWNDfUNts8dX3xTG6Y= +github.com/leodido/go-urn v1.2.0/go.mod h1:+8+nEpDfqqsY+g338gtMEUOtuK+4dEMhiQEgxpxOKII= github.com/lib/pq v1.10.7 h1:p7ZhMD+KsSRozJr34udlUrhboJwWAgCg34+/ZZNvZZw= github.com/libp2p/go-buffer-pool v0.1.0 h1:oK4mSFcQz7cTQIfqbe4MIj9gLW+mnanjyFtc6cdF0Y8= github.com/libp2p/go-buffer-pool v0.1.0/go.mod h1:N+vh8gMqimBzdKkSMVuydVDq+UV5QTWy5HSiZacSbPg= +github.com/libp2p/go-cidranger v1.1.0 h1:ewPN8EZ0dd1LSnrtuwd4709PXVcITVeuwbag38yPW7c= +github.com/libp2p/go-cidranger v1.1.0/go.mod h1:KWZTfSr+r9qEo9OkI9/SIEeAtw+NNoU0dXIXt15Okic= +github.com/libp2p/go-flow-metrics v0.1.0 h1:0iPhMI8PskQwzh57jB9WxIuIOQ0r+15PChFGkx3Q3WM= +github.com/libp2p/go-flow-metrics v0.1.0/go.mod h1:4Xi8MX8wj5aWNDAZttg6UPmc0ZrnFNsMtpsYUClFtro= +github.com/libp2p/go-libp2p v0.27.3 h1:tkV/zm3KCZ4R5er9Xcs2pt0YNB4JH0iBfGAtHJdLHRs= +github.com/libp2p/go-libp2p v0.27.3/go.mod h1:FAvvfQa/YOShUYdiSS03IR9OXzkcJXwcNA2FUCh9ImE= +github.com/libp2p/go-libp2p-asn-util v0.3.0 h1:gMDcMyYiZKkocGXDQ5nsUQyquC9+H+iLEQHwOCZ7s8s= +github.com/libp2p/go-libp2p-asn-util v0.3.0/go.mod h1:B1mcOrKUE35Xq/ASTmQ4tN3LNzVVaMNmq2NACuqyB9w= +github.com/libp2p/go-libp2p-pubsub v0.9.3 h1:ihcz9oIBMaCK9kcx+yHWm3mLAFBMAUsM4ux42aikDxo= +github.com/libp2p/go-libp2p-pubsub v0.9.3/go.mod h1:RYA7aM9jIic5VV47WXu4GkcRxRhrdElWf8xtyli+Dzc= +github.com/libp2p/go-libp2p-testing v0.12.0 h1:EPvBb4kKMWO29qP4mZGyhVzUyR25dvfUIK5WDu6iPUA= +github.com/libp2p/go-msgio v0.3.0 h1:mf3Z8B1xcFN314sWX+2vOTShIE0Mmn2TXn3YCUQGNj0= +github.com/libp2p/go-msgio v0.3.0/go.mod h1:nyRM819GmVaF9LX3l03RMh10QdOroF++NBbxAb0mmDM= +github.com/libp2p/go-nat v0.1.0 h1:MfVsH6DLcpa04Xr+p8hmVRG4juse0s3J8HyNWYHffXg= +github.com/libp2p/go-nat v0.1.0/go.mod h1:X7teVkwRHNInVNWQiO/tAiAVRwSr5zoRz4YSTC3uRBM= +github.com/libp2p/go-netroute v0.1.2/go.mod h1:jZLDV+1PE8y5XxBySEBgbuVAXbhtuHSdmLPL2n9MKbk= +github.com/libp2p/go-netroute v0.2.1 h1:V8kVrpD8GK0Riv15/7VN6RbUQ3URNZVosw7H2v9tksU= +github.com/libp2p/go-netroute v0.2.1/go.mod h1:hraioZr0fhBjG0ZRXJJ6Zj2IVEVNx6tDTFQfSmcq7mQ= +github.com/libp2p/go-reuseport v0.2.0 h1:18PRvIMlpY6ZK85nIAicSBuXXvrYoSw3dsBAR7zc560= +github.com/libp2p/go-reuseport v0.2.0/go.mod h1:bvVho6eLMm6Bz5hmU0LYN3ixd3nPPvtIlaURZZgOY4k= +github.com/libp2p/go-sockaddr v0.0.2/go.mod h1:syPvOmNs24S3dFVGJA1/mrqdeijPxLV2Le3BRLKd68k= +github.com/libp2p/go-yamux/v4 v4.0.0 h1:+Y80dV2Yx/kv7Y7JKu0LECyVdMXm1VUoko+VQ9rBfZQ= +github.com/libp2p/go-yamux/v4 v4.0.0/go.mod h1:NWjl8ZTLOGlozrXSOZ/HlfG++39iKNnM5wwmtQP1YB4= +github.com/lunixbochs/vtclean v1.0.0/go.mod h1:pHhQNgMf3btfWnGBVipUOjRYhoOsdGqdm/+2c2E2WMI= github.com/magiconair/properties v1.8.0/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ= github.com/magiconair/properties v1.8.6 h1:5ibWZ6iY0NctNGWo87LalDlEZ6R41TqbbDamhfG/Qzo= github.com/magiconair/properties v1.8.6/go.mod h1:y3VJvCyxH9uVvJTWEGAELF3aiYNyPKd5NZ3oSwXrF60= +github.com/mailru/easyjson v0.0.0-20190312143242-1de009706dbe/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= github.com/manifoldco/promptui v0.9.0 h1:3V4HzJk1TtXW1MTZMP7mdlwbBpIinw3HztaIlYthEiA= +github.com/marten-seemann/tcp v0.0.0-20210406111302-dfbc87cc63fd h1:br0buuQ854V8u83wA0rVZ8ttrq5CpaPZdvrK0LP2lOk= +github.com/marten-seemann/tcp v0.0.0-20210406111302-dfbc87cc63fd/go.mod h1:QuCEs1Nt24+FYQEqAAncTDPJIuGs+LxK1MCiFL25pMU= github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= github.com/mattn/go-colorable v0.1.8/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= github.com/mattn/go-colorable v0.1.9/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= @@ -426,15 +551,30 @@ github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hd github.com/mattn/go-isatty v0.0.10/go.mod h1:qgIWMr58cqv1PHHyhnkY9lrL7etaEgOFcMEpPG5Rm84= github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= -github.com/mattn/go-isatty v0.0.16 h1:bq3VjFmv/sOjHtdEhmkEV4x1AJtvUvOJ2PFAZ5+peKQ= github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= +github.com/mattn/go-isatty v0.0.18 h1:DOKFKCQ7FNG2L1rbrmstDN4QVRdS89Nkh85u68Uwp98= +github.com/mattn/go-isatty v0.0.18/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= -github.com/matttproud/golang_protobuf_extensions v1.0.2-0.20181231171920-c182affec369 h1:I0XW9+e1XWDxdcEniV4rQAIOPUGDq67JSCiRCgGCZLI= -github.com/matttproud/golang_protobuf_extensions v1.0.2-0.20181231171920-c182affec369/go.mod h1:BSXmuO+STAnVfrANrmjBb36TMTDstsz7MSK+HVaYKv4= +github.com/matttproud/golang_protobuf_extensions v1.0.4 h1:mmDVorXM7PCGKw94cs5zkfA9PSy5pEvNWRP0ET0TIVo= +github.com/matttproud/golang_protobuf_extensions v1.0.4/go.mod h1:BSXmuO+STAnVfrANrmjBb36TMTDstsz7MSK+HVaYKv4= +github.com/microcosm-cc/bluemonday v1.0.1/go.mod h1:hsXNsILzKxV+sX77C5b8FSuKF00vh2OMYv+xgHpAMF4= +github.com/miekg/dns v1.1.41/go.mod h1:p6aan82bvRIyn+zDIv9xYNUpwa73JcSh9BKwknJysuI= +github.com/miekg/dns v1.1.53 h1:ZBkuHr5dxHtB1caEOlZTLPo7D3L3TWckgUUs/RHfDxw= +github.com/miekg/dns v1.1.53/go.mod h1:uInx36IzPl7FYnDcMeVWxj9byh7DutNykX4G9Sj60FY= +github.com/mikioh/tcp v0.0.0-20190314235350-803a9b46060c h1:bzE/A84HN25pxAuk9Eej1Kz9OUelF97nAc82bDquQI8= +github.com/mikioh/tcp v0.0.0-20190314235350-803a9b46060c/go.mod h1:0SQS9kMwD2VsyFEB++InYyBJroV/FRmBgcydeSUcJms= +github.com/mikioh/tcpinfo v0.0.0-20190314235526-30a79bb1804b h1:z78hV3sbSMAUoyUMM0I83AUIT6Hu17AWfgjzIbtrYFc= +github.com/mikioh/tcpinfo v0.0.0-20190314235526-30a79bb1804b/go.mod h1:lxPUiZwKoFL8DUUmalo2yJJUCxbPKtm8OKfqr2/FTNU= +github.com/mikioh/tcpopt v0.0.0-20190314235656-172688c1accc h1:PTfri+PuQmWDqERdnNMiD9ZejrlswWrCpBEZgWOiTrc= +github.com/mikioh/tcpopt v0.0.0-20190314235656-172688c1accc/go.mod h1:cGKTAVKx4SxOuR/czcZ/E2RSJ3sfHs8FpHhQ5CWMf9s= github.com/mimoo/StrobeGo v0.0.0-20181016162300-f8f6d4d2b643/go.mod h1:43+3pMjjKimDBf5Kr4ZFNGbLql1zKkbImw+fZbw3geM= github.com/mimoo/StrobeGo v0.0.0-20210601165009-122bf33a46e0 h1:QRUSJEgZn2Snx0EmT/QLXibWjSUDjKWvXIT19NBVp94= github.com/mimoo/StrobeGo v0.0.0-20210601165009-122bf33a46e0/go.mod h1:43+3pMjjKimDBf5Kr4ZFNGbLql1zKkbImw+fZbw3geM= +github.com/minio/blake2b-simd v0.0.0-20160723061019-3f5f724cb5b1/go.mod h1:pD8RvIylQ358TN4wwqatJ8rNavkEINozVn9DtGI3dfQ= github.com/minio/highwayhash v1.0.2 h1:Aak5U0nElisjDCfPSG79Tgzkn2gl66NxOMspRrKnA/g= +github.com/minio/sha256-simd v0.1.1-0.20190913151208-6de447530771/go.mod h1:B5e1o+1/KgNmWrSQK08Y6Z1Vb5pwIktudl0J58iy0KM= +github.com/minio/sha256-simd v1.0.0 h1:v1ta+49hkWZyvaKwrQB8elexRqm6Y0aMLjCNsrYxo6g= +github.com/minio/sha256-simd v1.0.0/go.mod h1:OuYzVNI5vcoYIAmbIvHPl3N3jUzVedXbKy5RFepssQM= github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y= github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= @@ -442,38 +582,74 @@ github.com/mitchellh/mapstructure v1.5.0 h1:jeMsZIYE/09sWLaz43PL7Gy6RuMjD2eJVyua github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo= github.com/moby/sys/mountinfo v0.5.0/go.mod h1:3bMD3Rg+zkqx8MRYPi7Pyb0Ie97QEBmdxbhnCLlSvSU= github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg= github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= -github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= +github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M= +github.com/mr-tron/base58 v1.1.2/go.mod h1:BinMc/sQntlIE1frQmRFPUoPA1Zkr8VRgBdjWI2mNwc= +github.com/mr-tron/base58 v1.2.0 h1:T/HDJBh4ZCPbU39/+c3rRvE0uKBQlU27+QI8LJ4t64o= +github.com/mr-tron/base58 v1.2.0/go.mod h1:BinMc/sQntlIE1frQmRFPUoPA1Zkr8VRgBdjWI2mNwc= github.com/mrunalp/fileutils v0.5.0/go.mod h1:M1WthSahJixYnrXQl/DFQuteStB1weuxD2QJNHXfbSQ= github.com/mtibben/percent v0.2.1 h1:5gssi8Nqo8QU/r2pynCm+hBQHpkB/uNK7BJCFogWdzs= github.com/mtibben/percent v0.2.1/go.mod h1:KG9uO+SZkUp+VkRHsCdYQV3XSZrrSpR3O9ibNBTZrns= +github.com/multiformats/go-base32 v0.1.0 h1:pVx9xoSPqEIQG8o+UbAe7DNi51oej1NtK+aGkbLYxPE= +github.com/multiformats/go-base32 v0.1.0/go.mod h1:Kj3tFY6zNr+ABYMqeUNeGvkIC/UYgtWibDcT0rExnbI= +github.com/multiformats/go-base36 v0.2.0 h1:lFsAbNOGeKtuKozrtBsAkSVhv1p9D0/qedU9rQyccr0= +github.com/multiformats/go-base36 v0.2.0/go.mod h1:qvnKE++v+2MWCfePClUEjE78Z7P2a1UV0xHgWc0hkp4= +github.com/multiformats/go-multiaddr v0.1.1/go.mod h1:aMKBKNEYmzmDmxfX88/vz+J5IU55txyt0p4aiWVohjo= +github.com/multiformats/go-multiaddr v0.2.0/go.mod h1:0nO36NvPpyV4QzvTLi/lafl2y95ncPj0vFwVF6k6wJ4= +github.com/multiformats/go-multiaddr v0.9.0 h1:3h4V1LHIk5w4hJHekMKWALPXErDfz/sggzwC/NcqbDQ= +github.com/multiformats/go-multiaddr v0.9.0/go.mod h1:mI67Lb1EeTOYb8GQfL/7wpIZwc46ElrvzhYnoJOmTT0= +github.com/multiformats/go-multiaddr-dns v0.3.1 h1:QgQgR+LQVt3NPTjbrLLpsaT2ufAA2y0Mkk+QRVJbW3A= +github.com/multiformats/go-multiaddr-dns v0.3.1/go.mod h1:G/245BRQ6FJGmryJCrOuTdB37AMA5AMOVuO6NY3JwTk= +github.com/multiformats/go-multiaddr-fmt v0.1.0 h1:WLEFClPycPkp4fnIzoFoV9FVd49/eQsuaL3/CWe167E= +github.com/multiformats/go-multiaddr-fmt v0.1.0/go.mod h1:hGtDIW4PU4BqJ50gW2quDuPVjyWNZxToGUh/HwTZYJo= +github.com/multiformats/go-multibase v0.2.0 h1:isdYCVLvksgWlMW9OZRYJEa9pZETFivncJHmHnnd87g= +github.com/multiformats/go-multibase v0.2.0/go.mod h1:bFBZX4lKCA/2lyOFSAoKH5SS6oPyjtnzK/XTFDPkNuk= +github.com/multiformats/go-multicodec v0.8.1 h1:ycepHwavHafh3grIbR1jIXnKCsFm0fqsfEOsJ8NtKE8= +github.com/multiformats/go-multicodec v0.8.1/go.mod h1:L3QTQvMIaVBkXOXXtVmYE+LI16i14xuaojr/H7Ai54k= +github.com/multiformats/go-multihash v0.0.8/go.mod h1:YSLudS+Pi8NHE7o6tb3D8vrpKa63epEDmG8nTduyAew= +github.com/multiformats/go-multihash v0.2.1 h1:aem8ZT0VA2nCHHk7bPJ1BjUbHNciqZC/d16Vve9l108= +github.com/multiformats/go-multihash v0.2.1/go.mod h1:WxoMcYG85AZVQUyRyo9s4wULvW5qrI9vb2Lt6evduFc= +github.com/multiformats/go-multistream v0.4.1 h1:rFy0Iiyn3YT0asivDUIR05leAdwZq3de4741sbiSdfo= +github.com/multiformats/go-multistream v0.4.1/go.mod h1:Mz5eykRVAjJWckE2U78c6xqdtyNUEhKSM0Lwar2p77Q= +github.com/multiformats/go-varint v0.0.1/go.mod h1:3Ls8CIEsrijN6+B7PbrXRPxHRPuXSrVKRY101jdMZYE= +github.com/multiformats/go-varint v0.0.7 h1:sWSGR+f/eu5ABZA2ZpYKBILXTTs9JWpdEM/nEGOHFS8= +github.com/multiformats/go-varint v0.0.7/go.mod h1:r8PUYw/fD/SjBCiKOoDlGF6QawOELpZAu9eioSos/OU= github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= -github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= +github.com/neelance/astrewrite v0.0.0-20160511093645-99348263ae86/go.mod h1:kHJEU3ofeGjhHklVoIGuVj85JJwZ6kWPaJwCIxgnFmo= +github.com/neelance/sourcemap v0.0.0-20151028013722-8c68805598ab/go.mod h1:Qr6/a/Q4r9LP1IltGz7tA7iOK1WonHEYhu1HRBA7ZiM= github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno= github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= github.com/onsi/ginkgo v1.14.0/go.mod h1:iSB4RoI2tjJc9BBv4NKIKWKya62Rps+oPG/Lv9klQyY= -github.com/onsi/ginkgo v1.16.4 h1:29JGrr5oVBm5ulCWet69zQkzWipVXIol6ygQUe/EzNc= +github.com/onsi/ginkgo v1.16.5 h1:8xi0RTUf59SOSfEtZMvwTvXYMzG4gV23XVHOZiXNtnE= +github.com/onsi/ginkgo/v2 v2.9.2 h1:BA2GMJOtfGAfagzYtrAlufIP0lq6QERkFmHLMLPwFSU= +github.com/onsi/ginkgo/v2 v2.9.2/go.mod h1:WHcJJG2dIlcCqVfBAwUCrJxSPFb6v4azBwgxeMeDuts= github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY= github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo= -github.com/onsi/gomega v1.20.0 h1:8W0cWlwFkflGPLltQvLRB7ZVD5HuP6ng320w2IS245Q= +github.com/onsi/gomega v1.27.4 h1:Z2AnStgsdSayCMDiCU42qIz+HLqEPcgiOCXjAU/w+8E= github.com/opencontainers/go-digest v1.0.0 h1:apOUWs51W5PlhuyGyz9FCeeBIOUDA/6nW8Oi/yOhh5U= github.com/opencontainers/go-digest v1.0.0/go.mod h1:0JzlMkj0TRzQZfJkVvzbP0HBR3IKzErnv2BNG4W4MAM= github.com/opencontainers/image-spec v1.1.0-rc2 h1:2zx/Stx4Wc5pIPDvIxHXvXtQFW/7XWJGmnM7r3wg034= github.com/opencontainers/image-spec v1.1.0-rc2/go.mod h1:3OVijpioIKYWTqjiG0zfF6wvoJ4fAXGbjdZuI2NgsRQ= github.com/opencontainers/runc v1.1.3 h1:vIXrkId+0/J2Ymu2m7VjGvbSlAId9XNRPhn2p4b+d8w= github.com/opencontainers/runc v1.1.3/go.mod h1:1J5XiS+vdZ3wCyZybsuxXZWGrgSr8fFJHLXuG2PsnNg= +github.com/opencontainers/runtime-spec v1.0.2/go.mod h1:jwyrGlmzljRJv/Fgzds9SsS/C5hL+LL3ko9hs6T5lQ0= +github.com/opencontainers/runtime-spec v1.0.3-0.20210326190908-1c3f411f0417 h1:3snG66yBm59tKhhSPQrQ/0bCrv1LQbKt40LnUPiUxdc= github.com/opencontainers/runtime-spec v1.0.3-0.20210326190908-1c3f411f0417/go.mod h1:jwyrGlmzljRJv/Fgzds9SsS/C5hL+LL3ko9hs6T5lQ0= github.com/opencontainers/selinux v1.10.0/go.mod h1:2i0OySw99QjzBBQByd1Gr9gSjvuho1lHsJxIJ3gGbJI= github.com/opentracing/opentracing-go v1.1.0/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o= +github.com/openzipkin/zipkin-go v0.1.1/go.mod h1:NtoC/o8u3JlF1lSlyPNswIbeQH9bJTmOf0Erfk+hxe8= github.com/ory/dockertest v3.3.5+incompatible h1:iLLK6SQwIhcbrG783Dghaaa3WPzGc+4Emza6EbVUUGA= github.com/ory/dockertest v3.3.5+incompatible/go.mod h1:1vX4m9wsvi00u5bseYwXaSnhNrne+V0E6LAcBILJdPs= github.com/pascaldekloe/goe v0.1.0 h1:cBOtyMzM9HTpWjXfbbunk26uA6nG3a8n06Wieeh0MwY= github.com/pascaldekloe/goe v0.1.0/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= +github.com/pbnjay/memory v0.0.0-20210728143218-7b4eea64cf58 h1:onHthvaw9LFnH4t2DcNVpwGmV9E1BkGknEliJkfwQj0= +github.com/pbnjay/memory v0.0.0-20210728143218-7b4eea64cf58/go.mod h1:DXv8WO4yhMYhSNPKjeNKa5WY9YCIEBRbNzFFPJbWO6Y= github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic= github.com/pelletier/go-toml v1.9.5 h1:4yBQzkHv+7BHq2PQUZF3Mx0IYxG7LsP222s7Agd3ve8= github.com/pelletier/go-toml v1.9.5/go.mod h1:u1nR/EPcESfeI/szUZKdtJ0xRNbUoANCkoOuaOx1Y+c= @@ -481,6 +657,7 @@ github.com/pelletier/go-toml/v2 v2.0.5 h1:ipoSadvV8oGUjnUbMub59IDPPwfxF694nG/jwb github.com/pelletier/go-toml/v2 v2.0.5/go.mod h1:OMHamSCAODeSsVrwwvcJOaoN0LIUIaFVNZzmWyNfXas= github.com/petermattis/goid v0.0.0-20180202154549-b0b1615b78e5 h1:q2e307iGHPdTGp0hoxKjt1H5pDo6utceo3dQVK3I5XQ= github.com/petermattis/goid v0.0.0-20180202154549-b0b1615b78e5/go.mod h1:jvVRKCrJTQWu0XVbaOlby/2lO20uSCHEMzzplHXte1o= +github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA= github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= @@ -488,13 +665,11 @@ github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINE github.com/pkg/sftp v1.13.1/go.mod h1:3HaPG6Dq1ILlpPZRO0HVMrsydcdLt6HRDccSgb87qRg= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/prometheus/client_golang v0.8.0/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= github.com/prometheus/client_golang v0.9.2/go.mod h1:OsXs2jCmiKlQ1lTBmv21f2mNfw4xf/QclQDMrYNZzcM= github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= github.com/prometheus/client_golang v1.4.0/go.mod h1:e9GMxYsXl05ICDXkRhurwBS4Q3OK1iX/F2sw+iXX5zU= -github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M= -github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= -github.com/prometheus/client_golang v1.12.1/go.mod h1:3Z9XVyYiZYEO+YQWt3RD2R3jrbd179Rt297l4aS6nDY= github.com/prometheus/client_golang v1.14.0 h1:nJdhIvne2eSX/XRAFV9PcvFFRbrjbcTUj0VP62TMhnw= github.com/prometheus/client_golang v1.14.0/go.mod h1:8vpkKitgIVNcqrRBWh1C4TIUQgYNtG/XQE4E/Zae36Y= github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= @@ -503,30 +678,39 @@ github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1: github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= github.com/prometheus/client_model v0.3.0 h1:UBgGFHqYdG/TPFD1B1ogZywDqEkwp3fBMvqdiQ7Xew4= github.com/prometheus/client_model v0.3.0/go.mod h1:LDGWKZIo7rky3hgvBe+caln+Dr3dPggB5dvjtD7w9+w= +github.com/prometheus/common v0.0.0-20180801064454-c7de2306084e/go.mod h1:daVV7qP5qjZbuso7PdcryaAu0sAZbrN9i7WWcTMWvro= github.com/prometheus/common v0.0.0-20181126121408-4724e9255275/go.mod h1:daVV7qP5qjZbuso7PdcryaAu0sAZbrN9i7WWcTMWvro= github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= github.com/prometheus/common v0.9.1/go.mod h1:yhUN8i9wzaXS3w1O07YhxHEBxD+W35wd8bs7vj7HSQ4= -github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo= -github.com/prometheus/common v0.26.0/go.mod h1:M7rCNAaPfAosfx8veZJCuw84e35h3Cfd9VFqTh1DIvc= -github.com/prometheus/common v0.32.1/go.mod h1:vu+V0TpY+O6vW9J44gczi3Ap/oXXR10b+M/gUGO4Hls= -github.com/prometheus/common v0.37.0 h1:ccBbHCgIiT9uSoFY0vX8H3zsNR5eLt17/RQLUvn8pXE= -github.com/prometheus/common v0.37.0/go.mod h1:phzohg0JFMnBEFGxTDbfu3QyL5GI8gTQJFhYO5B3mfA= +github.com/prometheus/common v0.42.0 h1:EKsfXEYo4JpWMHH5cg+KOUWeuJSov1Id8zGR8eeI1YM= +github.com/prometheus/common v0.42.0/go.mod h1:xBwqVerjNdUDjgODMpudtOMwlOwf2SaTr1yjz4b7Zbc= +github.com/prometheus/procfs v0.0.0-20180725123919-05ee40e3a273/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= github.com/prometheus/procfs v0.0.0-20181204211112-1dc9a6cbc91a/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= github.com/prometheus/procfs v0.0.8/go.mod h1:7Qr8sr6344vo1JqZ6HhLceV9o3AJ1Ff+GxbHq6oeK9A= -github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU= -github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= -github.com/prometheus/procfs v0.7.3/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= -github.com/prometheus/procfs v0.8.0 h1:ODq8ZFEaYeCaZOJlZZdJA2AbQR98dSHSM1KW/You5mo= -github.com/prometheus/procfs v0.8.0/go.mod h1:z7EfXMXOkbkqb9IINtpCn86r/to3BnA0uaxHdg830/4= +github.com/prometheus/procfs v0.9.0 h1:wzCHvIvM5SxWqYvwgVL7yJY8Lz3PKn49KQtpgMYJfhI= +github.com/prometheus/procfs v0.9.0/go.mod h1:+pB4zwohETzFnmlpe6yd2lSc+0/46IYZRB/chUwxUZY= +github.com/quic-go/qpack v0.4.0 h1:Cr9BXA1sQS2SmDUWjSofMPNKmvF6IiIfDRmgU0w1ZCo= +github.com/quic-go/qpack v0.4.0/go.mod h1:UZVnYIfi5GRk+zI9UMaCPsmZ2xKJP7XBUvVyT1Knj9A= +github.com/quic-go/qtls-go1-19 v0.3.2 h1:tFxjCFcTQzK+oMxG6Zcvp4Dq8dx4yD3dDiIiyc86Z5U= +github.com/quic-go/qtls-go1-19 v0.3.2/go.mod h1:ySOI96ew8lnoKPtSqx2BlI5wCpUVPT05RMAlajtnyOI= +github.com/quic-go/qtls-go1-20 v0.2.2 h1:WLOPx6OY/hxtTxKV1Zrq20FtXtDEkeY00CGQm8GEa3E= +github.com/quic-go/qtls-go1-20 v0.2.2/go.mod h1:JKtK6mjbAVcUTN/9jZpvLbGxvdWIKS8uT7EiStoU1SM= +github.com/quic-go/quic-go v0.33.0 h1:ItNoTDN/Fm/zBlq769lLJc8ECe9gYaW40veHCCco7y0= +github.com/quic-go/quic-go v0.33.0/go.mod h1:YMuhaAV9/jIu0XclDXwZPAsP/2Kgr5yMYhe9oxhhOFA= +github.com/quic-go/webtransport-go v0.5.2 h1:GA6Bl6oZY+g/flt00Pnu0XtivSD8vukOu3lYhJjnGEk= +github.com/quic-go/webtransport-go v0.5.2/go.mod h1:OhmmgJIzTTqXK5xvtuX0oBpLV2GkLWNDA+UeTGJXErU= github.com/rakyll/statik v0.1.7 h1:OF3QCZUuyPxuGEP7B4ypUa7sB/iHtqOTDYZXGM8KOdQ= +github.com/raulk/go-watchdog v1.3.0 h1:oUmdlHxdkXRJlwfG0O9omj8ukerm8MEQavSiDTEtBsk= +github.com/raulk/go-watchdog v1.3.0/go.mod h1:fIvOnLbF0b0ZwkB9YU4mOW9Did//4vPZtDqv66NfsMU= github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475 h1:N/ElC8H3+5XpJzTSTfLsJV/mx9Q9g7kxmchpfZyxgzM= github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4= github.com/regen-network/gocuke v0.6.2 h1:pHviZ0kKAq2U2hN2q3smKNxct6hS0mGByFMHGnWA97M= github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ= github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8= +github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs= github.com/rs/cors v1.8.2 h1:KCooALfAYGs415Cwu5ABvv9n9509fSiG5SQJn/AQo4U= github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g= github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= @@ -534,13 +718,38 @@ github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQD github.com/sasha-s/go-deadlock v0.3.1 h1:sqv7fDNShgjcaxkO0JNcOAlr8B9+cV5Ey/OB71efZx0= github.com/sasha-s/go-deadlock v0.3.1/go.mod h1:F73l+cr82YSh10GxyRI6qZiCgK64VaZjwesgfQ1/iLM= github.com/seccomp/libseccomp-golang v0.9.2-0.20220502022130-f33da4d89646/go.mod h1:JA8cRccbGaA1s33RQf7Y1+q9gHmZX1yB/z9WDN1C6fg= +github.com/sergi/go-diff v1.0.0/go.mod h1:0CfEIISq7TuYL3j771MWULgwwjU+GofnZX9QAmXWZgo= +github.com/shurcooL/component v0.0.0-20170202220835-f88ec8f54cc4/go.mod h1:XhFIlyj5a1fBNx5aJTbKoIq0mNaPvOagO+HjB3EtxrY= +github.com/shurcooL/events v0.0.0-20181021180414-410e4ca65f48/go.mod h1:5u70Mqkb5O5cxEA8nxTsgrgLehJeAw6Oc4Ab1c/P1HM= +github.com/shurcooL/github_flavored_markdown v0.0.0-20181002035957-2122de532470/go.mod h1:2dOwnU2uBioM+SGy2aZoq1f/Sd1l9OkAeAUvjSyvgU0= +github.com/shurcooL/go v0.0.0-20180423040247-9e1955d9fb6e/go.mod h1:TDJrrUr11Vxrven61rcy3hJMUqaf/CLWYhHNPmT14Lk= +github.com/shurcooL/go-goon v0.0.0-20170922171312-37c2f522c041/go.mod h1:N5mDOmsrJOB+vfqUK+7DmDyjhSLIIBnXo9lvZJj3MWQ= +github.com/shurcooL/gofontwoff v0.0.0-20180329035133-29b52fc0a18d/go.mod h1:05UtEgK5zq39gLST6uB0cf3NEHjETfB4Fgr3Gx5R9Vw= +github.com/shurcooL/gopherjslib v0.0.0-20160914041154-feb6d3990c2c/go.mod h1:8d3azKNyqcHP1GaQE/c6dDgjkgSx2BZ4IoEi4F1reUI= +github.com/shurcooL/highlight_diff v0.0.0-20170515013008-09bb4053de1b/go.mod h1:ZpfEhSmds4ytuByIcDnOLkTHGUI6KNqRNPDLHDk+mUU= +github.com/shurcooL/highlight_go v0.0.0-20181028180052-98c3abbbae20/go.mod h1:UDKB5a1T23gOMUJrI+uSuH0VRDStOiUVSjBTRDVBVag= +github.com/shurcooL/home v0.0.0-20181020052607-80b7ffcb30f9/go.mod h1:+rgNQw2P9ARFAs37qieuu7ohDNQ3gds9msbT2yn85sg= +github.com/shurcooL/htmlg v0.0.0-20170918183704-d01228ac9e50/go.mod h1:zPn1wHpTIePGnXSHpsVPWEktKXHr6+SS6x/IKRb7cpw= +github.com/shurcooL/httperror v0.0.0-20170206035902-86b7830d14cc/go.mod h1:aYMfkZ6DWSJPJ6c4Wwz3QtW22G7mf/PEgaB9k/ik5+Y= +github.com/shurcooL/httpfs v0.0.0-20171119174359-809beceb2371/go.mod h1:ZY1cvUeJuFPAdZ/B6v7RHavJWZn2YPVFQ1OSXhCGOkg= +github.com/shurcooL/httpgzip v0.0.0-20180522190206-b1c53ac65af9/go.mod h1:919LwcH0M7/W4fcZ0/jy0qGght1GIhqyS/EgWGH2j5Q= +github.com/shurcooL/issues v0.0.0-20181008053335-6292fdc1e191/go.mod h1:e2qWDig5bLteJ4fwvDAc2NHzqFEthkqn7aOZAOpj+PQ= +github.com/shurcooL/issuesapp v0.0.0-20180602232740-048589ce2241/go.mod h1:NPpHK2TI7iSaM0buivtFUc9offApnI0Alt/K8hcHy0I= +github.com/shurcooL/notifications v0.0.0-20181007000457-627ab5aea122/go.mod h1:b5uSkrEVM1jQUspwbixRBhaIjIzL2xazXp6kntxYle0= +github.com/shurcooL/octicon v0.0.0-20181028054416-fa4f57f9efb2/go.mod h1:eWdoE5JD4R5UVWDucdOPg1g2fqQRq78IQa9zlOV1vpQ= +github.com/shurcooL/reactions v0.0.0-20181006231557-f2e0b4ca5b82/go.mod h1:TCR1lToEk4d2s07G3XGfz2QrgHXg4RJBvjrOozvoWfk= +github.com/shurcooL/sanitized_anchor_name v0.0.0-20170918181015-86672fcb3f95/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= +github.com/shurcooL/users v0.0.0-20180125191416-49c67e49c537/go.mod h1:QJTqeLYEDaXHZDBsXlPCDqdhQuJkuw4NOtaxYe3xii4= +github.com/shurcooL/webdavfs v0.0.0-20170829043945-18c3829fa133/go.mod h1:hKmq5kWdCj2z2KEozexVbfEZIWiTjhE0+UjmZgPqehw= github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= -github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88= +github.com/sirupsen/logrus v1.7.0/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= github.com/sirupsen/logrus v1.8.1/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= github.com/sirupsen/logrus v1.9.0 h1:trlNQbNUG3OdDrDil03MCb1H2o9nJ1x4/5LYw7byDE0= github.com/sirupsen/logrus v1.9.0/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= +github.com/sourcegraph/annotate v0.0.0-20160123013949-f4cad6c6324d/go.mod h1:UdhH50NIW0fCiwBSr0co2m7BnFLdv4fQTgdqdJTHFeE= +github.com/sourcegraph/syntaxhighlight v0.0.0-20170531221838-bd320f5d308e/go.mod h1:HuIsMU8RRBOtsCgI77wP899iHVBQpCmg4ErYMZB+2IA= github.com/spaolacci/murmur3 v0.0.0-20180118202830-f09979ecbc72/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA= github.com/spaolacci/murmur3 v1.1.0 h1:7c1g84S4BPRrfL5Xrdp6fOJ206sU9y293DDHaoy0bLI= github.com/spaolacci/murmur3 v1.1.0/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA= @@ -583,6 +792,7 @@ github.com/subosito/gotenv v1.4.1/go.mod h1:ayKnFf/c6rvx/2iiLrJUk1e6plDbT3edrFNG github.com/syndtr/gocapability v0.0.0-20200815063812-42c35b437635/go.mod h1:hkRG7XYTFWNJGYcbNJQlaLq0fg1yr4J4t/NcTQtrfww= github.com/syndtr/goleveldb v1.0.1-0.20210819022825-2ae1ddf74ef7 h1:epCh84lMvA70Z7CTTCmYQn2CKbY8j86K7/FAIr141uY= github.com/syndtr/goleveldb v1.0.1-0.20210819022825-2ae1ddf74ef7/go.mod h1:q4W45IWZaF22tdD+VEXcAWRA037jwmWEB5VWYORlTpc= +github.com/tarm/serial v0.0.0-20180830185346-98f6abe2eb07/go.mod h1:kDXzergiv9cbyO7IOYJZWg1U88JhDg3PB6klq9Hg2pA= github.com/taurusgroup/frost-ed25519 v0.0.0-20210707140332-5abc84a4dba7 h1:hUvSA2LcuXFrzcU/vDnomemN+trNt+KfrA3s28dfXDw= github.com/taurusgroup/frost-ed25519 v0.0.0-20210707140332-5abc84a4dba7/go.mod h1:HvWEpeV7Nptyy1OAzxcb/DvYOhrFfzMUx0CFrzUgQbg= github.com/tecbot/gorocksdb v0.0.0-20191217155057-f0fad39f321c h1:g+WoO5jjkqGAzHWCjJB1zZfXPIAaDpzXIEJ0eS6B5Ok= @@ -592,8 +802,15 @@ github.com/tendermint/go-amino v0.16.0/go.mod h1:TQU0M1i/ImAo+tYpZi73AU3V/dKeCoM github.com/tidwall/btree v1.6.0 h1:LDZfKfQIBHGHWSwckhXI0RPSXzlo+KYdjK7FWSqOzzg= github.com/tidwall/btree v1.6.0/go.mod h1:twD9XRA5jj9VUQGELzDO4HPQTNJsoWWfYEL+EUQ2cKY= github.com/tv42/httpunix v0.0.0-20150427012821-b75d8614f926/go.mod h1:9ESjWnEqriFuLhtthL60Sar/7RFoluCcXsuvEwTV5KM= +github.com/ugorji/go v1.1.7 h1:/68gy2h+1mWMrwZFeD1kQialdSzAb432dtpeJ42ovdo= +github.com/ugorji/go v1.1.7/go.mod h1:kZn38zHttfInRq0xu/PH0az30d+z6vm202qpg1oXVMw= github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0= +github.com/ugorji/go/codec v1.1.7 h1:2SvQaVZ1ouYrrKKwoSk2pzd4A9evlKJb9oTL+OaLUSs= +github.com/ugorji/go/codec v1.1.7/go.mod h1:Ax+UKWsSmolVDwsd+7N3ZtXu+yMGCf907BLYF3GoBXY= github.com/urfave/cli v1.22.1/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= +github.com/urfave/cli v1.22.2/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= +github.com/viant/assertly v0.4.8/go.mod h1:aGifi++jvCrUaklKEKT0BU95igDNaqkvz+49uaYMPRU= +github.com/viant/toolbox v0.24.0/go.mod h1:OxMCG57V0PXuIP2HNQrtJf2CjqdmbrOx5EkMILuUhzM= github.com/vishvananda/netlink v1.1.0/go.mod h1:cTgwzPIzzgDAYoQrMm0EdrjRUBkTqKYppBueQtXaqoE= github.com/vishvananda/netns v0.0.0-20191106174202-0a2b9b5464df/go.mod h1:JP3t17pCcGlemwknint6hfoeCVQrEMVwxRLRjXpq+BU= github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q= @@ -613,6 +830,7 @@ gitlab.com/unit410/threshold-ed25519 v0.0.0-20220725172740-6ee731f539ac/go.mod h go.etcd.io/bbolt v1.3.5/go.mod h1:G5EMThwa9y8QZGBClrRx5EY+Yw9kAhnjy3bSjsnlVTQ= go.etcd.io/bbolt v1.3.7 h1:j+zJOnnEjF/kyHlDDgGnVL/AIqIJPq8UoB2GSNfkUfQ= go.etcd.io/bbolt v1.3.7/go.mod h1:N9Mkw9X8x5fupy0IKsmuqVtoGDyxsaDlbk4Rd05IAQw= +go.opencensus.io v0.18.0/go.mod h1:vKdFvxhtzZ9onBp9VKHK8z/sRpBMnKAsufL7wlDrCOA= go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU= go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8= go.opencensus.io v0.22.2/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= @@ -621,19 +839,40 @@ go.opencensus.io v0.22.4/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= go.opencensus.io v0.22.5/go.mod h1:5pWMHQbX5EPX2/62yrJeAkowc+lfs/XD7Uxpq3pI6kk= go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqeYNgFYFoEGnI= go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE= +go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= +go.uber.org/atomic v1.10.0 h1:9qC72Qh0+3MqyJbAn8YU5xVq1frD8bn3JtD2oXtafVQ= +go.uber.org/atomic v1.10.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0= +go.uber.org/dig v1.16.1 h1:+alNIBsl0qfY0j6epRubp/9obgtrObRAc5aD+6jbWY8= +go.uber.org/dig v1.16.1/go.mod h1:557JTAUZT5bUK0SvCwikmLPPtdQhfvLYtO5tJgQSbnk= +go.uber.org/fx v1.19.2 h1:SyFgYQFr1Wl0AYstE8vyYIzP4bFz2URrScjwC4cwUvY= +go.uber.org/fx v1.19.2/go.mod h1:43G1VcqSzbIv77y00p1DRAsyZS8WdzuYdhZXmEUkMyQ= +go.uber.org/goleak v1.1.11-0.20210813005559-691160354723/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ= go.uber.org/goleak v1.1.12 h1:gZAh5/EyT/HQwlpkCy6wTpqfH9H8Lz8zbm3dZh+OyzA= go.uber.org/goleak v1.1.12/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ= go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0= +go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU= +go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0= +go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y= go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q= +go.uber.org/zap v1.19.1/go.mod h1:j3DNczoxDZroyBnOT1L/Q79cfUMGZxlv/9dzN7SM1rI= +go.uber.org/zap v1.24.0 h1:FiJd5l1UOLj0wCgbSE0rwwXHzEdAZS6hiiSnxJN/D60= +go.uber.org/zap v1.24.0/go.mod h1:2kMP+WWQ8aoFoedH3T2sq6iJ2yDWpHbP0f6MQbS9Gkg= +go4.org v0.0.0-20180809161055-417644f6feb5/go.mod h1:MkTOUMDaeVYJUOUsaDXIhWPZYa1yOyC1qaOBpL57BhE= +golang.org/x/build v0.0.0-20190111050920-041ab4dc3f9d/go.mod h1:OWs+y06UdEOHN4y+MfF/py+xQ/tYqIWW03b70/CG9Rw= golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= +golang.org/x/crypto v0.0.0-20181030102418-4d3f4d9ffa16/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= golang.org/x/crypto v0.0.0-20181203042331-505ab145d0a9/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20190313024323-a1f597ede03a/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20190611184440-5c40567a22f8/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20191206172530-e9b2fee46413/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.0.0-20200602180216-279210d13fed/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20200728195943-123391ffb6de/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.0.0-20210322153248-0c34fe9e7dc2/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= golang.org/x/crypto v0.0.0-20210421170649-83a5a9bb288b/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= golang.org/x/crypto v0.0.0-20211108221036-ceb1ce70b4fa/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/crypto v0.7.0 h1:AvwMYaRytfdeVt3u6mLaxYtErKYjxA2OXjJ1HHq6t3A= @@ -652,6 +891,7 @@ golang.org/x/exp v0.0.0-20230321023759-10a507213a29 h1:ooxPy7fPvB4kwsA2h+iBNHkAb golang.org/x/exp v0.0.0-20230321023759-10a507213a29/go.mod h1:CxIveKay+FTh1D0yPZemJVgC/95VzuuOLq5Qi4xnoYc= golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js= golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= +golang.org/x/lint v0.0.0-20180702182130-06c8688daad7/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= @@ -675,16 +915,20 @@ golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.4.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.4.1/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= -golang.org/x/mod v0.8.0 h1:LUYupSeNrTNCGzR/hVBk2NHZO4hXcVaW1k4Qx7rjPx8= -golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= +golang.org/x/mod v0.10.0 h1:lFO9qtOdlre5W1jxS3r/4szv2/6iXxScdzjoBMXNhYk= +golang.org/x/mod v0.10.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181011144130-49bb7cea24b1/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181029044818-c44066c5c816/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181106065722-10aee1819953/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20181201002055-351d144fa1fc/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190313220215-9f648a60d977/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190501004415-9ce7a6920f09/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190503192946-f4e77d36d62c/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= @@ -713,15 +957,15 @@ golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwY golang.org/x/net v0.0.0-20201031054903-ff519b6c9102/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= golang.org/x/net v0.0.0-20201209123823-ac852fbbde11/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= golang.org/x/net v0.0.0-20201224014010-6772e930b67b/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20210119194325-5f4716e94777/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= -golang.org/x/net v0.0.0-20210525063256-abc453219eb5/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20210907225631-ff17edfbf26d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= -golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= -golang.org/x/net v0.0.0-20220225172249-27dd8689420f/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= golang.org/x/net v0.8.0 h1:Zrh2ngAOFYneWTAIAPethzeaQLuHwhuBkuV6ZiRnUaQ= golang.org/x/net v0.8.0/go.mod h1:QVkue5JL9kW//ek3r6jTKnTFis1tRmNAW2P1shuFdJc= golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= +golang.org/x/oauth2 v0.0.0-20181017192945-9dcd33a902f4/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= +golang.org/x/oauth2 v0.0.0-20181203162652-d668ce993890/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= golang.org/x/oauth2 v0.0.0-20191202225959-858c2ad4c8b6/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= @@ -730,8 +974,7 @@ golang.org/x/oauth2 v0.0.0-20200902213428-5d25da1a8d43/go.mod h1:KelEdhl1UZF7XfJ golang.org/x/oauth2 v0.0.0-20201109201403-9fd604954f58/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= golang.org/x/oauth2 v0.0.0-20201208152858-08078c50e5b5/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= golang.org/x/oauth2 v0.0.0-20210218202405-ba52d332ba99/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20210514164344-f6687ab2804c/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20220223155221-ee480838109b/go.mod h1:DAh4E804XQdzx2j+YRIaUnCqCV2RuMz24cGBJ5QYIrc= +golang.org/x/perf v0.0.0-20180704124530-6e6d33e29852/go.mod h1:JLpeXjPJfIyPr5TlbXLkXWLhP8nz10XfvxElABhCtcw= golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= @@ -745,14 +988,19 @@ golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJ golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.1.0 h1:wsuoTGHzEhffawBOhz5CYhcrV4IdKZbEyZjBMuTp12o= golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sys v0.0.0-20180810173357-98c5dad5d1a0/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181029174526-d69651ed3497/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20181205085412-a5c9d58dba9a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190228124157-a34e9553db1e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190316082340-a2f829d7f35f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190405154228-4b34438f7a67/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -772,10 +1020,10 @@ golang.org/x/sys v0.0.0-20191115151921-52ab43148777/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191228213918-04cbcbbfeed8/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200106162015-b016eb3dc98e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200113162924-86b910548bc1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200122134326-e047566fdf82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200124204421-9fbb57f87de9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200202164722-d101bd2416d5/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200212091648-12a6c2dcc1e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -787,8 +1035,7 @@ golang.org/x/sys v0.0.0-20200511232937-7e40ca221e25/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20200515095857-1151b9dac4a9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200519105757-fe76b779f299/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200523222454-059865788121/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200615200032-f1bc736245b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200602225109-6fdc65e7d980/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200814200057-3d37ad5750ed/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200905004654-be1d3432aa8f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -799,11 +1046,11 @@ golang.org/x/sys v0.0.0-20210104204734-6f8348627aad/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210225134936-a50acf3fe073/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210303074136-134d130e1a04/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210423185535-09eb48e85fd7/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210616094352-59db8d763f22/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= @@ -812,17 +1059,16 @@ golang.org/x/sys v0.0.0-20210906170528-6f6e22806c34/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.0.0-20210927094055-39ccf1dd6fa6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211025201205-69cdffdb9359/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211116061358-0a5406a5449c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220114195835-da31bd327af9/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220503163025-988cb79eb6c6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220704084225-05e143d24a9e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220908164124-27713097b956/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20221010170243-090e33056c14/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.6.0 h1:MVltZSvRTcU2ljQOhs94SXPftV6DCNnZViHeQps87pQ= golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.7.0 h1:3jlCCIQZPdOYu1h8BkNvLz8Kgwtae2cagcG/VamtZRU= +golang.org/x/sys v0.7.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= -golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/term v0.6.0 h1:clScbb1cHjoCkyRbWwBEUZ5H/tIFu5TAXIqaZD0Gcjw= golang.org/x/term v0.6.0/go.mod h1:m6U89DPEgQRMq3DNkDClhWw02AUbt2daBVO4cn4Hv9U= golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= @@ -836,10 +1082,14 @@ golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= golang.org/x/text v0.8.0 h1:57P1ETyNKtuIjB4SRd15iJxuhj8Gc416Y78H3qgMh68= golang.org/x/text v0.8.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= +golang.org/x/time v0.0.0-20180412165947-fbb02b2291d2/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= +golang.org/x/tools v0.0.0-20180828015842-6cd1fcedba52/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20181030000716-a0a13e073c7b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20181030221726-6c7e314b6563/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= @@ -889,13 +1139,17 @@ golang.org/x/tools v0.0.0-20210105154028-b0ab187a4818/go.mod h1:emZCQorbCU4vsT4f golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20210108195828-e2f9c7f1fc8e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.1.0/go.mod h1:xkSsbof2nBLbhDlRMhhhyNLN/zl3eTqcnHD5viDpcZ0= +golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= -golang.org/x/tools v0.6.0 h1:BOw41kyTf3PuCW1pVQf8+Cyg8pMlkYB1oo9iJ6D/lKM= -golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU= +golang.org/x/tools v0.7.0 h1:W4OVu8VVOaIO0yzWMNdepAulS7YfoS3Zabrm8DOXXU4= +golang.org/x/tools v0.7.0/go.mod h1:4pg6aUX35JBAogB10C9AtvVL+qowtN4pT3CGSQex14s= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +google.golang.org/api v0.0.0-20180910000450-7ca32eb868bf/go.mod h1:4mhQ8q/RsB7i+udVvVy5NUi08OU8ZlA0gRVgrF7VFY0= +google.golang.org/api v0.0.0-20181030000543-1d582fd0359e/go.mod h1:4mhQ8q/RsB7i+udVvVy5NUi08OU8ZlA0gRVgrF7VFY0= +google.golang.org/api v0.1.0/go.mod h1:UGEZY7KEX120AnNLIHFMKIo4obdJhkp2tPbaPlQx13Y= google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE= google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M= google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= @@ -916,6 +1170,8 @@ google.golang.org/api v0.35.0/go.mod h1:/XrVsuzM0rZmrsbjJutiuftIzeuTQcEeaYcSk/mQ google.golang.org/api v0.36.0/go.mod h1:+z5ficQTmoYpPn8LCUNVpK5I7hwkpjbcgqA7I34qYtE= google.golang.org/api v0.40.0/go.mod h1:fYKFpnQN0DsDSKRVRcQSDQNtqWPfM9i+zNPxepjRCQ8= google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= +google.golang.org/appengine v1.2.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= +google.golang.org/appengine v1.3.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0= @@ -924,6 +1180,9 @@ google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCID google.golang.org/appengine v1.6.7/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= google.golang.org/genproto v0.0.0-20180831171423-11092d34479b/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= +google.golang.org/genproto v0.0.0-20181029155118-b69ba1387ce2/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= +google.golang.org/genproto v0.0.0-20181202183823-bd91e49a0898/go.mod h1:7Ep/1NZk928CDR8SjdVbjWNpdIf6nzjE3BTgJDr2Atg= +google.golang.org/genproto v0.0.0-20190306203927-b5d61aea6440/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= google.golang.org/genproto v0.0.0-20190425155659-357c62f0e4bb/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= @@ -964,6 +1223,9 @@ google.golang.org/genproto v0.0.0-20210226172003-ab064af71705/go.mod h1:FWY/as6D google.golang.org/genproto v0.0.0-20210903162649-d08c68adba83/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= google.golang.org/genproto v0.0.0-20230306155012-7f2fa6fef1f4 h1:DdoeryqhaXp1LtT/emMP1BRJPHHKFi5akj/nbx/zNTA= google.golang.org/genproto v0.0.0-20230306155012-7f2fa6fef1f4/go.mod h1:NWraEVixdDnqcqQ30jipen1STv2r/n24Wb7twVTGR4s= +google.golang.org/grpc v1.14.0/go.mod h1:yo6s7OP7yaDglbqo1J04qKzAhqBH6lvTonzMVmEdcZw= +google.golang.org/grpc v1.16.0/go.mod h1:0JHn/cJsOMiMfNA9+DeHDlAU7KAAB5GDlYFpa9MZMio= +google.golang.org/grpc v1.17.0/go.mod h1:6QZJwpn2B+Zp71q/5VxRsJ6NXXVCE5NRUHRo+f3cWCs= google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= @@ -1006,8 +1268,10 @@ gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8 gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20200902074654-038fdea0a05b/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= +gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= +gopkg.in/inf.v0 v0.9.1/go.mod h1:cWUDdTG/fYaXco+Dcufb5Vnc6Gp2YChqWtbxRZE0mXw= gopkg.in/ini.v1 v1.67.0 h1:Dgnx+6+nfE+IfzjUEISNeydPJh9AXNNsWbGP9KzCsOA= gopkg.in/ini.v1 v1.67.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= @@ -1017,14 +1281,18 @@ gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gotest.tools v2.2.0+incompatible h1:VsBPFP1AI068pPrMxtb/S8Zkgf9xEmTLJjfM+P5UIEo= gotest.tools/v3 v3.4.0 h1:ZazjZUfuVeZGLAmlKKuyv3IKP5orXcwtOwDQH6YVr6o= +grpc.go4.org v0.0.0-20170609214715-11d0a25b4919/go.mod h1:77eQGdRu53HpSqPFJFmuJdjuHRquDANNeA4x7B8WQ9o= +honnef.co/go/tools v0.0.0-20180728063816-88497007e858/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= @@ -1032,7 +1300,10 @@ honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWh honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg= honnef.co/go/tools v0.0.1-2020.1.3/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k= honnef.co/go/tools v0.0.1-2020.1.4/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k= -nhooyr.io/websocket v1.8.6 h1:s+C3xAMLwGmlI31Nyn/eAehUlZPwfYZu2JXM621Q5/k= +lukechampine.com/blake3 v1.1.7 h1:GgRMhmdsuK8+ii6UZFDL8Nb+VyMwadAgcJyfYHxG6n0= +lukechampine.com/blake3 v1.1.7/go.mod h1:tkKEOtDkNtklkXtLNEOGNq5tcV90tJiA1vAA12R78LA= +nhooyr.io/websocket v1.8.7 h1:usjR2uOr/zjjkVMy0lW+PPohFok7PCow5sDjLgX4P4g= +nhooyr.io/websocket v1.8.7/go.mod h1:B70DZP8IakI65RVQ51MsWP/8jndNma26DVA/nFSCgW0= pgregory.net/rapid v0.5.5 h1:jkgx1TjbQPD/feRoK+S/mXw9e1uj6WilpHrXJowi6oA= pgregory.net/rapid v0.5.5/go.mod h1:PY5XlDGj0+V1FCq0o192FdRhpKHGTRIWBgqjDBTrq04= rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8= @@ -1040,3 +1311,5 @@ rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0= rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA= sigs.k8s.io/yaml v1.3.0 h1:a2VclLzOGrwOHDiV8EfBGhvjHvP46CtW5j6POvhYGGo= sigs.k8s.io/yaml v1.3.0/go.mod h1:GeOyir5tyXNByN85N/dRIT9es5UQNerPYEKK56eTBm8= +sourcegraph.com/sourcegraph/go-diff v0.5.0/go.mod h1:kuch7UrkMzY0X+p9CRK03kfuPQ2zzQcaEFbx8wA8rck= +sourcegraph.com/sqs/pbtypes v0.0.0-20180604144634-d3ebe8f20ae4/go.mod h1:ketZ/q3QxT9HOBeFhu6RdvsftgpsbFHBF5Cas6cDKZ0= diff --git a/signer/config.go b/signer/config.go index 482e874f..fca7737f 100644 --- a/signer/config.go +++ b/signer/config.go @@ -76,7 +76,7 @@ func (c *Config) ValidateThresholdModeConfig() error { return errors.Join(errs...) } - numShards := len(c.ThresholdModeConfig.Cosigners) + numShards := uint8(len(c.ThresholdModeConfig.Cosigners)) if c.ThresholdModeConfig.Threshold <= numShards/2 { errs = append(errs, fmt.Errorf("threshold (%d) must be greater than number of shards (%d) / 2", @@ -186,7 +186,7 @@ func (c RuntimeConfig) KeyFileExistsCosignerRSA() (string, error) { // ThresholdModeConfig is the on disk config format for threshold sign mode. type ThresholdModeConfig struct { - Threshold int `yaml:"threshold"` + Threshold uint8 `yaml:"threshold"` Cosigners CosignersConfig `yaml:"cosigners"` GRPCTimeout string `yaml:"grpcTimeout"` RaftTimeout string `yaml:"raftTimeout"` @@ -202,12 +202,64 @@ func (cfg *ThresholdModeConfig) LeaderElectMultiAddress() (string, error) { // CosignerConfig is the on disk format representing a cosigner for threshold sign mode. type CosignerConfig struct { - ShardID int `yaml:"shardID"` + ShardID uint8 `yaml:"shardID"` P2PAddr string `yaml:"p2pAddr"` + DKGID string `yaml:"dkgID,omitempty"` +} + +func (c CosignerConfig) HostPort() (string, string, error) { + url, err := url.Parse(c.P2PAddr) + if err != nil { + return "", "", fmt.Errorf("failed to parse cosigner (shard ID: %d) p2p address: %w", c.ShardID, err) + } + + host, port, err := net.SplitHostPort(url.Host) + if err != nil { + return "", "", fmt.Errorf("failed to parse cosigner (shard ID: %d) host port: %w", c.ShardID, err) + } + + return host, port, nil +} + +func (c CosignerConfig) LibP2PHostAddr() (string, error) { + _, port, err := c.HostPort() + if err != nil { + return "", err + } + + return fmt.Sprintf("/ip4/0.0.0.0/tcp/%s", port), nil +} + +func (c CosignerConfig) LibP2PAddr() (string, error) { + host, port, err := c.HostPort() + if err != nil { + return "", err + } + + return fmt.Sprintf("/ip4/%s/tcp/%s", host, port), nil } type CosignersConfig []CosignerConfig +func (cosigners CosignersConfig) MyCosigner(myID uint8) (CosignerConfig, error) { + for _, c := range cosigners { + if c.ShardID == myID { + return c, nil + } + } + return CosignerConfig{}, fmt.Errorf("cosigner with that id: %d cannot be found", myID) +} + +func (cosigners CosignersConfig) OtherCosigners(myID uint8) CosignersConfig { + others := make(CosignersConfig, 0, len(cosigners)-1) + for _, c := range cosigners { + if c.ShardID != myID { + others = append(others, c) + } + } + return others +} + func (cosigners CosignersConfig) Validate() error { var errs []error // Check IDs to make sure none are duplicated @@ -219,22 +271,16 @@ func (cosigners CosignersConfig) Validate() error { // Make sure that the cosigner IDs match the number of cosigners. for _, cosigner := range cosigners { - if cosigner.ShardID < 1 || cosigner.ShardID > shards { + if cosigner.ShardID < 1 || cosigner.ShardID > uint8(shards) { errs = append(errs, fmt.Errorf("cosigner shard ID %d in args is out of range, must be between 1 and %d, inclusive", cosigner.ShardID, shards)) } - url, err := url.Parse(cosigner.P2PAddr) + host, _, err := cosigner.HostPort() if err != nil { - errs = append(errs, fmt.Errorf("failed to parse cosigner (shard ID: %d) p2p address: %w", cosigner.ShardID, err)) - continue + errs = append(errs, err) } - host, _, err := net.SplitHostPort(url.Host) - if err != nil { - errs = append(errs, fmt.Errorf("failed to parse cosigner (shard ID: %d) host port: %w", cosigner.ShardID, err)) - continue - } if host == "0.0.0.0" { errs = append(errs, fmt.Errorf("host cannot be 0.0.0.0, must be reachable from other cosigners")) } @@ -249,8 +295,8 @@ func (cosigners CosignersConfig) Validate() error { return errors.Join(errs...) } -func duplicateCosigners(cosigners []CosignerConfig) (duplicates map[int][]string) { - idAddrs := make(map[int][]string) +func duplicateCosigners(cosigners []CosignerConfig) (duplicates map[uint8][]string) { + idAddrs := make(map[uint8][]string) for _, cosigner := range cosigners { // Collect all addresses assigned to each cosigner. idAddrs[cosigner.ShardID] = append(idAddrs[cosigner.ShardID], cosigner.P2PAddr) @@ -275,7 +321,7 @@ func duplicateCosigners(cosigners []CosignerConfig) (duplicates map[int][]string func CosignersFromFlag(cosigners []string) (out []CosignerConfig, err error) { var errs []error for i, c := range cosigners { - out = append(out, CosignerConfig{ShardID: i + 1, P2PAddr: c}) + out = append(out, CosignerConfig{ShardID: uint8(i) + 1, P2PAddr: c}) } if len(errs) > 0 { return nil, errors.Join(errs...) diff --git a/signer/cosigner.go b/signer/cosigner.go index 83ff34f1..19039b2e 100644 --- a/signer/cosigner.go +++ b/signer/cosigner.go @@ -52,8 +52,8 @@ type CosignerSignResponse struct { } type CosignerEphemeralSecretPart struct { - SourceID int - DestinationID int + SourceID uint8 + DestinationID uint8 SourceEphemeralSecretPublicKey []byte EncryptedSharePart []byte SourceSig []byte @@ -80,8 +80,8 @@ func (secretParts CosignerEphemeralSecretParts) toProto() (out []*proto.Ephemera func CosignerEphemeralSecretPartFromProto(secretPart *proto.EphemeralSecretPart) CosignerEphemeralSecretPart { return CosignerEphemeralSecretPart{ - SourceID: int(secretPart.SourceID), - DestinationID: int(secretPart.DestinationID), + SourceID: uint8(secretPart.SourceID), + DestinationID: uint8(secretPart.DestinationID), SourceEphemeralSecretPublicKey: secretPart.SourceEphemeralSecretPublicKey, EncryptedSharePart: secretPart.EncryptedSharePart, SourceSig: secretPart.SourceSig, @@ -98,7 +98,7 @@ func CosignerEphemeralSecretPartsFromProto( type CosignerSetEphemeralSecretPartRequest struct { ChainID string - SourceID int + SourceID uint8 SourceEphemeralSecretPublicKey []byte EncryptedSharePart []byte SourceSig []byte @@ -133,7 +133,7 @@ type CosignerSetEphemeralSecretPartsAndSignRequest struct { type Cosigner interface { // Get the ID of the cosigner // The ID is the shamir index: 1, 2, etc... - GetID() int + GetID() uint8 // Get the P2P URL (GRPC and Raft) GetAddress() string diff --git a/signer/cosigner_key.go b/signer/cosigner_key.go index 76cf1ea5..9fc784be 100644 --- a/signer/cosigner_key.go +++ b/signer/cosigner_key.go @@ -17,7 +17,7 @@ import ( type CosignerEd25519Key struct { PubKey cometcrypto.PubKey `json:"pubKey"` PrivateShard []byte `json:"privateShard"` - ID int `json:"id"` + ID uint8 `json:"id"` } func (key *CosignerEd25519Key) MarshalJSON() ([]byte, error) { @@ -105,7 +105,7 @@ func LoadCosignerEd25519Key(file string) (CosignerEd25519Key, error) { // CosignerEd25519Key is a single RSA key for an m-of-n threshold signer. type CosignerRSAKey struct { RSAKey rsa.PrivateKey `json:"rsaKey"` - ID int `json:"id"` + ID uint8 `json:"id"` RSAPubs []*rsa.PublicKey `json:"rsaPubs"` } diff --git a/signer/cosigner_key_shares.go b/signer/cosigner_key_shares.go index 66500e81..138bc75b 100644 --- a/signer/cosigner_key_shares.go +++ b/signer/cosigner_key_shares.go @@ -27,7 +27,7 @@ func CreateCosignerEd25519Shards(pv privval.FilePVKey, threshold, shards uint8) out = append(out, CosignerEd25519Key{ PubKey: pv.PubKey, PrivateShard: shard, - ID: i + 1, + ID: uint8(i) + 1, }) } return out, nil @@ -41,7 +41,7 @@ func CreateCosignerRSAShards(shards int) (out []CosignerRSAKey, err error) { } for i, key := range rsaKeys { out = append(out, CosignerRSAKey{ - ID: i + 1, + ID: uint8(i) + 1, RSAKey: *key, RSAPubs: pubKeys, }) diff --git a/signer/dkg.go b/signer/dkg.go new file mode 100644 index 00000000..ffcde512 --- /dev/null +++ b/signer/dkg.go @@ -0,0 +1,315 @@ +package signer + +import ( + "context" + "encoding/json" + "fmt" + "sync" + "time" + + "github.com/avast/retry-go/v4" + cometcryptoed25519 "github.com/cometbft/cometbft/crypto/ed25519" + "github.com/libp2p/go-libp2p" + pubsub "github.com/libp2p/go-libp2p-pubsub" + "github.com/libp2p/go-libp2p/core/crypto" + "github.com/libp2p/go-libp2p/core/host" + "github.com/libp2p/go-libp2p/core/peer" + "github.com/strangelove-ventures/horcrux/signer/keygen" + "golang.org/x/sync/errgroup" +) + +var ( + topicDKG = "dkg" +) + +type RoundMessages struct { + Round uint8 `json:"round"` + ID uint8 `json:"id"` + Messages [][]byte `json:"messages"` +} + +func NetworkDKG( + ctx context.Context, + cosigners CosignersConfig, + id uint8, + p2pPrivateKey crypto.PrivKey, + threshold uint8, +) (*CosignerEd25519Key, error) { + cosigner, err := cosigners.MyCosigner(id) + if err != nil { + return nil, err + } + + hostAddr, err := cosigner.LibP2PHostAddr() + if err != nil { + return nil, err + } + + h, err := libp2p.New(libp2p.ListenAddrStrings(hostAddr), libp2p.Identity(p2pPrivateKey)) + if err != nil { + return nil, fmt.Errorf("failed to construct libp2p node: %w", err) + } + + defer h.Close() + + fmt.Println("Starting cosigner discovery") + + if err := waitForAllCosigners(ctx, h, cosigners.OtherCosigners(id)); err != nil { + return nil, err + } + + fmt.Println("Cosigner discovery complete") + + ps, err := pubsub.NewGossipSub(ctx, h) + if err != nil { + return nil, fmt.Errorf("failed to setup libp2p pub sub: %w", err) + } + + total := len(cosigners) + + keygenCosigner, err := keygen.NewCosigner(id, threshold, uint8(total)) + if err != nil { + return nil, err + } + + round1, err := keygenCosigner.Round1() + if err != nil { + return nil, err + } + + round1Msgs := RoundMessages{ + Round: 1, + ID: id, + Messages: round1, + } + + round1MsgsBz, err := json.Marshal(round1Msgs) + if err != nil { + return nil, err + } + + dkgTopic, err := ps.Join(topicDKG) + if err != nil { + return nil, fmt.Errorf("failed to join topic: %w", err) + } + + sub, err := dkgTopic.Subscribe() + if err != nil { + return nil, fmt.Errorf("failed to subscribe to topic: %w", err) + } + + var eg errgroup.Group + eg.Go(func() error { + return processRounds(ctx, dkgTopic, sub, keygenCosigner, uint8(total), round1[0]) + }) + + if err := dkgTopic.Publish(ctx, round1MsgsBz); err != nil { + return nil, err + } + + if err := eg.Wait(); err != nil { + return nil, err + } + + return &CosignerEd25519Key{ + PubKey: cometcryptoed25519.PubKey(keygenCosigner.Public().GroupKey.ToEd25519()), + PrivateShard: keygenCosigner.Secret().Secret.Bytes(), + ID: id, + }, nil +} + +func processRounds( + ctx context.Context, + dkgTopic *pubsub.Topic, + sub *pubsub.Subscription, + keygenCosigner keygen.Cosigner, + total uint8, + myRound1Msg []byte, +) error { + id := uint8(keygenCosigner.ID) + + allRound1Msgs := make([][]byte, total) + + allRound1Msgs[id-1] = myRound1Msg + + allRound2Msgs := make([][]byte, total*(total-1)) + + allRound3Msgs := make([][]byte, total) + +RecvRoundMsgsLoop: + for { + m, err := sub.Next(ctx) + if err != nil { + fmt.Printf("Failed to retrieve next message from topic subscription: %v\n", err) + continue RecvRoundMsgsLoop + } + fmt.Println(m.ReceivedFrom, ": ", string(m.Message.Data)) + + var cosignerRoundMsgs RoundMessages + if err := json.Unmarshal(m.Message.Data, &cosignerRoundMsgs); err != nil { + fmt.Printf("Failed to unmarshal message from topic subscription: %v\n", err) + continue RecvRoundMsgsLoop + } + cID := cosignerRoundMsgs.ID + + switch cosignerRoundMsgs.Round { + case 1: + allRound1Msgs[cID-1] = cosignerRoundMsgs.Messages[0] + + for _, m := range allRound1Msgs { + if len(m) == 0 { + continue RecvRoundMsgsLoop + } + } + + // have messages from all cosigners for round 1 + round2, err := keygenCosigner.Round2(allRound1Msgs) + if err != nil { + return err + } + + round2Msgs := RoundMessages{ + Round: 2, + ID: id, + Messages: round2, + } + + round2MsgsBz, err := json.Marshal(round2Msgs) + if err != nil { + return err + } + + copy(allRound2Msgs[(id-1)*(total-1):id*(total-1)], round2) + + if err := dkgTopic.Publish(ctx, round2MsgsBz); err != nil { + fmt.Printf("Failed to publish round 2 messages: %v\n", err) + } + + case 2: + copy(allRound2Msgs[(cID-1)*(total-1):cID*(total-1)], cosignerRoundMsgs.Messages) + + for _, m := range allRound2Msgs { + if len(m) == 0 { + continue RecvRoundMsgsLoop + } + } + + // have messages from all cosigners for round 2 + if err := keygenCosigner.Round3(allRound2Msgs); err != nil { + return err + } + + if err := keygenCosigner.WaitForCompletion(); err != nil { + return err + } + + round3Msgs := RoundMessages{ + Round: 3, + ID: id, + } + + round3MsgsBz, err := json.Marshal(round3Msgs) + if err != nil { + return err + } + + if err := dkgTopic.Publish(ctx, round3MsgsBz); err != nil { + fmt.Printf("Failed to publish round 3 result: %v\n", err) + } + + case 3: + allRound3Msgs[cID-1] = []byte{0x01} + for _, m := range allRound3Msgs { + if len(m) == 0 { + continue RecvRoundMsgsLoop + } + } + + // have success from all cosigners, sharding complete + return nil + default: + fmt.Printf("Unexpected round: %d\n", cosignerRoundMsgs.Round) + } + } +} + +func waitForAllCosigners(ctx context.Context, h host.Host, cosigners CosignersConfig) error { + var wg sync.WaitGroup + for _, c := range cosigners { + peerAddr, err := c.LibP2PAddr() + if err != nil { + return err + } + + peerinfo, err := peer.AddrInfoFromString(peerAddr + "/p2p/" + c.DKGID) + if err != nil { + return err + } + + wg.Add(1) + go func() { + defer wg.Done() + _ = retry.Do( + func() error { + return h.Connect(ctx, *peerinfo) + }, + retry.OnRetry(func(n uint, err error) { + fmt.Printf("Attempt %d: Failed to connect to cosigner %s: %v\n", n, peerinfo.String(), err) + }), + retry.DelayType(retry.BackOffDelay), + retry.Attempts(0), + retry.Delay(time.Second), + ) + + fmt.Printf("Connection established with bootstrap node: %s\n", *peerinfo) + }() + } + wg.Wait() + return nil +} + +// LocalDKG simulates a DKG key combination ceremony. TEST USE ONLY. +func LocalDKG(threshold, total uint8) (map[uint8]keygen.Cosigner, error) { + cosigners := make(map[uint8]keygen.Cosigner) + + var err error + for i := uint8(1); i <= total; i++ { + cosigners[i], err = keygen.NewCosigner(i, threshold, total) + if err != nil { + return nil, err + } + } + + msgsOut1 := make([][]byte, 0, total) + + for _, c := range cosigners { + msgs1, err := c.Round1() + if err != nil { + return nil, err + } + + msgsOut1 = append(msgsOut1, msgs1...) + } + + msgsOut2 := make([][]byte, 0, total*(total-1)/2) + + for _, c := range cosigners { + msgs2, err := c.Round2(msgsOut1) + if err != nil { + return nil, err + } + + msgsOut2 = append(msgsOut2, msgs2...) + } + + for _, c := range cosigners { + if err := c.Round3(msgsOut2); err != nil { + return nil, err + } + if err := c.WaitForCompletion(); err != nil { + return nil, err + } + } + + return cosigners, nil +} diff --git a/signer/dkg_test.go b/signer/dkg_test.go new file mode 100644 index 00000000..7a838caa --- /dev/null +++ b/signer/dkg_test.go @@ -0,0 +1,73 @@ +package signer + +import ( + "context" + "crypto/rand" + "testing" + + "github.com/libp2p/go-libp2p/core/crypto" + "github.com/libp2p/go-libp2p/core/peer" + "github.com/stretchr/testify/require" + tsed25519 "gitlab.com/unit410/threshold-ed25519/pkg" + "golang.org/x/sync/errgroup" +) + +func TestNetworkDKG(t *testing.T) { + ctx := context.Background() + + cosigners := []CosignerConfig{ + { + ShardID: 1, + P2PAddr: "tcp://127.0.0.1:8040", + }, + { + ShardID: 2, + P2PAddr: "tcp://127.0.0.1:8041", + }, + { + ShardID: 3, + P2PAddr: "tcp://127.0.0.1:8042", + }, + } + + cosignerPrivKeys := make([]crypto.PrivKey, len(cosigners)) + + for i := range cosigners { + privKey, _, err := crypto.GenerateEd25519Key(rand.Reader) + require.NoError(t, err) + + p2pID, err := peer.IDFromPrivateKey(privKey) + require.NoError(t, err) + + cosigners[i].DKGID = p2pID.String() + + cosignerPrivKeys[i] = privKey + } + + const threshold = 2 + + var eg errgroup.Group + + shards := make([]*CosignerEd25519Key, len(cosigners)) + + for i, c := range cosigners { + id := c.ShardID + i := i + eg.Go(func() (err error) { + shards[i], err = NetworkDKG(ctx, cosigners, id, cosignerPrivKeys[i], threshold) + return err + }) + } + + require.NoError(t, eg.Wait()) + + pubKey := shards[0].PubKey + + privShards := make([]tsed25519.Scalar, len(cosigners)) + + for i, shard := range shards { + privShards[i] = shard.PrivateShard + } + + testThresholdValidatorWithShards(t, threshold, uint8(len(cosigners)), pubKey, privShards) +} diff --git a/signer/keygen/cosigner.go b/signer/keygen/cosigner.go index 674d514e..b16d3554 100644 --- a/signer/keygen/cosigner.go +++ b/signer/keygen/cosigner.go @@ -12,7 +12,7 @@ import ( ) // Cosigner corresponds to a horcrux cosigner, but in this case strictly for the purposes of -// DKG key combination (sharding without a trusted "dealer"). +// distributed key generation (DKG) (sharding without a trusted "dealer"). type Cosigner struct { ID party.ID allCosigners party.IDSlice diff --git a/signer/keygen/dkg.go b/signer/keygen/dkg.go deleted file mode 100644 index 9f238668..00000000 --- a/signer/keygen/dkg.go +++ /dev/null @@ -1,47 +0,0 @@ -package keygen - -// LocalDKG simulates a DKG key combination ceremony. TEST USE ONLY. -func LocalDKG(threshold, total uint8) (map[uint8]Cosigner, error) { - cosigners := make(map[uint8]Cosigner) - - var err error - for i := uint8(1); i <= total; i++ { - cosigners[i], err = NewCosigner(i, threshold, total) - if err != nil { - return nil, err - } - } - - msgsOut1 := make([][]byte, 0, total) - - for _, c := range cosigners { - msgs1, err := c.Round1() - if err != nil { - return nil, err - } - - msgsOut1 = append(msgsOut1, msgs1...) - } - - msgsOut2 := make([][]byte, 0, total*(total-1)/2) - - for _, c := range cosigners { - msgs2, err := c.Round2(msgsOut1) - if err != nil { - return nil, err - } - - msgsOut2 = append(msgsOut2, msgs2...) - } - - for _, c := range cosigners { - if err := c.Round3(msgsOut2); err != nil { - return nil, err - } - if err := c.WaitForCompletion(); err != nil { - return nil, err - } - } - - return cosigners, nil -} diff --git a/signer/keygen/keygen_test.go b/signer/keygen_test.go similarity index 94% rename from signer/keygen/keygen_test.go rename to signer/keygen_test.go index 4ff92b73..4868f6ad 100644 --- a/signer/keygen/keygen_test.go +++ b/signer/keygen_test.go @@ -1,10 +1,10 @@ -package keygen_test +package signer_test import ( "errors" "testing" - "github.com/strangelove-ventures/horcrux/signer/keygen" + "github.com/strangelove-ventures/horcrux/signer" "github.com/stretchr/testify/require" "github.com/taurusgroup/frost-ed25519/pkg/eddsa" "github.com/taurusgroup/frost-ed25519/pkg/frost/party" @@ -17,7 +17,7 @@ func TestKeygenCosigner(t *testing.T) { total = 3 ) - cosigners, err := keygen.LocalDKG(threshold, total) + cosigners, err := signer.LocalDKG(threshold, total) require.NoError(t, err) var pubCmp *eddsa.Public diff --git a/signer/local_cosigner.go b/signer/local_cosigner.go index 25160857..4670e869 100644 --- a/signer/local_cosigner.go +++ b/signer/local_cosigner.go @@ -75,13 +75,13 @@ func (hrst *HRSTKey) Less(other HRSTKey) bool { } type CosignerRSAPubKey struct { - ID int + ID uint8 PublicKey rsa.PublicKey } type CosignerGetEphemeralSecretPartRequest struct { ChainID string - ID int + ID uint8 Height int64 Round int64 Step int8 @@ -97,7 +97,7 @@ type LocalCosigner struct { key CosignerRSAKey threshold uint8 chainState sync.Map - rsaPubKeys map[int]CosignerRSAPubKey + rsaPubKeys map[uint8]CosignerRSAPubKey address string pendingDiskWG sync.WaitGroup } @@ -141,7 +141,7 @@ func NewLocalCosigner( cosigner := &LocalCosigner{ config: config, key: key, - rsaPubKeys: make(map[int]CosignerRSAPubKey), + rsaPubKeys: make(map[uint8]CosignerRSAPubKey), threshold: threshold, address: address, } @@ -155,7 +155,7 @@ func NewLocalCosigner( // GetID returns the id of the cosigner // Implements Cosigner interface -func (cosigner *LocalCosigner) GetID() int { +func (cosigner *LocalCosigner) GetID() uint8 { return cosigner.key.ID } diff --git a/signer/remote_cosigner.go b/signer/remote_cosigner.go index aaa04b05..d37b7deb 100644 --- a/signer/remote_cosigner.go +++ b/signer/remote_cosigner.go @@ -14,12 +14,12 @@ import ( // RemoteCosigner uses CosignerGRPC to request signing from a remote cosigner type RemoteCosigner struct { - id int + id uint8 address string } // NewRemoteCosigner returns a newly initialized RemoteCosigner -func NewRemoteCosigner(id int, address string) *RemoteCosigner { +func NewRemoteCosigner(id uint8, address string) *RemoteCosigner { cosigner := &RemoteCosigner{ id: id, @@ -38,7 +38,7 @@ func getContext() (context.Context, context.CancelFunc) { // GetID returns the ID of the remote cosigner // Implements the cosigner interface -func (cosigner *RemoteCosigner) GetID() int { +func (cosigner *RemoteCosigner) GetID() uint8 { return cosigner.id } diff --git a/signer/threshold_signer.go b/signer/threshold_signer.go index 55d7a72d..01d356c3 100644 --- a/signer/threshold_signer.go +++ b/signer/threshold_signer.go @@ -16,14 +16,14 @@ type ThresholdSigner interface { DealShares(req CosignerGetEphemeralSecretPartRequest) (HrsMetadata, error) GetEphemeralSecretPart(req CosignerGetEphemeralSecretPartRequest, m *LastSignStateWrapper, - pubKeys map[int]CosignerRSAPubKey) (CosignerEphemeralSecretPart, error) + pubKeys map[uint8]CosignerRSAPubKey) (CosignerEphemeralSecretPart, error) SetEphemeralSecretPart(req CosignerSetEphemeralSecretPartRequest, m *LastSignStateWrapper, - pubKeys map[int]CosignerRSAPubKey) error + pubKeys map[uint8]CosignerRSAPubKey) error Sign(req CosignerSignRequest, m *LastSignStateWrapper) (CosignerSignResponse, error) - GetID() (int, error) + GetID() (uint8, error) // Stop performs any cleanup work, such as flushing state files to disk, then shut down. Stop() diff --git a/signer/threshold_signer_soft.go b/signer/threshold_signer_soft.go index a4f09139..c38e2c84 100644 --- a/signer/threshold_signer_soft.go +++ b/signer/threshold_signer_soft.go @@ -65,7 +65,7 @@ func (softSigner *ThresholdSignerSoft) Type() string { } // Implements ThresholdSigner -func (softSigner *ThresholdSignerSoft) GetID() (int, error) { +func (softSigner *ThresholdSignerSoft) GetID() (uint8, error) { return softSigner.key.ID, nil } @@ -201,7 +201,7 @@ func (softSigner *ThresholdSignerSoft) DealShares( // The ephemeral secret part is encrypted for the receiver // Implements ThresholdSigner func (softSigner *ThresholdSignerSoft) GetEphemeralSecretPart( - req CosignerGetEphemeralSecretPartRequest, m *LastSignStateWrapper, pubKeys map[int]CosignerRSAPubKey) ( + req CosignerGetEphemeralSecretPartRequest, m *LastSignStateWrapper, pubKeys map[uint8]CosignerRSAPubKey) ( CosignerEphemeralSecretPart, error) { res := CosignerEphemeralSecretPart{} @@ -286,7 +286,7 @@ func (softSigner *ThresholdSignerSoft) GetEphemeralSecretPart( // Store an ephemeral secret share part provided by another cosigner (signer) // Implements ThresholdSigner func (softSigner *ThresholdSignerSoft) SetEphemeralSecretPart( - req CosignerSetEphemeralSecretPartRequest, m *LastSignStateWrapper, pubKeys map[int]CosignerRSAPubKey) error { + req CosignerSetEphemeralSecretPartRequest, m *LastSignStateWrapper, pubKeys map[uint8]CosignerRSAPubKey) error { // Verify the source signature if req.SourceSig == nil { diff --git a/signer/threshold_validator.go b/signer/threshold_validator.go index d0e9674b..45edc771 100644 --- a/signer/threshold_validator.go +++ b/signer/threshold_validator.go @@ -22,7 +22,7 @@ var _ PrivValidator = &ThresholdValidator{} type ThresholdValidator struct { config *RuntimeConfig - threshold int + threshold uint8 grpcTimeout time.Duration @@ -56,7 +56,7 @@ type ChainSignState struct { func NewThresholdValidator( logger log.Logger, config *RuntimeConfig, - threshold int, + threshold uint8, grpcTimeout time.Duration, myCosigner Cosigner, peerCosigners []Cosigner, @@ -245,7 +245,7 @@ func (pv *ThresholdValidator) waitForPeerEphemeralShares( // Check so that getEphemeralWaitGroup.Done is not called more than (threshold - 1) times which causes hardlock thresholdPeersMutex.Lock() - if len(encryptedEphemeralSharesThresholdMap) < pv.threshold-1 { + if len(encryptedEphemeralSharesThresholdMap) < int(pv.threshold)-1 { (encryptedEphemeralSharesThresholdMap)[peer] = ephemeralSecretParts.EncryptedSecrets defer wg.Done() } @@ -254,7 +254,7 @@ func (pv *ThresholdValidator) waitForPeerEphemeralShares( func (pv *ThresholdValidator) waitForPeerSetEphemeralSharesAndSign( chainID string, - ourID int, + ourID uint8, peer Cosigner, hrst HRSTKey, encryptedEphemeralSharesThresholdMap map[Cosigner][]CosignerEphemeralSecretPart, @@ -502,7 +502,7 @@ func (pv *ThresholdValidator) SignBlock(chainID string, block *Block) ([]byte, t getEphemeralWaitGroup := sync.WaitGroup{} // Only wait until we have threshold sigs - getEphemeralWaitGroup.Add(pv.threshold - 1) + getEphemeralWaitGroup.Add(int(pv.threshold) - 1) // Used to track how close we are to threshold ourID := pv.myCosigner.GetID() @@ -543,7 +543,7 @@ func (pv *ThresholdValidator) SignBlock(chainID string, block *Block) ([]byte, t setEphemeralAndSignWaitGroup := sync.WaitGroup{} // Only wait until we have threshold sigs - setEphemeralAndSignWaitGroup.Add(pv.threshold) + setEphemeralAndSignWaitGroup.Add(int(pv.threshold)) // destination for share signatures shareSignatures := make([][]byte, total) @@ -588,7 +588,7 @@ func (pv *ThresholdValidator) SignBlock(chainID string, block *Block) ([]byte, t shareSigs = append(shareSigs, shareSig) } - if len(sigIds) < pv.threshold { + if len(sigIds) < int(pv.threshold) { totalInsufficientCosigners.Inc() return nil, stamp, errors.New("not enough co-signers") } diff --git a/signer/threshold_validator_test.go b/signer/threshold_validator_test.go index 65729c5f..b6cfb6d4 100644 --- a/signer/threshold_validator_test.go +++ b/signer/threshold_validator_test.go @@ -18,7 +18,6 @@ import ( cometrand "github.com/cometbft/cometbft/libs/rand" cometproto "github.com/cometbft/cometbft/proto/tendermint/types" comet "github.com/cometbft/cometbft/types" - "github.com/strangelove-ventures/horcrux/signer/keygen" "github.com/stretchr/testify/require" tsed25519 "gitlab.com/unit410/threshold-ed25519/pkg" ) @@ -76,7 +75,7 @@ func loadKeyForLocalCosigner( } func testThresholdValidatorDKG(t *testing.T, threshold, total uint8) { - cosigners, err := keygen.LocalDKG(threshold, total) + cosigners, err := LocalDKG(threshold, total) require.NoError(t, err) privShards := make([]tsed25519.Scalar, total) @@ -116,7 +115,7 @@ func testThresholdValidatorWithShards( rsaKeys[i] = rsaKey pubKeys[i] = CosignerRSAPubKey{ - ID: int(i) + 1, + ID: i + 1, PublicKey: rsaKey.PublicKey, } } @@ -155,7 +154,7 @@ func testThresholdValidatorWithShards( thresholdCosigners := make([]Cosigner, 0, threshold-1) for i, cosigner := range cosigners { - require.Equal(t, i+1, cosigner.GetID()) + require.Equal(t, uint8(i)+1, cosigner.GetID()) if i != 0 && len(thresholdCosigners) != int(threshold)-1 { thresholdCosigners = append(thresholdCosigners, cosigner) @@ -167,7 +166,7 @@ func testThresholdValidatorWithShards( validator := NewThresholdValidator( cometlog.NewTMLogger(cometlog.NewSyncWriter(os.Stdout)).With("module", "validator"), cosigners[0].config, - int(threshold), + threshold, time.Second, cosigners[0], thresholdCosigners, @@ -250,7 +249,7 @@ func testThresholdValidatorWithShards( newValidator := NewThresholdValidator( cometlog.NewTMLogger(cometlog.NewSyncWriter(os.Stdout)).With("module", "validator"), cosigners[0].config, - int(threshold), + threshold, time.Second, cosigners[0], thresholdCosigners, From a9743120cdbf3f39b867cab5683394c717da4858 Mon Sep 17 00:00:00 2001 From: Andrew Gouin Date: Fri, 19 May 2023 00:57:18 -0600 Subject: [PATCH 4/6] switch to using existing rsa keys for libp2p --- cmd/horcrux/cmd/dkg.go | 92 ++----------------- signer/config.go | 1 - signer/cosigner_key_test.go | 2 +- signer/dkg.go | 165 +++++++++++++++++++++++----------- signer/dkg_test.go | 141 +++++++++++++++++++++++++---- signer/keygen_test.go | 97 -------------------- signer/local_cosigner_test.go | 6 +- 7 files changed, 248 insertions(+), 256 deletions(-) delete mode 100644 signer/keygen_test.go diff --git a/cmd/horcrux/cmd/dkg.go b/cmd/horcrux/cmd/dkg.go index 3f2050dd..da64f010 100644 --- a/cmd/horcrux/cmd/dkg.go +++ b/cmd/horcrux/cmd/dkg.go @@ -1,100 +1,23 @@ package cmd import ( - "crypto/rand" "encoding/json" "errors" "fmt" "os" "path/filepath" - "github.com/libp2p/go-libp2p/core/crypto" - "github.com/libp2p/go-libp2p/core/peer" "github.com/spf13/cobra" "github.com/strangelove-ventures/horcrux/signer" ) const flagID = "id" -func dkgCmd() *cobra.Command { - cmd := &cobra.Command{ - Use: "dkg", - Short: "Commands for DKG trustless key generation", - } - - cmd.AddCommand(dkgInitCmd()) - cmd.AddCommand(dkgRunCmd()) - - return cmd -} - -func dkgInitCmd() *cobra.Command { - cmd := &cobra.Command{ - Use: "init", - Args: cobra.NoArgs, - Short: `Command to initialize peer private key for libp2p communication during DKG`, - RunE: func(cmd *cobra.Command, args []string) (err error) { - flags := cmd.Flags() - - id, _ := flags.GetUint8(flagID) - - // This ed25519 key is used for libp2p communication during DKG. - // i.e. this key is not used for consensus. - privKey, _, err := crypto.GenerateEd25519Key(rand.Reader) - if err != nil { - return err - } - - rawKey, err := privKey.Raw() - if err != nil { - return err - } - - filename := filepath.Join(config.HomeDir, "libp2p.key") - - if err := os.WriteFile(filename, rawKey, 0600); err != nil { - return err - } - - p2pID, err := peer.IDFromPrivateKey(privKey) - if err != nil { - return err - } - - for i, cosigner := range config.Config.ThresholdModeConfig.Cosigners { - if cosigner.ShardID == id { - config.Config.ThresholdModeConfig.Cosigners[i].DKGID = p2pID.String() - } else if cosigner.DKGID == "" { - config.Config.ThresholdModeConfig.Cosigners[i].DKGID = "REPLACE ME" - } - } - - if err := config.WriteConfigFile(); err != nil { - return err - } - - fmt.Fprintf( - cmd.OutOrStdout(), - "libp2p key generated for DKG. `dkgID` to share with other members:\n%s\n", - p2pID.String(), - ) - - return nil - }, - } - - f := cmd.Flags() - f.Uint8(flagID, 0, "cosigner shard ID as participant in DKG ceremony") - _ = cmd.MarkFlagRequired(flagID) - - return cmd -} - // dkgCmd is a cobra command for performing // a DKG key ceremony as a participating cosigner. -func dkgRunCmd() *cobra.Command { +func dkgCmd() *cobra.Command { cmd := &cobra.Command{ - Use: "run", + Use: "dkg", Args: cobra.NoArgs, Short: `Perform DKG key sharding ceremony (no trusted "dealer")`, RunE: func(cmd *cobra.Command, args []string) (err error) { @@ -150,21 +73,20 @@ func dkgRunCmd() *cobra.Command { } } - libp2pKeyFile := filepath.Join(config.HomeDir, "libp2p.key") - p2pKeyBz, err := os.ReadFile(libp2pKeyFile) + keyFile, err := config.KeyFileExistsCosignerRSA() if err != nil { - return fmt.Errorf("failed to read libp2p.key. Did you run horcrux dkg init first?: %w", err) + return err } - p2pKey, err := crypto.UnmarshalEd25519PrivateKey(p2pKeyBz) + key, err := signer.LoadCosignerRSAKey(keyFile) if err != nil { - return err + return fmt.Errorf("error reading cosigner key (%s): %w", keyFile, err) } // silence usage after all input has been validated cmd.SilenceUsage = true - shard, err := signer.NetworkDKG(cmd.Context(), config.Config.ThresholdModeConfig.Cosigners, id, p2pKey, threshold) + shard, err := signer.NetworkDKG(cmd.Context(), config.Config.ThresholdModeConfig.Cosigners, id, key, threshold) if err != nil { return err } diff --git a/signer/config.go b/signer/config.go index fca7737f..d85df45e 100644 --- a/signer/config.go +++ b/signer/config.go @@ -204,7 +204,6 @@ func (cfg *ThresholdModeConfig) LeaderElectMultiAddress() (string, error) { type CosignerConfig struct { ShardID uint8 `yaml:"shardID"` P2PAddr string `yaml:"p2pAddr"` - DKGID string `yaml:"dkgID,omitempty"` } func (c CosignerConfig) HostPort() (string, string, error) { diff --git a/signer/cosigner_key_test.go b/signer/cosigner_key_test.go index b06f927c..3e00aab3 100644 --- a/signer/cosigner_key_test.go +++ b/signer/cosigner_key_test.go @@ -19,7 +19,7 @@ func TestLoadCosignerEd25519Key(t *testing.T) { key, err := LoadCosignerRSAKey(rsaKeyFile) require.NoError(t, err) - require.Equal(t, key.ID, 3) + require.Equal(t, uint8(3), key.ID) // public key from cosigner pubs array should match public key from our private key require.Equal(t, &key.RSAKey.PublicKey, key.RSAPubs[key.ID-1]) diff --git a/signer/dkg.go b/signer/dkg.go index ffcde512..4063fd03 100644 --- a/signer/dkg.go +++ b/signer/dkg.go @@ -2,8 +2,11 @@ package signer import ( "context" + "crypto/rsa" + "crypto/x509" "encoding/json" "fmt" + "math/rand" "sync" "time" @@ -15,7 +18,6 @@ import ( "github.com/libp2p/go-libp2p/core/host" "github.com/libp2p/go-libp2p/core/peer" "github.com/strangelove-ventures/horcrux/signer/keygen" - "golang.org/x/sync/errgroup" ) var ( @@ -32,7 +34,7 @@ func NetworkDKG( ctx context.Context, cosigners CosignersConfig, id uint8, - p2pPrivateKey crypto.PrivKey, + rsaKeys CosignerRSAKey, threshold uint8, ) (*CosignerEd25519Key, error) { cosigner, err := cosigners.MyCosigner(id) @@ -45,6 +47,13 @@ func NetworkDKG( return nil, err } + x509Bz := x509.MarshalPKCS1PrivateKey(&rsaKeys.RSAKey) + + p2pPrivateKey, err := crypto.UnmarshalRsaPrivateKey(x509Bz) + if err != nil { + return nil, err + } + h, err := libp2p.New(libp2p.ListenAddrStrings(hostAddr), libp2p.Identity(p2pPrivateKey)) if err != nil { return nil, fmt.Errorf("failed to construct libp2p node: %w", err) @@ -54,7 +63,7 @@ func NetworkDKG( fmt.Println("Starting cosigner discovery") - if err := waitForAllCosigners(ctx, h, cosigners.OtherCosigners(id)); err != nil { + if err := waitForAllCosigners(ctx, h, id, cosigners.OtherCosigners(id), rsaKeys.RSAPubs); err != nil { return nil, err } @@ -72,22 +81,6 @@ func NetworkDKG( return nil, err } - round1, err := keygenCosigner.Round1() - if err != nil { - return nil, err - } - - round1Msgs := RoundMessages{ - Round: 1, - ID: id, - Messages: round1, - } - - round1MsgsBz, err := json.Marshal(round1Msgs) - if err != nil { - return nil, err - } - dkgTopic, err := ps.Join(topicDKG) if err != nil { return nil, fmt.Errorf("failed to join topic: %w", err) @@ -98,16 +91,7 @@ func NetworkDKG( return nil, fmt.Errorf("failed to subscribe to topic: %w", err) } - var eg errgroup.Group - eg.Go(func() error { - return processRounds(ctx, dkgTopic, sub, keygenCosigner, uint8(total), round1[0]) - }) - - if err := dkgTopic.Publish(ctx, round1MsgsBz); err != nil { - return nil, err - } - - if err := eg.Wait(); err != nil { + if err := processRounds(ctx, dkgTopic, sub, keygenCosigner, uint8(total)); err != nil { return nil, err } @@ -118,23 +102,61 @@ func NetworkDKG( }, nil } +func publishUntilRoundDone( + ctx context.Context, + topic *pubsub.Topic, + msg []byte, + doneCh chan struct{}, +) { + for { + nextPublish := rand.Intn(1000) + 2000 //nolint + select { + case <-ctx.Done(): + return + case <-doneCh: + return + case <-time.After(time.Duration(nextPublish) * time.Millisecond): + if err := topic.Publish(ctx, msg); err != nil { + fmt.Printf("Failed to publish msg to topic: %v\n", err) + } + } + } +} + func processRounds( ctx context.Context, dkgTopic *pubsub.Topic, sub *pubsub.Subscription, keygenCosigner keygen.Cosigner, total uint8, - myRound1Msg []byte, ) error { id := uint8(keygenCosigner.ID) allRound1Msgs := make([][]byte, total) + allRound2Msgs := make([][]byte, total*(total-1)) + allRound3Msgs := make([][]byte, total) - allRound1Msgs[id-1] = myRound1Msg + round1DoneCh := make(chan struct{}, 1) + round2DoneCh := make(chan struct{}, 1) + round3DoneCh := make(chan struct{}, 1) - allRound2Msgs := make([][]byte, total*(total-1)) + round1, err := keygenCosigner.Round1() + if err != nil { + return err + } - allRound3Msgs := make([][]byte, total) + round1Msgs := RoundMessages{ + Round: 1, + ID: id, + Messages: round1, + } + + round1MsgsBz, err := json.Marshal(round1Msgs) + if err != nil { + return err + } + + go publishUntilRoundDone(ctx, dkgTopic, round1MsgsBz, round1DoneCh) RecvRoundMsgsLoop: for { @@ -143,7 +165,7 @@ RecvRoundMsgsLoop: fmt.Printf("Failed to retrieve next message from topic subscription: %v\n", err) continue RecvRoundMsgsLoop } - fmt.Println(m.ReceivedFrom, ": ", string(m.Message.Data)) + fmt.Printf("Received data on cosigner %d - %s\n", id, string(m.Message.Data)) var cosignerRoundMsgs RoundMessages if err := json.Unmarshal(m.Message.Data, &cosignerRoundMsgs); err != nil { @@ -156,12 +178,20 @@ RecvRoundMsgsLoop: case 1: allRound1Msgs[cID-1] = cosignerRoundMsgs.Messages[0] - for _, m := range allRound1Msgs { + shouldContinue := false + for i, m := range allRound1Msgs { if len(m) == 0 { - continue RecvRoundMsgsLoop + fmt.Printf("Still waiting on msgs for round 1 on cosigner %d from cosigner %d\n", id, i+1) + shouldContinue = true } } + if shouldContinue { + continue RecvRoundMsgsLoop + } + + round1DoneCh <- struct{}{} + // have messages from all cosigners for round 1 round2, err := keygenCosigner.Round2(allRound1Msgs) if err != nil { @@ -179,21 +209,25 @@ RecvRoundMsgsLoop: return err } - copy(allRound2Msgs[(id-1)*(total-1):id*(total-1)], round2) - - if err := dkgTopic.Publish(ctx, round2MsgsBz); err != nil { - fmt.Printf("Failed to publish round 2 messages: %v\n", err) - } + go publishUntilRoundDone(ctx, dkgTopic, round2MsgsBz, round2DoneCh) case 2: copy(allRound2Msgs[(cID-1)*(total-1):cID*(total-1)], cosignerRoundMsgs.Messages) - for _, m := range allRound2Msgs { + shouldContinue := false + for i, m := range allRound2Msgs { if len(m) == 0 { - continue RecvRoundMsgsLoop + fmt.Printf("Still waiting on msgs for round 2 on cosigner %d from cosigner %d\n", id, i+1) + shouldContinue = true } } + if shouldContinue { + continue RecvRoundMsgsLoop + } + + round2DoneCh <- struct{}{} + // have messages from all cosigners for round 2 if err := keygenCosigner.Round3(allRound2Msgs); err != nil { return err @@ -213,18 +247,24 @@ RecvRoundMsgsLoop: return err } - if err := dkgTopic.Publish(ctx, round3MsgsBz); err != nil { - fmt.Printf("Failed to publish round 3 result: %v\n", err) - } - + go publishUntilRoundDone(ctx, dkgTopic, round3MsgsBz, round3DoneCh) case 3: allRound3Msgs[cID-1] = []byte{0x01} - for _, m := range allRound3Msgs { + + shouldContinue := false + for i, m := range allRound3Msgs { if len(m) == 0 { - continue RecvRoundMsgsLoop + fmt.Printf("Still waiting on msgs for round 3 on cosigner %d from cosigner %d\n", id, i+1) + shouldContinue = true } } + if shouldContinue { + continue RecvRoundMsgsLoop + } + + round3DoneCh <- struct{}{} + // have success from all cosigners, sharding complete return nil default: @@ -233,7 +273,13 @@ RecvRoundMsgsLoop: } } -func waitForAllCosigners(ctx context.Context, h host.Host, cosigners CosignersConfig) error { +func waitForAllCosigners( + ctx context.Context, + h host.Host, + id uint8, + cosigners CosignersConfig, + rsaPubs []*rsa.PublicKey, +) error { var wg sync.WaitGroup for _, c := range cosigners { peerAddr, err := c.LibP2PAddr() @@ -241,7 +287,22 @@ func waitForAllCosigners(ctx context.Context, h host.Host, cosigners CosignersCo return err } - peerinfo, err := peer.AddrInfoFromString(peerAddr + "/p2p/" + c.DKGID) + x509Bz, err := x509.MarshalPKIXPublicKey(rsaPubs[c.ShardID-1]) + if err != nil { + return err + } + + pubKey, err := crypto.UnmarshalRsaPublicKey(x509Bz) + if err != nil { + return err + } + + peerID, err := peer.IDFromPublicKey(pubKey) + if err != nil { + return err + } + + peerinfo, err := peer.AddrInfoFromString(peerAddr + "/p2p/" + peerID.String()) if err != nil { return err } @@ -261,7 +322,7 @@ func waitForAllCosigners(ctx context.Context, h host.Host, cosigners CosignersCo retry.Delay(time.Second), ) - fmt.Printf("Connection established with bootstrap node: %s\n", *peerinfo) + fmt.Printf("Connection established with bootstrap node from cosigner %d: %s\n", id, *peerinfo) }() } wg.Wait() diff --git a/signer/dkg_test.go b/signer/dkg_test.go index 7a838caa..81b3f83f 100644 --- a/signer/dkg_test.go +++ b/signer/dkg_test.go @@ -3,11 +3,14 @@ package signer import ( "context" "crypto/rand" + "crypto/rsa" + "errors" "testing" - "github.com/libp2p/go-libp2p/core/crypto" - "github.com/libp2p/go-libp2p/core/peer" "github.com/stretchr/testify/require" + "github.com/taurusgroup/frost-ed25519/pkg/eddsa" + "github.com/taurusgroup/frost-ed25519/pkg/frost/party" + "github.com/taurusgroup/frost-ed25519/pkg/ristretto" tsed25519 "gitlab.com/unit410/threshold-ed25519/pkg" "golang.org/x/sync/errgroup" ) @@ -15,33 +18,52 @@ import ( func TestNetworkDKG(t *testing.T) { ctx := context.Background() + rsaKey1, err := rsa.GenerateKey(rand.Reader, bitSize) + require.NoError(t, err) + + rsaKey2, err := rsa.GenerateKey(rand.Reader, bitSize) + require.NoError(t, err) + + rsaKey3, err := rsa.GenerateKey(rand.Reader, bitSize) + require.NoError(t, err) + cosigners := []CosignerConfig{ { ShardID: 1, - P2PAddr: "tcp://127.0.0.1:8040", + P2PAddr: "tcp://127.0.0.1:8062", }, { ShardID: 2, - P2PAddr: "tcp://127.0.0.1:8041", + P2PAddr: "tcp://127.0.0.1:8063", }, { ShardID: 3, - P2PAddr: "tcp://127.0.0.1:8042", + P2PAddr: "tcp://127.0.0.1:8064", }, } - cosignerPrivKeys := make([]crypto.PrivKey, len(cosigners)) - - for i := range cosigners { - privKey, _, err := crypto.GenerateEd25519Key(rand.Reader) - require.NoError(t, err) - - p2pID, err := peer.IDFromPrivateKey(privKey) - require.NoError(t, err) - - cosigners[i].DKGID = p2pID.String() + rsaPubs := []*rsa.PublicKey{ + &rsaKey1.PublicKey, + &rsaKey2.PublicKey, + &rsaKey3.PublicKey, + } - cosignerPrivKeys[i] = privKey + rsaKeys := []CosignerRSAKey{ + { + ID: 1, + RSAKey: *rsaKey1, + RSAPubs: rsaPubs, + }, + { + ID: 2, + RSAKey: *rsaKey2, + RSAPubs: rsaPubs, + }, + { + ID: 3, + RSAKey: *rsaKey3, + RSAPubs: rsaPubs, + }, } const threshold = 2 @@ -54,7 +76,7 @@ func TestNetworkDKG(t *testing.T) { id := c.ShardID i := i eg.Go(func() (err error) { - shards[i], err = NetworkDKG(ctx, cosigners, id, cosignerPrivKeys[i], threshold) + shards[i], err = NetworkDKG(ctx, cosigners, id, rsaKeys[i], threshold) return err }) } @@ -71,3 +93,88 @@ func TestNetworkDKG(t *testing.T) { testThresholdValidatorWithShards(t, threshold, uint8(len(cosigners)), pubKey, privShards) } + +func TestKeygenCosigner(t *testing.T) { + const ( + threshold = 2 + total = 3 + ) + + cosigners, err := LocalDKG(threshold, total) + require.NoError(t, err) + + var pubCmp *eddsa.Public + secrets := make(map[party.ID]*eddsa.SecretShare) + + for _, c := range cosigners { + p := c.Public() + secrets[c.ID] = c.Secret() + if pubCmp == nil { + pubCmp = p + continue + } + err := CompareOutput(pubCmp, p) + require.NoError(t, err) + } + + err = ValidateSecrets(secrets, pubCmp) + require.NoError(t, err) +} + +func CompareOutput(public1, public2 *eddsa.Public) error { + if !public1.Equal(public2) { + return errors.New("shares not equal") + } + partyIDs1 := public1.PartyIDs + partyIDs2 := public2.PartyIDs + if len(partyIDs1) != len(partyIDs2) { + return errors.New("partyIDs are not the same length") + } + + for i, id1 := range partyIDs1 { + if id1 != partyIDs2[i] { + return errors.New("partyIDs are not the same") + } + + publicShare1 := public1.Shares[partyIDs1[i]] + publicShare2 := public2.Shares[partyIDs2[i]] + if publicShare1.Equal(publicShare2) != 1 { + return errors.New("different public keys") + } + } + + if !public1.GroupKey.Equal(public2.GroupKey) { + return errors.New("groupKeys not computed the same way") + } + + return nil +} + +func ValidateSecrets(secrets map[party.ID]*eddsa.SecretShare, public *eddsa.Public) error { + fullSecret := ristretto.NewScalar() + + for id, secret := range secrets { + pk1 := &secret.Public + pk2, ok := public.Shares[id] + if !ok { + return errors.New("party %d has no share") + } + + if pk1.Equal(pk2) != 1 { + return errors.New("pk not the same") + } + + lagrange, err := id.Lagrange(public.PartyIDs) + if err != nil { + return err + } + fullSecret.MultiplyAdd(lagrange, &secret.Secret, fullSecret) + } + + fullPk := eddsa.NewPublicKeyFromPoint(new(ristretto.Element).ScalarBaseMult(fullSecret)) + if !public.GroupKey.Equal(fullPk) { + return errors.New("computed groupKey does not match") + } + + return nil +} diff --git a/signer/keygen_test.go b/signer/keygen_test.go deleted file mode 100644 index 4868f6ad..00000000 --- a/signer/keygen_test.go +++ /dev/null @@ -1,97 +0,0 @@ -package signer_test - -import ( - "errors" - "testing" - - "github.com/strangelove-ventures/horcrux/signer" - "github.com/stretchr/testify/require" - "github.com/taurusgroup/frost-ed25519/pkg/eddsa" - "github.com/taurusgroup/frost-ed25519/pkg/frost/party" - "github.com/taurusgroup/frost-ed25519/pkg/ristretto" -) - -func TestKeygenCosigner(t *testing.T) { - const ( - threshold = 2 - total = 3 - ) - - cosigners, err := signer.LocalDKG(threshold, total) - require.NoError(t, err) - - var pubCmp *eddsa.Public - secrets := make(map[party.ID]*eddsa.SecretShare) - - for _, c := range cosigners { - p := c.Public() - secrets[c.ID] = c.Secret() - if pubCmp == nil { - pubCmp = p - continue - } - err := CompareOutput(pubCmp, p) - require.NoError(t, err) - } - - err = ValidateSecrets(secrets, pubCmp) - require.NoError(t, err) -} - -func CompareOutput(public1, public2 *eddsa.Public) error { - if !public1.Equal(public2) { - return errors.New("shares not equal") - } - partyIDs1 := public1.PartyIDs - partyIDs2 := public2.PartyIDs - if len(partyIDs1) != len(partyIDs2) { - return errors.New("partyIDs are not the same length") - } - - for i, id1 := range partyIDs1 { - if id1 != partyIDs2[i] { - return errors.New("partyIDs are not the same") - } - - publicShare1 := public1.Shares[partyIDs1[i]] - publicShare2 := public2.Shares[partyIDs2[i]] - if publicShare1.Equal(publicShare2) != 1 { - return errors.New("different public keys") - } - } - - if !public1.GroupKey.Equal(public2.GroupKey) { - return errors.New("groupKeys not computed the same way") - } - - return nil -} - -func ValidateSecrets(secrets map[party.ID]*eddsa.SecretShare, public *eddsa.Public) error { - fullSecret := ristretto.NewScalar() - - for id, secret := range secrets { - pk1 := &secret.Public - pk2, ok := public.Shares[id] - if !ok { - return errors.New("party %d has no share") - } - - if pk1.Equal(pk2) != 1 { - return errors.New("pk not the same") - } - - lagrange, err := id.Lagrange(public.PartyIDs) - if err != nil { - return err - } - fullSecret.MultiplyAdd(lagrange, &secret.Secret, fullSecret) - } - - fullPk := eddsa.NewPublicKeyFromPoint(new(ristretto.Element).ScalarBaseMult(fullSecret)) - if !public.GroupKey.Equal(fullPk) { - return errors.New("computed groupKey does not match") - } - - return nil -} diff --git a/signer/local_cosigner_test.go b/signer/local_cosigner_test.go index 5a104f17..1c94e3ed 100644 --- a/signer/local_cosigner_test.go +++ b/signer/local_cosigner_test.go @@ -47,7 +47,7 @@ func TestLocalCosignerGetID(t *testing.T) { 0, ) - require.Equal(t, 1, cosigner.GetID()) + require.Equal(t, uint8(1), cosigner.GetID()) } func TestLocalCosignerSign2of2(t *testing.T) { @@ -145,8 +145,8 @@ func TestLocalCosignerSign2of2(t *testing.T) { err = cosigner2.LoadSignStateIfNecessary(testChainID) require.NoError(t, err) - require.Equal(t, cosigner1.GetID(), 1) - require.Equal(t, cosigner2.GetID(), 2) + require.Equal(t, uint8(1), cosigner1.GetID()) + require.Equal(t, uint8(2), cosigner2.GetID()) publicKeys := make([]tsed25519.Element, 0) From b204af17e53c1cdf3135a808e30eb027db380f61 Mon Sep 17 00:00:00 2001 From: Andrew Gouin Date: Fri, 19 May 2023 15:46:18 -0600 Subject: [PATCH 5/6] Add dkg e2e test --- signer/config.go | 7 ++++++- test/horcrux_test.go | 24 ++++++++++++------------ test/signer.go | 27 +++++++++++++++++---------- test/validator.go | 44 +++++++++++++++++++++++++++++++++++++++++--- 4 files changed, 76 insertions(+), 26 deletions(-) diff --git a/signer/config.go b/signer/config.go index d85df45e..0bac1ad8 100644 --- a/signer/config.go +++ b/signer/config.go @@ -235,7 +235,12 @@ func (c CosignerConfig) LibP2PAddr() (string, error) { return "", err } - return fmt.Sprintf("/ip4/%s/tcp/%s", host, port), nil + protocol := "ip4" + if net.ParseIP(host) == nil { + protocol = "dns" + } + + return fmt.Sprintf("/%s/%s/tcp/%s", protocol, host, port), nil } type CosignersConfig []CosignerConfig diff --git a/test/horcrux_test.go b/test/horcrux_test.go index 72258c9b..6254c0d6 100644 --- a/test/horcrux_test.go +++ b/test/horcrux_test.go @@ -26,7 +26,7 @@ func testChainSingleNodeAndHorcrux( chain := getSimdChain(chainID, totalSentries) // setup a horcrux validator for us - ourValidator, err := NewHorcruxValidator(t, pool, network, home, 0, totalSigners, threshold, chain) + ourValidator, err := NewHorcruxValidator(ctx, t, pool, network, home, 0, totalSigners, threshold, true, chain) require.NoError(t, err) // other vals are single node (non-horcrux) @@ -186,7 +186,7 @@ func TestUpgradeValidatorToHorcrux(t *testing.T) { require.NoError(t, err) // create horcrux validator with same consensus key - ourValidatorUpgradedToHorcrux, err := NewHorcruxValidatorWithPrivValKey(t, pool, network, home, + ourValidatorUpgradedToHorcrux, err := NewHorcruxValidatorWithPrivValKey(ctx, t, pool, network, home, 0, 0, totalSigners, threshold, chain, ourValidatorPrivValKey) require.NoError(t, err) @@ -232,8 +232,8 @@ func TestDownedSigners2of3(t *testing.T) { chain := getSimdChain(chainID, sentriesPerSigner) // setup a horcrux validator for us - ourValidator, err := NewHorcruxValidator(t, pool, network, home, - 0, totalSigners, threshold, chain) + ourValidator, err := NewHorcruxValidator(ctx, t, pool, network, home, + 0, totalSigners, threshold, true, chain) require.NoError(t, err) // remaining validators are single-node non-horcrux @@ -281,8 +281,8 @@ func TestLeaderElection2of3(t *testing.T) { chain := getSimdChain(chainID, totalSentries) // setup a horcrux validator for us - ourValidator, err := NewHorcruxValidator(t, pool, network, home, - 0, totalSigners, threshold, chain) + ourValidator, err := NewHorcruxValidator(ctx, t, pool, network, home, + 0, totalSigners, threshold, true, chain) require.NoError(t, err) // remaining validators are single-node non-horcrux @@ -354,8 +354,8 @@ func TestDownedSigners3of5(t *testing.T) { chain := getSimdChain(chainID, totalSentries) // setup a horcrux validator for us - ourValidator, err := NewHorcruxValidator(t, pool, network, home, - 0, totalSigners, threshold, chain) + ourValidator, err := NewHorcruxValidator(ctx, t, pool, network, home, + 0, totalSigners, threshold, true, chain) require.NoError(t, err) // remaining validators are single-node non-horcrux @@ -431,8 +431,8 @@ func TestChainPureHorcrux(t *testing.T) { // start horcrux cluster for each validator for i := 0; i < totalValidators; i++ { - validator, err := NewHorcruxValidator(t, pool, network, home, i, - signersPerValidator, threshold, chain) + validator, err := NewHorcruxValidator(ctx, t, pool, network, home, i, + signersPerValidator, threshold, true, chain) require.NoError(t, err) validators = append(validators, validator) allNodes = append(allNodes, validator.Sentries[chainID]...) @@ -488,8 +488,8 @@ func TestMultipleChainHorcrux(t *testing.T) { // start horcrux cluster for each validator for i := 0; i < totalValidators; i++ { - validator, err := NewHorcruxValidator(t, pool, network, home, i, - signersPerValidator, threshold, chain1, chain2) + validator, err := NewHorcruxValidator(ctx, t, pool, network, home, i, + signersPerValidator, threshold, true, chain1, chain2) require.NoError(t, err) validators = append(validators, validator) allNodes = append(allNodes, validator.Sentries[chainID1]...) diff --git a/test/signer.go b/test/signer.go index 597e197c..6753c81c 100644 --- a/test/signer.go +++ b/test/signer.go @@ -205,8 +205,7 @@ func StartCosignerContainers( s := s eg.Go(func() error { return s.InitThresholdModeConfig(ctx, peers[i], signers, threshold) }) } - err := eg.Wait() - if err != nil { + if err := eg.Wait(); err != nil { return err } @@ -217,8 +216,7 @@ func StartCosignerContainers( return s.CreateCosignerContainer() }) } - err = eg.Wait() - if err != nil { + if err := eg.Wait(); err != nil { return err } @@ -311,15 +309,20 @@ func (ts *Signer) GRPCAddress() string { // ExecHorcruxCmd executes a CLI subcommand for the horcrux binary for the specific cosigner. // The config home directory will be appended as a flag. -func (ts *Signer) ExecHorcruxCmd(ctx context.Context, cmd ...string) error { +func (ts *Signer) ExecHorcruxCmd(ctx context.Context, hostnameOverride string, cmd ...string) error { cmd = ts.horcruxCmd(cmd) container := RandLowerCaseLetterString(10) ts.tl.Logf("{%s}[%s] -> '%s'", ts.Name(), container, strings.Join(cmd, " ")) + + hostname := container + if hostnameOverride != "" { + hostname = hostnameOverride + } cont, err := ts.Pool.Client.CreateContainer(docker.CreateContainerOptions{ Name: container, Config: &docker.Config{ User: getDockerUserString(), - Hostname: container, + Hostname: hostname, ExposedPorts: map[docker.Port]struct{}{ docker.Port(signerPortDocker): {}, }, @@ -378,7 +381,7 @@ func (ts *Signer) ExecHorcruxCmd(ctx context.Context, cmd ...string) error { func (ts *Signer) InitSingleSignerConfig(ctx context.Context, listenNodes Nodes) error { cmd := []string{"config", "init", "--mode", "single"} cmd = append(cmd, listenNodes.ConfigInitFlags()...) - return ts.ExecHorcruxCmd(ctx, cmd...) + return ts.ExecHorcruxCmd(ctx, "", cmd...) } // InitThresholdModeConfig creates and runs a container to init a signer nodes config files @@ -388,8 +391,12 @@ func (ts *Signer) InitThresholdModeConfig( cmd := []string{"config", "init"} cmd = append(cmd, listenNodes.ConfigInitFlags()...) cmd = append(cmd, cosigners.ConfigInitFlags()...) - cmd = append(cmd, "--threshold", fmt.Sprint(threshold)) - return ts.ExecHorcruxCmd(ctx, cmd...) + cmd = append(cmd, + "--threshold", fmt.Sprint(threshold), + "--overwrite", + ) + + return ts.ExecHorcruxCmd(ctx, "", cmd...) } // StartContainer starts a Signers container and assigns the new running container to replace the old one @@ -514,7 +521,7 @@ func (ts *Signer) CreateCosignerContainer() error { // TransferLeadership elects a new raft leader. func (ts *Signer) TransferLeadership(ctx context.Context, newLeaderID int) error { - return ts.ExecHorcruxCmd(ctx, + return ts.ExecHorcruxCmd(ctx, "", "elect", strconv.FormatInt(int64(newLeaderID), 10), ) } diff --git a/test/validator.go b/test/validator.go index b577315f..9de18651 100644 --- a/test/validator.go +++ b/test/validator.go @@ -1,6 +1,7 @@ package test import ( + "context" "fmt" "os" "path/filepath" @@ -10,6 +11,7 @@ import ( "github.com/cometbft/cometbft/privval" "github.com/ory/dockertest" "github.com/strangelove-ventures/horcrux/signer" + "golang.org/x/sync/errgroup" ) type Validator struct { @@ -23,6 +25,7 @@ type Validator struct { } func NewHorcruxValidator( + ctx context.Context, tl Logger, pool *dockertest.Pool, networkID string, @@ -30,6 +33,7 @@ func NewHorcruxValidator( index int, numSigners int, threshold uint8, + dkg bool, chains ...*ChainType, ) (*Validator, error) { chainIDs := make([]string, 0, len(chains)) @@ -56,13 +60,14 @@ func NewHorcruxValidator( Threshold: threshold, PubKeys: make(map[string]cometcrypto.PubKey), } - if err := testValidator.genPrivKeyAndShards(nil, chainIDs...); err != nil { + if err := testValidator.genPrivKeyAndShards(ctx, nil, dkg, chainIDs...); err != nil { return nil, err } return testValidator, nil } func NewHorcruxValidatorWithPrivValKey( + ctx context.Context, tl Logger, pool *dockertest.Pool, networkID string, @@ -86,7 +91,7 @@ func NewHorcruxValidatorWithPrivValKey( Threshold: threshold, PubKeys: make(map[string]cometcrypto.PubKey), } - if err := testValidator.genPrivKeyAndShards(&privValKey, chainID); err != nil { + if err := testValidator.genPrivKeyAndShards(ctx, &privValKey, false, chainID); err != nil { return nil, err } return testValidator, nil @@ -123,14 +128,47 @@ func (tv *Validator) genRSAShares() error { return nil } +// performDKG runs the DKG keygen process via libp2p. +func (tv *Validator) performDKG(ctx context.Context, chainID string) error { + var eg errgroup.Group + for _, s := range tv.Signers { + s := s + eg.Go(func() error { return s.InitThresholdModeConfig(ctx, tv.Sentries[chainID], tv.Signers, tv.Threshold) }) + } + if err := eg.Wait(); err != nil { + return err + } + + for i, s := range tv.Signers { + s := s + i := i + eg.Go(func() error { + return s.ExecHorcruxCmd(ctx, s.Name(), "dkg", "--chain-id", chainID, "--id", fmt.Sprint(i+1)) + }) + } + return eg.Wait() +} + // genPrivKeyAndShards generates cosigner RSA shards. // If existingKey is nil, generates Ed25519 key shards, otherwise shards existing key. -func (tv *Validator) genPrivKeyAndShards(existingKey *privval.FilePVKey, chainIDs ...string) error { +func (tv *Validator) genPrivKeyAndShards( + ctx context.Context, + existingKey *privval.FilePVKey, + dkg bool, + chainIDs ...string, +) error { if err := tv.genRSAShares(); err != nil { return err } for _, chainID := range chainIDs { + if dkg { + if err := tv.performDKG(ctx, chainID); err != nil { + return err + } + continue + } + if err := tv.genEd25519Shards(existingKey, chainID); err != nil { return err } From 3f5e2ed5586922802f3191ddc6b587c6b5cc10b1 Mon Sep 17 00:00:00 2001 From: Andrew Gouin Date: Fri, 19 May 2023 17:59:37 -0600 Subject: [PATCH 6/6] fix e2e test --- cmd/horcrux/cmd/dkg.go | 18 +++++++----------- signer/dkg.go | 16 ++++++++-------- signer/dkg_test.go | 2 +- test/signer.go | 11 +++++++++++ test/validator.go | 13 ++++++++++++- 5 files changed, 39 insertions(+), 21 deletions(-) diff --git a/cmd/horcrux/cmd/dkg.go b/cmd/horcrux/cmd/dkg.go index da64f010..35894ded 100644 --- a/cmd/horcrux/cmd/dkg.go +++ b/cmd/horcrux/cmd/dkg.go @@ -1,7 +1,6 @@ package cmd import ( - "encoding/json" "errors" "fmt" "os" @@ -66,13 +65,6 @@ func dkgCmd() *cobra.Command { return errors.Join(errs...) } - out, _ := cmd.Flags().GetString(flagOutputDir) - if out != "" { - if err := os.MkdirAll(out, 0700); err != nil { - return err - } - } - keyFile, err := config.KeyFileExistsCosignerRSA() if err != nil { return err @@ -91,11 +83,17 @@ func dkgCmd() *cobra.Command { return err } - shardBz, err := json.Marshal(shard) + shardBz, err := shard.MarshalJSON() if err != nil { return err } + out := config.HomeDir + + if config.Config.PrivValKeyDir != nil { + out = *config.Config.PrivValKeyDir + } + filename := filepath.Join(out, fmt.Sprintf("%s_shard.json", chainID)) if err := os.WriteFile(filename, shardBz, 0600); err != nil { @@ -108,8 +106,6 @@ func dkgCmd() *cobra.Command { }, } - addOutputDirFlag(cmd) - f := cmd.Flags() f.Uint8(flagID, 0, "cosigner shard ID as participant in DKG ceremony") _ = cmd.MarkFlagRequired(flagID) diff --git a/signer/dkg.go b/signer/dkg.go index 4063fd03..4dd96452 100644 --- a/signer/dkg.go +++ b/signer/dkg.go @@ -61,14 +61,6 @@ func NetworkDKG( defer h.Close() - fmt.Println("Starting cosigner discovery") - - if err := waitForAllCosigners(ctx, h, id, cosigners.OtherCosigners(id), rsaKeys.RSAPubs); err != nil { - return nil, err - } - - fmt.Println("Cosigner discovery complete") - ps, err := pubsub.NewGossipSub(ctx, h) if err != nil { return nil, fmt.Errorf("failed to setup libp2p pub sub: %w", err) @@ -91,6 +83,14 @@ func NetworkDKG( return nil, fmt.Errorf("failed to subscribe to topic: %w", err) } + fmt.Println("Starting cosigner discovery") + + if err := waitForAllCosigners(ctx, h, id, cosigners.OtherCosigners(id), rsaKeys.RSAPubs); err != nil { + return nil, err + } + + fmt.Println("Cosigner discovery complete") + if err := processRounds(ctx, dkgTopic, sub, keygenCosigner, uint8(total)); err != nil { return nil, err } diff --git a/signer/dkg_test.go b/signer/dkg_test.go index 81b3f83f..36958fcd 100644 --- a/signer/dkg_test.go +++ b/signer/dkg_test.go @@ -34,7 +34,7 @@ func TestNetworkDKG(t *testing.T) { }, { ShardID: 2, - P2PAddr: "tcp://127.0.0.1:8063", + P2PAddr: "tcp://localhost:8063", }, { ShardID: 3, diff --git a/test/signer.go b/test/signer.go index 6753c81c..b22f3f4f 100644 --- a/test/signer.go +++ b/test/signer.go @@ -12,6 +12,8 @@ import ( "strings" "time" + cometcrypto "github.com/cometbft/cometbft/crypto" + cometcryptoed25519 "github.com/cometbft/cometbft/crypto/ed25519" cometjson "github.com/cometbft/cometbft/libs/json" "github.com/ory/dockertest" "github.com/ory/dockertest/docker" @@ -296,6 +298,15 @@ func (ts *Signer) GetConfigFile() string { return filepath.Join(ts.Dir(), "config.yaml") } +// PubKey returns the public key representing all cosigners for a chain ID. +func (ts *Signer) PubKey(chainID string) (cometcrypto.PubKey, error) { + key, err := signer.LoadCosignerEd25519Key(filepath.Join(ts.Dir(), fmt.Sprintf("%s_shard.json", chainID))) + if err != nil { + return nil, err + } + return cometcryptoed25519.PubKey(key.PubKey.Bytes()), nil +} + // Name is the hostname of the Signer container func (ts *Signer) Name() string { return fmt.Sprintf("val-%d-sgn-%d-%s", ts.ValidatorIndex, ts.Index, ts.tl.Name()) diff --git a/test/validator.go b/test/validator.go index 9de18651..9a76d661 100644 --- a/test/validator.go +++ b/test/validator.go @@ -146,7 +146,18 @@ func (tv *Validator) performDKG(ctx context.Context, chainID string) error { return s.ExecHorcruxCmd(ctx, s.Name(), "dkg", "--chain-id", chainID, "--id", fmt.Sprint(i+1)) }) } - return eg.Wait() + if err := eg.Wait(); err != nil { + return err + } + + pubKey, err := tv.Signers[0].PubKey(chainID) + if err != nil { + return err + } + + tv.PubKeys[chainID] = pubKey + + return nil } // genPrivKeyAndShards generates cosigner RSA shards.