goldwarden-vaultwarden-bitw.../agent/ssh/ssh.go

221 lines
6.2 KiB
Go
Raw Normal View History

2023-07-17 03:23:26 +02:00
package ssh
import (
"bytes"
"crypto/rand"
"errors"
"fmt"
2024-01-19 05:14:25 +01:00
"time"
2023-07-17 03:23:26 +02:00
2023-09-19 21:49:56 +02:00
"github.com/quexten/goldwarden/agent/config"
"github.com/quexten/goldwarden/agent/notify"
2023-07-17 03:23:26 +02:00
"github.com/quexten/goldwarden/agent/sockets"
2023-09-12 18:56:35 +02:00
"github.com/quexten/goldwarden/agent/systemauth"
2023-09-12 02:54:46 +02:00
"github.com/quexten/goldwarden/agent/systemauth/pinentry"
2023-07-17 03:23:26 +02:00
"github.com/quexten/goldwarden/agent/vault"
2023-08-21 18:37:34 +02:00
"github.com/quexten/goldwarden/logging"
2023-07-17 03:23:26 +02:00
"golang.org/x/crypto/ssh"
"golang.org/x/crypto/ssh/agent"
)
2023-08-21 18:37:34 +02:00
var log = logging.GetLogger("Goldwarden", "SSH")
2023-07-17 03:23:26 +02:00
type vaultAgent struct {
vault *vault.Vault
2023-09-19 21:49:56 +02:00
config *config.Config
2023-07-17 03:23:26 +02:00
unlockRequestAction func() bool
context sockets.CallingContext
}
func (vaultAgent) Add(key agent.AddedKey) error {
2024-01-19 07:37:06 +01:00
log.Warn("Add Request - Not implemented")
2023-07-17 03:23:26 +02:00
return nil
}
func (vaultAgent vaultAgent) List() ([]*agent.Key, error) {
2024-01-19 07:37:06 +01:00
log.Info("List Request")
2023-07-17 03:23:26 +02:00
if vaultAgent.vault.Keyring.IsLocked() {
if !vaultAgent.unlockRequestAction() {
2024-01-19 07:37:06 +01:00
log.Warn("List request failed - Vault is locked")
2023-07-17 03:23:26 +02:00
return nil, errors.New("vault is locked")
}
2023-09-19 22:13:11 +02:00
2024-02-09 19:32:01 +01:00
systemauth.CreatePinSession(vaultAgent.context, systemauth.SSHTTL)
2023-07-17 03:23:26 +02:00
}
vaultSSHKeys := (*vaultAgent.vault).GetSSHKeys()
var sshKeys []*agent.Key
for _, vaultSSHKey := range vaultSSHKeys {
signer, err := ssh.ParsePrivateKey([]byte(vaultSSHKey.Key))
if err != nil {
2024-01-19 07:37:06 +01:00
log.Warn("List request key skipped - Could not parse key: %s", err)
2023-07-17 03:23:26 +02:00
continue
}
pub := signer.PublicKey()
sshKeys = append(sshKeys, &agent.Key{
Format: pub.Type(),
Blob: pub.Marshal(),
Comment: vaultSSHKey.Name})
}
return sshKeys, nil
}
func (vaultAgent) Lock(passphrase []byte) error {
2024-01-19 07:37:06 +01:00
log.Warn("Lock Request - Not implemented")
2023-07-17 03:23:26 +02:00
return nil
}
func (vaultAgent) Remove(key ssh.PublicKey) error {
2024-01-19 07:37:06 +01:00
log.Warn("Remove Request - Not implemented")
2023-07-17 03:23:26 +02:00
return nil
}
func (vaultAgent) RemoveAll() error {
2024-01-19 07:37:06 +01:00
log.Warn("RemoveAll Request - Not implemented")
2023-07-17 03:23:26 +02:00
return nil
}
func Eq(a, b ssh.PublicKey) bool {
return 0 == bytes.Compare(a.Marshal(), b.Marshal())
}
func (vaultAgent vaultAgent) Sign(key ssh.PublicKey, data []byte) (*ssh.Signature, error) {
return vaultAgent.SignWithFlags(key, data, agent.SignatureFlagReserved)
}
func (vaultAgent vaultAgent) SignWithFlags(key ssh.PublicKey, data []byte, flags agent.SignatureFlags) (*ssh.Signature, error) {
2023-07-17 03:23:26 +02:00
log.Info("Sign Request for key: %s", ssh.FingerprintSHA256(key))
if vaultAgent.vault.Keyring.IsLocked() {
if !vaultAgent.unlockRequestAction() {
return nil, errors.New("vault is locked")
}
2023-09-12 18:56:35 +02:00
2024-02-09 19:32:01 +01:00
systemauth.CreatePinSession(vaultAgent.context, systemauth.SSHTTL)
2023-07-17 03:23:26 +02:00
}
var signer ssh.Signer
var sshKey *vault.SSHKey
vaultSSHKeys := (*vaultAgent.vault).GetSSHKeys()
for _, vaultSSHKey := range vaultSSHKeys {
sg, err := ssh.ParsePrivateKey([]byte(vaultSSHKey.Key))
if err != nil {
return nil, err
}
if Eq(sg.PublicKey(), key) {
signer = sg
sshKey = &vaultSSHKey
break
}
}
2023-12-28 13:11:00 +01:00
isGit := false
magicHeader := []byte("SSHSIG\x00\x00\x00\x03git")
if bytes.HasPrefix(data, magicHeader) {
isGit = true
}
2024-01-19 07:32:41 +01:00
requestTemplate := ""
message := ""
if !vaultAgent.context.Error {
if isGit {
requestTemplate = "%s on %s>%s>%s is requesting git signage with key %s"
} else {
requestTemplate = "%s on %s>%s>%s is requesting ssh signage with key %s"
}
message = fmt.Sprintf(requestTemplate, vaultAgent.context.UserName, vaultAgent.context.GrandParentProcessName, vaultAgent.context.ParentProcessName, vaultAgent.context.ProcessName, sshKey.Name)
} else {
if isGit {
requestTemplate = "%s is requesting git signage with key %s"
} else {
requestTemplate = "%s is requesting ssh signage with key %s"
}
message = fmt.Sprintf(requestTemplate, vaultAgent.context.UserName, sshKey.Name)
2023-12-28 13:11:00 +01:00
}
2023-07-17 03:23:26 +02:00
2024-02-12 17:41:23 +01:00
// todo refactor
if !systemauth.GetSSHSession(vaultAgent.context) {
if approved, err := pinentry.GetApproval("SSH Key Signing Request", message); err != nil || !approved {
log.Info("Sign Request for key: %s denied", sshKey.Name)
return nil, errors.New("Approval not given")
}
2023-07-17 03:23:26 +02:00
2024-02-12 17:41:23 +01:00
if !systemauth.VerifyPinSession(vaultAgent.context) {
if permission, err := systemauth.GetPermission(systemauth.SSHKey, vaultAgent.context, vaultAgent.config); err != nil || !permission {
log.Info("Sign Request for key: %s denied", key.Marshal())
return nil, errors.New("Biometrics not checked")
}
}
systemauth.CreateSSHSession(vaultAgent.context)
} else {
log.Info("Using cached session approval")
2023-07-17 03:23:26 +02:00
}
var rand = rand.Reader
log.Info("Sign Request for key: %s %s accepted", ssh.FingerprintSHA256(key), sshKey.Name)
2023-12-28 13:11:00 +01:00
if isGit {
2024-01-19 05:14:25 +01:00
notify.Notify("Goldwarden", fmt.Sprintf("Git Signing Request Approved for %s", sshKey.Name), "", 10*time.Second, func() {})
2023-12-28 13:11:00 +01:00
} else {
2024-01-19 05:14:25 +01:00
notify.Notify("Goldwarden", fmt.Sprintf("SSH Signing Request Approved for %s", sshKey.Name), "", 10*time.Second, func() {})
2023-12-28 13:11:00 +01:00
}
algo := ""
switch flags {
case agent.SignatureFlagRsaSha256:
algo = ssh.KeyAlgoRSASHA256
case agent.SignatureFlagRsaSha512:
algo = ssh.KeyAlgoRSASHA512
default:
return signer.Sign(rand, data)
}
log.Info("%s algorithm requested", algo)
algoSigner, err := ssh.NewSignerWithAlgorithms(signer.(ssh.AlgorithmSigner), []string{algo})
if err != nil {
return nil, err
}
return algoSigner.SignWithAlgorithm(rand, data, algo)
}
func (vaultAgent) Extension(extensionType string, contents []byte) ([]byte, error) {
return nil, agent.ErrExtensionUnsupported
2023-07-17 03:23:26 +02:00
}
func (vaultAgent) Signers() ([]ssh.Signer, error) {
2024-01-19 07:37:06 +01:00
log.Warn("Signers Request - Not implemented")
2023-07-17 03:23:26 +02:00
return []ssh.Signer{}, nil
}
func (vaultAgent) Unlock(passphrase []byte) error {
2024-01-19 07:37:06 +01:00
log.Warn("Unlock Request - Not implemented")
2023-07-17 03:23:26 +02:00
return nil
}
type SSHAgentServer struct {
vault *vault.Vault
2023-09-19 21:49:56 +02:00
config *config.Config
2023-12-30 18:53:01 +01:00
runtimeConfig *config.RuntimeConfig
2023-07-17 03:23:26 +02:00
unlockRequestAction func() bool
}
func (v *SSHAgentServer) SetUnlockRequestAction(action func() bool) {
v.unlockRequestAction = action
}
2023-12-30 18:53:01 +01:00
func NewVaultAgent(vault *vault.Vault, config *config.Config, runtimeConfig *config.RuntimeConfig) SSHAgentServer {
2023-07-17 03:23:26 +02:00
return SSHAgentServer{
2023-12-30 18:53:01 +01:00
vault: vault,
config: config,
runtimeConfig: runtimeConfig,
2023-07-17 03:23:26 +02:00
unlockRequestAction: func() bool {
log.Info("Unlock Request, but no action defined")
return false
},
}
}