newssh

package util

import (
"bytes"
"errors"
"fmt"
"github.com/sirupsen/logrus"
"golang.org/x/crypto/ssh"
"strings"
"time"
)

func publicKeyAuthFunc(pemBytes, keyPassword []byte) ssh.AuthMethod {
// Create the Signer for this private key.
signer, err := ssh.ParsePrivateKeyWithPassphrase(pemBytes, keyPassword)
if err != nil {
logrus.WithError(err).Error("parse ssh key from bytes failed")
return nil
}
return ssh.PublicKeys(signer)
}

func SshRemoteRunCommand(sshClient *ssh.Client, command string) (string, error) {
session, err := sshClient.NewSession()
if err != nil {
return "", err
}
defer session.Close()
var buf bytes.Buffer
session.Stdout = &buf
err = session.Run(command)
logString := buf.String()
//logrus.WithField("CMD:", command).Info(logString)
if err != nil {
return logString, fmt.Errorf("CMD: %s OUT: %s ERROR: %s", command, logString, err)
}
return logString, err
}
func NewSshClientConfig(sshUser, sshPassword, sshType, sshKey, sshKeyPassword string) (config *ssh.ClientConfig, err error) {
if sshUser == "" {
return nil, errors.New("ssh_user can not be empty")
}
config = &ssh.ClientConfig{
Timeout: time.Second * 3,
User: sshUser,
HostKeyCallback: ssh.InsecureIgnoreHostKey(), //这个可以, 但是不够安全
//HostKeyCallback: hostKeyCallBackFunc(h.Host),
}
switch sshType {
case "password":
config.Auth = []ssh.AuthMethod{ssh.Password(sshPassword)}
case "key":
config.Auth = []ssh.AuthMethod{publicKeyAuthFunc([]byte(sshKey), []byte(sshKeyPassword))}
default:
return nil, fmt.Errorf("unknow ssh auth type: %s", sshType)
}
return
}
func CreateSimpleSshClient(sshUser, sshPassword, sshAddr string) (*ssh.Client, error) {
if !strings.Contains(sshAddr, ":") {
sshAddr = fmt.Sprintf("%s:22", sshAddr)
}

targetConfig, err := NewSshClientConfig(sshUser, sshPassword, "password", "", "")
if err != nil {
return nil, fmt.Errorf("cluster jumper proxy ssh config failed:%s", err)
}
return ssh.Dial("tcp", sshAddr, targetConfig)
}



package model

import (
"errors"
"fmt"
"golang.org/x/crypto/ssh"
"sshfortress/util"
)

func CreateSshClientAsAdmin(m *Machine, cj *ClusterJumper, cs *ClusterSsh) (c *ssh.Client, err error) {
if m.Id < 1 {
return nil, errors.New("CreateSshClientAsAdmin m is not valid")
}
targetConfig, err := util.NewSshClientConfig(cs.SshUser, cs.SshPassword, cs.SshType, cs.SshKey, cs.SshKeyPassword)
if err != nil {
return nil, fmt.Errorf("cluster jumper proxy ssh config failed:%s", err)
}
targetAddr := fmt.Sprintf("%s:%d", m.SshIp, m.SshPort)

var proxyConfig *ssh.ClientConfig
var proxyAddr string
if cj != nil && cj.Id > 0 {
//使用私有云集群跳板登陆
proxyConfig, err = util.NewSshClientConfig(cj.SshUser, cj.SshPassword, cj.SshType, cj.SshKey, cj.SshKeyPassword)
if err != nil {
return nil, fmt.Errorf("cluster jumper proxy ssh config failed:%s", err)
}
proxyAddr = fmt.Sprintf("%s:%d", cj.SshAddr, cj.SshPort)
}
c, err = createSshProxySshClient(targetConfig, proxyConfig, targetAddr, proxyAddr)
return
}

func CreateSshClientAsUser(m *Machine, user *User, cj *ClusterJumper) (c *ssh.Client, err error) {
targetConfig, err := util.NewSshClientConfig(user.SshUserName(), user.SshPassword, "password", "", "")
if err != nil {
return nil, fmt.Errorf("cluster jumper proxy ssh config failed:%s", err)
}
targetAddr := fmt.Sprintf("%s:%d", m.SshIp, m.SshPort)

var proxyConfig *ssh.ClientConfig
var proxyAddr string
if cj != nil && cj.Id > 0 {
//使用私有云集群跳板登陆
proxyConfig, err = util.NewSshClientConfig(cj.SshUser, cj.SshPassword, cj.SshType, cj.SshKey, cj.SshKeyPassword)
if err != nil {
return nil, fmt.Errorf("cluster jumper proxy ssh config failed:%s", err)
}
proxyAddr = fmt.Sprintf("%s:%d", cj.SshAddr, cj.SshPort)
}
c, err = createSshProxySshClient(targetConfig, proxyConfig, targetAddr, proxyAddr)
return
}

func createSshProxySshClient(targetSshConfig, proxySshConfig *ssh.ClientConfig, targetAddr, proxyAddr string) (client *ssh.Client, err error) {
if proxySshConfig == nil {
return ssh.Dial("tcp", targetAddr, targetSshConfig)
}

proxyClient, err := ssh.Dial("tcp", proxyAddr, proxySshConfig)
if err != nil {
return
}
conn, err := proxyClient.Dial("tcp", targetAddr)
if err != nil {
return
}
ncc, chans, reqs, err := ssh.NewClientConn(conn, targetAddr, targetSshConfig)
if err != nil {
return
}
client = ssh.NewClient(ncc, chans, reqs)
return
}




posted @ 2023-05-15 22:13  技术颜良  阅读(24)  评论(0编辑  收藏  举报