mirror of
https://github.com/thomiceli/opengist.git
synced 2025-01-10 10:12:39 +00:00
154 lines
3.5 KiB
Go
154 lines
3.5 KiB
Go
|
package ssh
|
||
|
|
||
|
import (
|
||
|
"errors"
|
||
|
"github.com/rs/zerolog/log"
|
||
|
"golang.org/x/crypto/ssh"
|
||
|
"gorm.io/gorm"
|
||
|
"io"
|
||
|
"net"
|
||
|
"opengist/internal/config"
|
||
|
"opengist/internal/models"
|
||
|
"os"
|
||
|
"os/exec"
|
||
|
"path/filepath"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
"syscall"
|
||
|
)
|
||
|
|
||
|
func Start() {
|
||
|
if !config.C.SSH.Enabled {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
sshConfig := &ssh.ServerConfig{
|
||
|
PublicKeyCallback: func(conn ssh.ConnMetadata, key ssh.PublicKey) (*ssh.Permissions, error) {
|
||
|
pkey, err := models.GetSSHKeyByContent(strings.TrimSpace(string(ssh.MarshalAuthorizedKey(key))))
|
||
|
if err != nil && !errors.Is(err, gorm.ErrRecordNotFound) {
|
||
|
return nil, err
|
||
|
}
|
||
|
return &ssh.Permissions{Extensions: map[string]string{"key-id": strconv.Itoa(int(pkey.ID))}}, nil
|
||
|
},
|
||
|
}
|
||
|
|
||
|
key, err := setupHostKey()
|
||
|
if err != nil {
|
||
|
log.Fatal().Err(err).Msg("SSH: Could not setup host key")
|
||
|
}
|
||
|
|
||
|
sshConfig.AddHostKey(key)
|
||
|
go listen(sshConfig)
|
||
|
}
|
||
|
|
||
|
func listen(serverConfig *ssh.ServerConfig) {
|
||
|
log.Info().Msg("Starting SSH server on ssh://" + config.C.SSH.Host + ":" + config.C.SSH.Port)
|
||
|
listener, err := net.Listen("tcp", config.C.SSH.Host+":"+config.C.SSH.Port)
|
||
|
if err != nil {
|
||
|
log.Fatal().Err(err).Msg("SSH: Failed to start SSH server")
|
||
|
}
|
||
|
defer listener.Close()
|
||
|
|
||
|
for {
|
||
|
nConn, err := listener.Accept()
|
||
|
if err != nil {
|
||
|
errorSsh("Failed to accept incoming connection", err)
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
go func() {
|
||
|
sConn, channels, reqs, err := ssh.NewServerConn(nConn, serverConfig)
|
||
|
if err != nil {
|
||
|
if !(err != io.EOF && !errors.Is(err, syscall.ECONNRESET)) {
|
||
|
errorSsh("Failed to handshake", err)
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
go ssh.DiscardRequests(reqs)
|
||
|
keyID, _ := strconv.Atoi(sConn.Permissions.Extensions["key-id"])
|
||
|
go handleConnexion(channels, uint(keyID))
|
||
|
}()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func handleConnexion(channels <-chan ssh.NewChannel, keyID uint) {
|
||
|
for channel := range channels {
|
||
|
if channel.ChannelType() != "session" {
|
||
|
_ = channel.Reject(ssh.UnknownChannelType, "Unknown channel type")
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
ch, reqs, err := channel.Accept()
|
||
|
if err != nil {
|
||
|
errorSsh("Could not accept channel", err)
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
go func(in <-chan *ssh.Request) {
|
||
|
defer func() {
|
||
|
_ = ch.Close()
|
||
|
}()
|
||
|
for req := range in {
|
||
|
switch req.Type {
|
||
|
case "env":
|
||
|
|
||
|
case "shell":
|
||
|
_, _ = ch.Write([]byte("Successfully connected to Opengist SSH server.\r\n"))
|
||
|
_, _ = ch.SendRequest("exit-status", false, []byte{0, 0, 0, 0})
|
||
|
return
|
||
|
case "exec":
|
||
|
payloadCmd := string(req.Payload)
|
||
|
i := strings.Index(payloadCmd, "git")
|
||
|
if i != -1 {
|
||
|
payloadCmd = payloadCmd[i:]
|
||
|
}
|
||
|
|
||
|
if err = runGitCommand(ch, payloadCmd, keyID); err != nil {
|
||
|
_, _ = ch.Stderr().Write([]byte("Opengist: " + err.Error() + "\r\n"))
|
||
|
}
|
||
|
_, _ = ch.SendRequest("exit-status", false, []byte{0, 0, 0, 0})
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
}(reqs)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func setupHostKey() (ssh.Signer, error) {
|
||
|
dir := filepath.Join(config.GetHomeDir(), "ssh")
|
||
|
|
||
|
if err := os.MkdirAll(dir, 0755); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
keyPath := filepath.Join(dir, "opengist-ed25519")
|
||
|
if _, err := os.Stat(keyPath); err != nil && !os.IsExist(err) {
|
||
|
cmd := exec.Command(config.C.SSH.Keygen,
|
||
|
"-t", "ssh-ed25519",
|
||
|
"-f", keyPath,
|
||
|
"-m", "PEM",
|
||
|
"-N", "")
|
||
|
err = cmd.Run()
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
keyData, err := os.ReadFile(keyPath)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
signer, err := ssh.ParsePrivateKey(keyData)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return signer, nil
|
||
|
}
|
||
|
|
||
|
func errorSsh(message string, err error) {
|
||
|
log.Error().Err(err).Msg("SSH: " + message)
|
||
|
}
|