From 25bacac03368b324dba0e6e8a8b8d5f5b9f68b72 Mon Sep 17 00:00:00 2001 From: John-Alan Simmons Date: Thu, 4 Apr 2024 06:35:04 -0400 Subject: [PATCH] persisent host keys --- pkg/host/host.go | 50 +++++++++++++++++++++++++++++++++++++++++------- 1 file changed, 43 insertions(+), 7 deletions(-) diff --git a/pkg/host/host.go b/pkg/host/host.go index b002c38..b12ad4a 100644 --- a/pkg/host/host.go +++ b/pkg/host/host.go @@ -2,6 +2,9 @@ package host import ( "context" + "errors" + "os" + "path/filepath" "crypto/rand" "fmt" @@ -55,15 +58,48 @@ func New(ctx context.Context, cfg config.Host) (*Host, error) { cryptoType = libp2pcrypto.ECDSA } - randomness := rand.Reader - if seed := cfg.Crypto.Seed; seed != 0 { - log.Warn("USING MANUAL SEED - !! WARNING !! INSECURE: ", seed) - randomness = mrand.New(mrand.NewSource(int64(seed))) + // check for existing key in ~/.orbis/keyfile + dirname, err := os.UserHomeDir() + if err != nil { + return nil, err } + var priv libp2pcrypto.PrivKey + path := filepath.Join(dirname, ".orbis", "keyfile") + _, err = os.Stat(path) + if err == nil { + // read + buf, err := os.ReadFile(path) + if err != nil { + return nil, fmt.Errorf("reading key file: %w", err) + } + priv, err = libp2pcrypto.UnmarshalPrivateKey(buf) + if err != nil { + return nil, fmt.Errorf("unmarshaling priv key: %w", err) + } + } else if errors.Is(err, os.ErrNotExist) { + // generate + randomness := rand.Reader + if seed := cfg.Crypto.Seed; seed != 0 { + log.Warn("USING MANUAL SEED - !! WARNING !! INSECURE: ", seed) + randomness = mrand.New(mrand.NewSource(int64(seed))) + } - priv, _, err := libp2pcrypto.GenerateKeyPairWithReader(cryptoType, cfg.Crypto.Bits, randomness) - if err != nil { - return nil, fmt.Errorf("generate key pair: %w", err) + priv, _, err = libp2pcrypto.GenerateKeyPairWithReader(cryptoType, cfg.Crypto.Bits, randomness) + if err != nil { + return nil, fmt.Errorf("generate key pair: %w", err) + } + // save new key + buf, err := libp2pcrypto.MarshalPrivateKey(priv) + if err != nil { + return nil, fmt.Errorf("marshaling priv key: %w", err) + } + err = os.WriteFile(path, buf, 0600) + if err != nil { + return nil, fmt.Errorf("writing priv key: %w", err) + } + } else { + // error out + return nil, fmt.Errorf("checking key file: %w", err) } cpriv, err := crypto.PrivateKeyFromLibP2P(priv)