mirror of
https://github.com/ipfs/kubo.git
synced 2025-08-03 01:38:00 +08:00

Updates: * go-kad-dht: Query performance improvements, DHT client fixes, validates records on *local* put. * go-libp2p-swarm/go-libp2p-transport: Timeout improvements. * go-multiaddr-net: Exposes useful Conn methods (CloseWrite, CloseRead, etc.) * go-log: fixes possible panic when enabling/disabling events. * go-multiaddr: fixes possible panic when stringifying malformed multiaddrs, adds support for consuming /p2p/ multiaddrs. fixes #5113 unblocks #4895 License: MIT Signed-off-by: Steven Allen <steven@stebalien.com>
203 lines
4.4 KiB
Go
203 lines
4.4 KiB
Go
package coreapi
|
|
|
|
import (
|
|
"context"
|
|
"crypto/rand"
|
|
"fmt"
|
|
"sort"
|
|
|
|
coreiface "github.com/ipfs/go-ipfs/core/coreapi/interface"
|
|
caopts "github.com/ipfs/go-ipfs/core/coreapi/interface/options"
|
|
ipfspath "github.com/ipfs/go-ipfs/path"
|
|
|
|
peer "gx/ipfs/QmdVrMn1LhB4ybb8hMVaMLXnA8XRSewMnK6YqXKXoTcRvN/go-libp2p-peer"
|
|
crypto "gx/ipfs/Qme1knMqwt1hKZbc1BmQFmnm9f36nyQGwXxPGVpVJ9rMK5/go-libp2p-crypto"
|
|
)
|
|
|
|
type KeyAPI CoreAPI
|
|
|
|
type key struct {
|
|
name string
|
|
peerId string
|
|
}
|
|
|
|
// Name returns the key name
|
|
func (k *key) Name() string {
|
|
return k.name
|
|
}
|
|
|
|
// Path returns the path of the key.
|
|
func (k *key) Path() coreiface.Path {
|
|
return &path{path: ipfspath.FromString(ipfspath.Join([]string{"/ipns", k.peerId}))}
|
|
}
|
|
|
|
// Generate generates new key, stores it in the keystore under the specified
|
|
// name and returns a base58 encoded multihash of its public key.
|
|
func (api *KeyAPI) Generate(ctx context.Context, name string, opts ...caopts.KeyGenerateOption) (coreiface.Key, error) {
|
|
options, err := caopts.KeyGenerateOptions(opts...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if name == "self" {
|
|
return nil, fmt.Errorf("cannot overwrite key with name 'self'")
|
|
}
|
|
|
|
_, err = api.node.Repo.Keystore().Get(name)
|
|
if err == nil {
|
|
return nil, fmt.Errorf("key with name '%s' already exists", name)
|
|
}
|
|
|
|
var sk crypto.PrivKey
|
|
var pk crypto.PubKey
|
|
|
|
switch options.Algorithm {
|
|
case "rsa":
|
|
if options.Size == -1 {
|
|
options.Size = caopts.DefaultRSALen
|
|
}
|
|
|
|
priv, pub, err := crypto.GenerateKeyPairWithReader(crypto.RSA, options.Size, rand.Reader)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
sk = priv
|
|
pk = pub
|
|
case "ed25519":
|
|
priv, pub, err := crypto.GenerateEd25519Key(rand.Reader)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
sk = priv
|
|
pk = pub
|
|
default:
|
|
return nil, fmt.Errorf("unrecognized key type: %s", options.Algorithm)
|
|
}
|
|
|
|
err = api.node.Repo.Keystore().Put(name, sk)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
pid, err := peer.IDFromPublicKey(pk)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &key{name, pid.Pretty()}, nil
|
|
}
|
|
|
|
// List returns a list keys stored in keystore.
|
|
func (api *KeyAPI) List(ctx context.Context) ([]coreiface.Key, error) {
|
|
keys, err := api.node.Repo.Keystore().List()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
sort.Strings(keys)
|
|
|
|
out := make([]coreiface.Key, len(keys)+1)
|
|
out[0] = &key{"self", api.node.Identity.Pretty()}
|
|
|
|
for n, k := range keys {
|
|
privKey, err := api.node.Repo.Keystore().Get(k)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
pubKey := privKey.GetPublic()
|
|
|
|
pid, err := peer.IDFromPublicKey(pubKey)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
out[n+1] = &key{k, pid.Pretty()}
|
|
}
|
|
return out, nil
|
|
}
|
|
|
|
// Rename renames `oldName` to `newName`. Returns the key and whether another
|
|
// key was overwritten, or an error.
|
|
func (api *KeyAPI) Rename(ctx context.Context, oldName string, newName string, opts ...caopts.KeyRenameOption) (coreiface.Key, bool, error) {
|
|
options, err := caopts.KeyRenameOptions(opts...)
|
|
if err != nil {
|
|
return nil, false, err
|
|
}
|
|
|
|
ks := api.node.Repo.Keystore()
|
|
|
|
if oldName == "self" {
|
|
return nil, false, fmt.Errorf("cannot rename key with name 'self'")
|
|
}
|
|
|
|
if newName == "self" {
|
|
return nil, false, fmt.Errorf("cannot overwrite key with name 'self'")
|
|
}
|
|
|
|
oldKey, err := ks.Get(oldName)
|
|
if err != nil {
|
|
return nil, false, fmt.Errorf("no key named %s was found", oldName)
|
|
}
|
|
|
|
pubKey := oldKey.GetPublic()
|
|
|
|
pid, err := peer.IDFromPublicKey(pubKey)
|
|
if err != nil {
|
|
return nil, false, err
|
|
}
|
|
|
|
overwrite := false
|
|
if options.Force {
|
|
exist, err := ks.Has(newName)
|
|
if err != nil {
|
|
return nil, false, err
|
|
}
|
|
|
|
if exist {
|
|
overwrite = true
|
|
err := ks.Delete(newName)
|
|
if err != nil {
|
|
return nil, false, err
|
|
}
|
|
}
|
|
}
|
|
|
|
err = ks.Put(newName, oldKey)
|
|
if err != nil {
|
|
return nil, false, err
|
|
}
|
|
|
|
return &key{newName, pid.Pretty()}, overwrite, ks.Delete(oldName)
|
|
}
|
|
|
|
// Remove removes keys from keystore. Returns ipns path of the removed key.
|
|
func (api *KeyAPI) Remove(ctx context.Context, name string) (coreiface.Path, error) {
|
|
ks := api.node.Repo.Keystore()
|
|
|
|
if name == "self" {
|
|
return nil, fmt.Errorf("cannot remove key with name 'self'")
|
|
}
|
|
|
|
removed, err := ks.Get(name)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("no key named %s was found", name)
|
|
}
|
|
|
|
pubKey := removed.GetPublic()
|
|
|
|
pid, err := peer.IDFromPublicKey(pubKey)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
err = ks.Delete(name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return (&key{"", pid.Pretty()}).Path(), nil
|
|
}
|