Skip to content

Instantly share code, notes, and snippets.

@gorsuch
Created October 24, 2012 15:26
Show Gist options
  • Save gorsuch/3946737 to your computer and use it in GitHub Desktop.
Save gorsuch/3946737 to your computer and use it in GitHub Desktop.
cargo-culted sshd
// largely cargo-culted from http://go.pkgdoc.org/code.google.com/p/go.crypto/ssh#_example_Listen
package main
import "fmt"
import "io/ioutil"
import "code.google.com/p/go.crypto/ssh"
import "code.google.com/p/go.crypto/ssh/terminal"
func main() {
// An SSH server is represented by a ServerConfig, which holds
// certificate details and handles authentication of ServerConns.
config := &ssh.ServerConfig{
PasswordCallback: func(conn *ssh.ServerConn, user, pass string) bool {
return user == "testuser" && pass == "tiger"
},
}
pemBytes, err := ioutil.ReadFile("id_rsa")
if err != nil {
panic("Failed to load private key")
}
if err = config.SetRSAPrivateKey(pemBytes); err != nil {
panic("Failed to parse private key")
}
// Once a ServerConfig has been configured, connections can be
// accepted.
listener, err := ssh.Listen("tcp", "0.0.0.0:2022", config)
if err != nil {
panic("failed to listen for connection")
}
for {
sConn, err := listener.Accept()
if err != nil {
panic("failed to accept incoming connection")
}
if err := sConn.Handshake(); err != nil {
panic("failed to handshake")
}
go func() {
// A ServerConn multiplexes several channels, which must
// themselves be Accepted.
for {
// Accept reads from the connection, demultiplexes packets
// to their corresponding channels and returns when a new
// channel request is seen. Some goroutine must always be
// calling Accept; otherwise no messages will be forwarded
// to the channels.
channel, err := sConn.Accept()
if err != nil {
// currently treating errors hear as a disconnected session
// this needs to be re-evaluated
return
}
// Channels have a type, depending on the application level
// protocol intended. In the case of a shell, the type is
// "session" and ServerShell may be used to present a simple
// terminal interface.
if channel.ChannelType() != "session" {
channel.Reject(ssh.UnknownChannelType, "unknown channel type")
continue
}
channel.Accept()
term := terminal.NewTerminal(channel, "> ")
serverTerm := &ssh.ServerTerminal{
Term: term,
Channel: channel,
}
go func() {
defer channel.Close()
for {
line, err := serverTerm.ReadLine()
if err != nil {
break
}
fmt.Println(line)
}
}()
}
}()
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment