mirror of
https://gitee.com/lauix/HFish
synced 2025-05-10 20:08:12 +08:00
138 lines
2.7 KiB
Go
138 lines
2.7 KiB
Go
package telnet
|
|
|
|
import (
|
|
"net"
|
|
"fmt"
|
|
"bufio"
|
|
"strings"
|
|
"os"
|
|
"HFish/utils/is"
|
|
"HFish/core/rpc/client"
|
|
"strconv"
|
|
"HFish/core/report"
|
|
"HFish/utils/log"
|
|
"github.com/bitly/go-simplejson"
|
|
"HFish/utils/json"
|
|
"HFish/utils/file"
|
|
)
|
|
|
|
// 服务端连接
|
|
func server(address string, exitChan chan int) {
|
|
l, err := net.Listen("tcp", address)
|
|
|
|
if err != nil {
|
|
fmt.Println(err.Error())
|
|
exitChan <- 1
|
|
}
|
|
|
|
defer l.Close()
|
|
|
|
for {
|
|
conn, err := l.Accept()
|
|
|
|
if err != nil {
|
|
fmt.Println(err.Error())
|
|
continue
|
|
}
|
|
|
|
arr := strings.Split(conn.RemoteAddr().String(), ":")
|
|
|
|
// 判断是否为 RPC 客户端
|
|
var id string
|
|
|
|
if is.Rpc() {
|
|
id = client.ReportResult("TELNET", "", arr[0], conn.RemoteAddr().String()+" 已经连接", "0")
|
|
} else {
|
|
id = strconv.FormatInt(report.ReportTelnet(arr[0], "本机", conn.RemoteAddr().String()+" 已经连接"), 10)
|
|
}
|
|
|
|
log.Pr("Telnet", arr[0], "已经连接")
|
|
|
|
// 根据连接开启会话, 这个过程需要并行执行
|
|
go handleSession(conn, exitChan, id)
|
|
}
|
|
}
|
|
|
|
func getJson() *simplejson.Json {
|
|
res, err := json.GetTelnet()
|
|
|
|
if err != nil {
|
|
log.Pr("HFish", "127.0.0.1", "解析 Telnet JSON 文件失败", err)
|
|
}
|
|
return res
|
|
}
|
|
|
|
// 会话处理
|
|
func handleSession(conn net.Conn, exitChan chan int, id string) {
|
|
fmt.Println("Session started")
|
|
reader := bufio.NewReader(conn)
|
|
|
|
for {
|
|
str, err := reader.ReadString('\n')
|
|
|
|
// telnet命令
|
|
if err == nil {
|
|
str = strings.TrimSpace(str)
|
|
|
|
if is.Rpc() {
|
|
go client.ReportResult("TELNET", "", "", "&&"+str, id)
|
|
} else {
|
|
go report.ReportUpdateTelnet(id, "&&"+str)
|
|
}
|
|
|
|
if !processTelnetCommand(str, exitChan) {
|
|
conn.Close()
|
|
break
|
|
}
|
|
|
|
res := getJson()
|
|
|
|
fileName := res.Get("command").Get(str).MustString()
|
|
|
|
if (fileName == "") {
|
|
fileName = res.Get("command").Get("default").MustString()
|
|
}
|
|
|
|
output := file.ReadLibsText("telnet", fileName)
|
|
|
|
conn.Write([]byte(output + "\r\n"))
|
|
} else {
|
|
// 发生错误
|
|
fmt.Println("Session closed")
|
|
conn.Close()
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
// telent协议命令
|
|
func processTelnetCommand(str string, exitChan chan int) bool {
|
|
// @close指令表示终止本次会话
|
|
if strings.HasPrefix(str, "@close") {
|
|
fmt.Println("Session closed")
|
|
// 告知外部需要断开连接
|
|
return false
|
|
// @shutdown指令表示终止服务进程
|
|
} else if strings.HasPrefix(str, "@shutdown") {
|
|
fmt.Println("Server shutdown")
|
|
// 往通道中写入0, 阻塞等待接收方处理
|
|
exitChan <- 0
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
func Start(addr string) {
|
|
// 创建一个程序结束码的通道
|
|
exitChan := make(chan int)
|
|
|
|
// 将服务器并发运行
|
|
go server(addr, exitChan)
|
|
|
|
// 通道阻塞,等待接受返回值
|
|
code := <-exitChan
|
|
|
|
// 标记程序返回值并退出
|
|
os.Exit(code)
|
|
}
|