mirror of
				https://github.com/containers/podman.git
				synced 2025-10-27 03:06:22 +08:00 
			
		
		
		
	 0177f74dc6
			
		
	
	0177f74dc6
	
	
	
		
			
			Read stderr from ssh-keygen before calling wait(), since cmd.Wait() closes cmd.StderrPipe() after it exits, causing a read-on-closed-pipe error. Signed-off-by: Ashley Cui <acui@redhat.com>
		
			
				
	
	
		
			119 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			119 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| //go:build amd64 || arm64
 | |
| 
 | |
| package machine
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"os"
 | |
| 	"os/exec"
 | |
| 	"path/filepath"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/containers/storage/pkg/fileutils"
 | |
| 	"github.com/sirupsen/logrus"
 | |
| )
 | |
| 
 | |
| var sshCommand = []string{"ssh-keygen", "-N", "", "-t", "ed25519", "-f"}
 | |
| 
 | |
| // CreateSSHKeys makes a priv and pub ssh key for interacting
 | |
| // the a VM.
 | |
| func CreateSSHKeys(writeLocation string) (string, error) {
 | |
| 	// If the SSH key already exists, hard fail
 | |
| 	if err := fileutils.Exists(writeLocation); err == nil {
 | |
| 		return "", fmt.Errorf("SSH key already exists: %s", writeLocation)
 | |
| 	}
 | |
| 	if err := os.MkdirAll(filepath.Dir(writeLocation), 0700); err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	if err := generatekeys(writeLocation); err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	b, err := os.ReadFile(writeLocation + ".pub")
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	return strings.TrimSuffix(string(b), "\n"), nil
 | |
| }
 | |
| 
 | |
| // GetSSHKeys checks to see if there is a ssh key at the provided location.
 | |
| // If not, we create the priv and pub keys. The ssh key is then returned.
 | |
| func GetSSHKeys(identityPath string) (string, error) {
 | |
| 	if err := fileutils.Exists(identityPath); err == nil {
 | |
| 		b, err := os.ReadFile(identityPath + ".pub")
 | |
| 		if err != nil {
 | |
| 			return "", err
 | |
| 		}
 | |
| 		return strings.TrimSuffix(string(b), "\n"), nil
 | |
| 	}
 | |
| 
 | |
| 	return CreateSSHKeys(identityPath)
 | |
| }
 | |
| 
 | |
| func CreateSSHKeysPrefix(identityPath string, passThru bool, skipExisting bool, prefix ...string) (string, error) {
 | |
| 	e := fileutils.Exists(identityPath)
 | |
| 	if !skipExisting || errors.Is(e, os.ErrNotExist) {
 | |
| 		if err := generatekeysPrefix(identityPath, passThru, prefix...); err != nil {
 | |
| 			return "", err
 | |
| 		}
 | |
| 	} else {
 | |
| 		fmt.Println("Keys already exist, reusing")
 | |
| 	}
 | |
| 	b, err := os.ReadFile(identityPath + ".pub")
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	return strings.TrimSuffix(string(b), "\n"), nil
 | |
| }
 | |
| 
 | |
| // generatekeys creates an ed25519 set of keys
 | |
| func generatekeys(writeLocation string) error {
 | |
| 	args := append(append([]string{}, sshCommand[1:]...), writeLocation)
 | |
| 	cmd := exec.Command(sshCommand[0], args...)
 | |
| 	stdErr := &bytes.Buffer{}
 | |
| 	cmd.Stderr = stdErr
 | |
| 
 | |
| 	if err := cmd.Start(); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	waitErr := cmd.Wait()
 | |
| 	if waitErr != nil {
 | |
| 		return fmt.Errorf("failed to generate keys: %s: %w", strings.TrimSpace(stdErr.String()), waitErr)
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // generatekeys creates an ed25519 set of keys
 | |
| func generatekeysPrefix(identityPath string, passThru bool, prefix ...string) error {
 | |
| 	dir := filepath.Dir(identityPath)
 | |
| 	file := filepath.Base(identityPath)
 | |
| 
 | |
| 	if err := os.MkdirAll(dir, 0700); err != nil {
 | |
| 		return fmt.Errorf("could not create ssh directory: %w", err)
 | |
| 	}
 | |
| 
 | |
| 	args := append([]string{}, prefix[1:]...)
 | |
| 	args = append(args, sshCommand...)
 | |
| 	args = append(args, file)
 | |
| 
 | |
| 	binary, err := exec.LookPath(prefix[0])
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	binary, err = filepath.Abs(binary)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	cmd := exec.Command(binary, args...)
 | |
| 	cmd.Dir = dir
 | |
| 	if passThru {
 | |
| 		cmd.Stdin = os.Stdin
 | |
| 		cmd.Stdout = os.Stdout
 | |
| 		cmd.Stderr = os.Stderr
 | |
| 	}
 | |
| 	logrus.Debugf("Running wsl cmd %v in dir: %s", args, dir)
 | |
| 	return cmd.Run()
 | |
| }
 |