// Copyright © 2017-2023 Weald Technology Trading // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package cmd import ( "context" "fmt" "os" spec "github.com/attestantio/go-eth2-client/spec/phase0" "github.com/pkg/errors" "github.com/spf13/cobra" "github.com/spf13/viper" "github.com/wealdtech/ethdo/util" "github.com/wealdtech/go-bytesutil" e2types "github.com/wealdtech/go-eth2-types/v2" e2wtypes "github.com/wealdtech/go-eth2-wallet-types/v2" ) // signatureSignCmd represents the signature sign command. var signatureSignCmd = &cobra.Command{ Use: "sign", Short: "Sign a 32-byte piece of data", Long: `Sign presented data. For example: ethdo signature sign --data=0x5f24e819400c6a8ee2bfc014343cd971b7eb707320025a7bcd83e621e26c35b7 --account="Personal wallet/Operations" --passphrase="my account passphrase" In quiet mode this will return 0 if the data can be signed, otherwise 1.`, Run: func(_ *cobra.Command, _ []string) { ctx, cancel := context.WithTimeout(context.Background(), viper.GetDuration("timeout")) defer cancel() assert(viper.GetString("signature-data") != "", "--data is required") data, err := bytesutil.FromHexString(viper.GetString("signature-data")) errCheck(err, "Failed to parse data") assert(len(data) == 32, "data to sign must be 32 bytes") domain := e2types.Domain(e2types.DomainType([4]byte{0, 0, 0, 0}), e2types.ZeroForkVersion, e2types.ZeroGenesisValidatorsRoot) if viper.GetString("signature-domain") != "" { domain, err = bytesutil.FromHexString(viper.GetString("signature-domain")) errCheck(err, "Failed to parse domain") assert(len(domain) == 32, "Domain data invalid") } outputIf(viper.GetBool("debug"), fmt.Sprintf("Domain is %#x", domain)) var account e2wtypes.Account switch { case viper.GetString("account") != "": account, err = util.ParseAccount(ctx, viper.GetString("account"), util.GetPassphrases(), true) case viper.GetString("private-key") != "": account, err = util.ParseAccount(ctx, viper.GetString("private-key"), nil, true) default: err = errors.New("account or private key must be supplied") } errCheck(err, "Failed to obtain account") var specDomain spec.Domain copy(specDomain[:], domain) var fixedSizeData [32]byte copy(fixedSizeData[:], data) outputIf(viper.GetBool("debug"), fmt.Sprintf("Signing %#x with domain %#x by public key %#x", fixedSizeData, specDomain, account.PublicKey().Marshal())) signature, err := util.SignRoot(account, fixedSizeData, specDomain) errCheck(err, "Failed to sign") outputIf(!viper.GetBool("quiet"), fmt.Sprintf("%#x", signature.Marshal())) os.Exit(_exitSuccess) }, } func init() { signatureCmd.AddCommand(signatureSignCmd) signatureFlags(signatureSignCmd) }