in google_guest_agent/instance_setup.go [232:311]
func generateSSHKeys(ctx context.Context) error {
config := cfg.Get()
hostKeyDir := config.InstanceSetup.HostKeyDir
dir, err := os.Open(hostKeyDir)
if err != nil {
return err
}
defer dir.Close()
files, err := dir.Readdirnames(0)
if err != nil {
return err
}
keytypes := make(map[string]bool)
// Find keys present on disk, and deduce their type from filename.
prefix := "ssh_host_"
suffix := "_key"
for _, file := range files {
if strings.HasPrefix(file, prefix) && strings.HasSuffix(file, suffix) && len(file) > len(prefix+suffix) {
keytype := file
keytype = strings.TrimPrefix(keytype, prefix)
keytype = strings.TrimSuffix(keytype, suffix)
keytypes[keytype] = true
}
}
// List keys we should generate, according to the config.
configKeys := config.InstanceSetup.HostKeyTypes
for _, keytype := range strings.Split(configKeys, ",") {
keytypes[keytype] = true
}
// Generate new keys and upload to guest attributes.
for keytype := range keytypes {
keyfile := fmt.Sprintf("%s/ssh_host_%s_key", hostKeyDir, keytype)
if err := run.Quiet(ctx, "ssh-keygen", "-t", keytype, "-f", keyfile+".temp", "-N", "", "-q"); err != nil {
logger.Warningf("Failed to generate SSH host key %q: %v", keyfile, err)
continue
}
if err := os.Chmod(keyfile+".temp", 0600); err != nil {
logger.Errorf("Failed to chmod SSH host key %q: %v", keyfile, err)
continue
}
if err := os.Chmod(keyfile+".temp.pub", 0644); err != nil {
logger.Errorf("Failed to chmod SSH host key %q: %v", keyfile+".pub", err)
continue
}
if err := os.Rename(keyfile+".temp", keyfile); err != nil {
logger.Errorf("Failed to overwrite %q: %v", keyfile, err)
continue
}
if err := os.Rename(keyfile+".temp.pub", keyfile+".pub"); err != nil {
logger.Errorf("Failed to overwrite %q: %v", keyfile+".pub", err)
continue
}
pubKey, err := os.ReadFile(keyfile + ".pub")
if err != nil {
logger.Errorf("Can't read %s public key: %v", keytype, err)
continue
}
if vals := strings.Split(string(pubKey), " "); len(vals) >= 2 {
if err := mdsClient.WriteGuestAttributes(ctx, "hostkeys/"+vals[0], vals[1]); err != nil {
logger.Errorf("Failed to upload %s key to guest attributes: %v", keytype, err)
}
} else {
logger.Warningf("Generated key is malformed, not uploading")
}
}
_, err = exec.LookPath("restorecon")
if err == nil {
if err := run.Quiet(ctx, "restorecon", "-FR", hostKeyDir); err != nil {
return fmt.Errorf("failed to restore SELinux context for: %s", hostKeyDir)
}
}
return nil
}