mirror of
https://github.com/moul/sshportal.git
synced 2025-01-22 23:39:24 +08:00
88 lines
2 KiB
Go
88 lines
2 KiB
Go
package bastion // import "moul.io/sshportal/pkg/bastion"
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"time"
|
|
|
|
"github.com/gliderlabs/ssh"
|
|
oi "github.com/reiver/go-oi"
|
|
telnet "github.com/reiver/go-telnet"
|
|
"moul.io/sshportal/pkg/dbmodels"
|
|
)
|
|
|
|
type bastionTelnetCaller struct {
|
|
ssh ssh.Session
|
|
}
|
|
|
|
func (caller bastionTelnetCaller) CallTELNET(ctx telnet.Context, w telnet.Writer, r telnet.Reader) {
|
|
go func(writer io.Writer, reader io.Reader) {
|
|
var buffer [1]byte // Seems like the length of the buffer needs to be small, otherwise will have to wait for buffer to fill up.
|
|
p := buffer[:]
|
|
|
|
for {
|
|
// Read 1 byte.
|
|
n, err := reader.Read(p)
|
|
if n <= 0 && err == nil {
|
|
continue
|
|
} else if n <= 0 && err != nil {
|
|
break
|
|
}
|
|
|
|
if _, err = oi.LongWrite(writer, p); err != nil {
|
|
log.Printf("telnet longwrite failed: %v", err)
|
|
}
|
|
}
|
|
}(caller.ssh, r)
|
|
|
|
var buffer bytes.Buffer
|
|
var p []byte
|
|
|
|
var crlfBuffer = [2]byte{'\r', '\n'}
|
|
crlf := crlfBuffer[:]
|
|
|
|
scanner := bufio.NewScanner(caller.ssh)
|
|
scanner.Split(scannerSplitFunc)
|
|
|
|
for scanner.Scan() {
|
|
buffer.Write(scanner.Bytes())
|
|
buffer.Write(crlf)
|
|
|
|
p = buffer.Bytes()
|
|
|
|
n, err := oi.LongWrite(w, p)
|
|
if nil != err {
|
|
break
|
|
}
|
|
if expected, actual := int64(len(p)), n; expected != actual {
|
|
err := fmt.Errorf("transmission problem: tried sending %d bytes, but actually only sent %d bytes", expected, actual)
|
|
fmt.Fprint(caller.ssh, err.Error())
|
|
return
|
|
}
|
|
buffer.Reset()
|
|
}
|
|
|
|
// Wait a bit to receive data from the server (that we would send to io.Stdout).
|
|
time.Sleep(3 * time.Millisecond)
|
|
}
|
|
|
|
func scannerSplitFunc(data []byte, atEOF bool) (advance int, token []byte, err error) {
|
|
if atEOF {
|
|
return 0, nil, nil
|
|
}
|
|
return bufio.ScanLines(data, atEOF)
|
|
}
|
|
|
|
func telnetHandler(host *dbmodels.Host) ssh.Handler {
|
|
return func(s ssh.Session) {
|
|
// FIXME: log session in db
|
|
//actx := s.Context().Value(authContextKey).(*authContext)
|
|
caller := bastionTelnetCaller{ssh: s}
|
|
if err := telnet.DialToAndCall(host.DialAddr(), caller); err != nil {
|
|
fmt.Fprintf(s, "error: %v", err)
|
|
}
|
|
}
|
|
}
|