123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255 |
- // Copyright 2014 The Go Authors. All rights reserved.
- // Use of this source code is governed by a BSD-style
- // license that can be found in the LICENSE file.
- package main
- import (
- "bytes"
- "crypto/rand"
- "crypto/subtle"
- "errors"
- "fmt"
- "sync"
- "time"
- "golang.org/x/crypto/ssh"
- "golang.org/x/crypto/ssh/agent"
- )
- type privKey struct {
- signer ssh.Signer
- comment string
- confirm bool
- expire *time.Time
- }
- type ConfirmFunction func(comment string) bool
- type keyring struct {
- confirmFunction ConfirmFunction
- mu sync.Mutex
- keys []privKey
- locked bool
- passphrase []byte
- }
- var errLocked = errors.New("agent: locked")
- // NewKeyring returns an Agent that holds keys in memory. It is safe
- // for concurrent use by multiple goroutines.
- func NewKeyring(confirmFunction ConfirmFunction) agent.ExtendedAgent {
- return &keyring{
- confirmFunction: confirmFunction,
- }
- }
- // RemoveAll removes all identities.
- func (r *keyring) RemoveAll() error {
- r.mu.Lock()
- defer r.mu.Unlock()
- if r.locked {
- return errLocked
- }
- r.keys = nil
- return nil
- }
- // removeLocked does the actual key removal. The caller must already be holding the
- // keyring mutex.
- func (r *keyring) removeLocked(want []byte) error {
- found := false
- for i := 0; i < len(r.keys); {
- if bytes.Equal(r.keys[i].signer.PublicKey().Marshal(), want) {
- found = true
- r.keys[i] = r.keys[len(r.keys)-1]
- r.keys = r.keys[:len(r.keys)-1]
- continue
- } else {
- i++
- }
- }
- if !found {
- return errors.New("agent: key not found")
- }
- return nil
- }
- // Remove removes all identities with the given public key.
- func (r *keyring) Remove(key ssh.PublicKey) error {
- r.mu.Lock()
- defer r.mu.Unlock()
- if r.locked {
- return errLocked
- }
- return r.removeLocked(key.Marshal())
- }
- // Lock locks the agent. Sign and Remove will fail, and List will return an empty list.
- func (r *keyring) Lock(passphrase []byte) error {
- r.mu.Lock()
- defer r.mu.Unlock()
- if r.locked {
- return errLocked
- }
- r.locked = true
- r.passphrase = passphrase
- return nil
- }
- // Unlock undoes the effect of Lock
- func (r *keyring) Unlock(passphrase []byte) error {
- r.mu.Lock()
- defer r.mu.Unlock()
- if !r.locked {
- return errors.New("agent: not locked")
- }
- if 1 != subtle.ConstantTimeCompare(passphrase, r.passphrase) {
- return fmt.Errorf("agent: incorrect passphrase")
- }
- r.locked = false
- r.passphrase = nil
- return nil
- }
- // expireKeysLocked removes expired keys from the keyring. If a key was added
- // with a lifetimesecs contraint and seconds >= lifetimesecs seconds have
- // elapsed, it is removed. The caller *must* be holding the keyring mutex.
- func (r *keyring) expireKeysLocked() {
- for _, k := range r.keys {
- if k.expire != nil && time.Now().After(*k.expire) {
- r.removeLocked(k.signer.PublicKey().Marshal())
- }
- }
- }
- // List returns the identities known to the agent.
- func (r *keyring) List() ([]*agent.Key, error) {
- r.mu.Lock()
- defer r.mu.Unlock()
- if r.locked {
- // section 2.7: locked agents return empty.
- return nil, nil
- }
- r.expireKeysLocked()
- var ids []*agent.Key
- for _, k := range r.keys {
- pub := k.signer.PublicKey()
- ids = append(ids, &agent.Key{
- Format: pub.Type(),
- Blob: pub.Marshal(),
- Comment: k.comment})
- }
- return ids, nil
- }
- // Insert adds a private key to the keyring. If a certificate
- // is given, that certificate is added as public key. Note that
- // any constraints given are ignored.
- func (r *keyring) Add(key agent.AddedKey) error {
- r.mu.Lock()
- defer r.mu.Unlock()
- if r.locked {
- return errLocked
- }
- signer, err := ssh.NewSignerFromKey(key.PrivateKey)
- if err != nil {
- return err
- }
- if cert := key.Certificate; cert != nil {
- signer, err = ssh.NewCertSigner(cert, signer)
- if err != nil {
- return err
- }
- }
- p := privKey{
- signer: signer,
- comment: key.Comment,
- confirm: key.ConfirmBeforeUse,
- }
- if key.LifetimeSecs > 0 {
- t := time.Now().Add(time.Duration(key.LifetimeSecs) * time.Second)
- p.expire = &t
- }
- r.keys = append(r.keys, p)
- return nil
- }
- // Sign returns a signature for the data.
- func (r *keyring) Sign(key ssh.PublicKey, data []byte) (*ssh.Signature, error) {
- return r.SignWithFlags(key, data, 0)
- }
- func (r *keyring) SignWithFlags(key ssh.PublicKey, data []byte, flags agent.SignatureFlags) (*ssh.Signature, error) {
- r.mu.Lock()
- defer r.mu.Unlock()
- if r.locked {
- return nil, errLocked
- }
- r.expireKeysLocked()
- wanted := key.Marshal()
- for _, k := range r.keys {
- if bytes.Equal(k.signer.PublicKey().Marshal(), wanted) {
- if k.confirm {
- if !r.confirmFunction(k.comment) {
- return nil, fmt.Errorf("agent: confirmation failed")
- }
- }
- if flags == 0 {
- return k.signer.Sign(rand.Reader, data)
- } else {
- if algorithmSigner, ok := k.signer.(ssh.AlgorithmSigner); !ok {
- return nil, fmt.Errorf("agent: signature does not support non-default signature algorithm: %T", k.signer)
- } else {
- var algorithm string
- switch flags {
- case agent.SignatureFlagRsaSha256:
- algorithm = ssh.KeyAlgoRSASHA256
- case agent.SignatureFlagRsaSha512:
- algorithm = ssh.KeyAlgoRSASHA512
- default:
- return nil, fmt.Errorf("agent: unsupported signature flags: %d", flags)
- }
- return algorithmSigner.SignWithAlgorithm(rand.Reader, data, algorithm)
- }
- }
- }
- }
- return nil, errors.New("not found")
- }
- // Signers returns signers for all the known keys.
- func (r *keyring) Signers() ([]ssh.Signer, error) {
- r.mu.Lock()
- defer r.mu.Unlock()
- if r.locked {
- return nil, errLocked
- }
- r.expireKeysLocked()
- s := make([]ssh.Signer, 0, len(r.keys))
- for _, k := range r.keys {
- s = append(s, k.signer)
- }
- return s, nil
- }
- // The keyring does not support any extensions
- func (r *keyring) Extension(extensionType string, contents []byte) ([]byte, error) {
- return nil, agent.ErrExtensionUnsupported
- }
|