perf: ws模块telnet分开处理避免类型指针错误导致panic程序崩溃
This commit is contained in:
@@ -50,10 +50,12 @@ func (c *ConnTelnet) NewClient() (*ConnTelnet, error) {
|
|||||||
// fmt.Fprintln(client, c.User)
|
// fmt.Fprintln(client, c.User)
|
||||||
// fmt.Fprintln(client, c.Password)
|
// fmt.Fprintln(client, c.Password)
|
||||||
|
|
||||||
c.Client = &client
|
|
||||||
|
|
||||||
// 调整窗口大小 (120 列 x 128 行)
|
// 调整窗口大小 (120 列 x 128 行)
|
||||||
requestPty(c.Client, 120, 128)
|
// 需要确保接收方理解并正确处理发送窗口大小设置命令
|
||||||
|
client.Write([]byte{255, 251, 31})
|
||||||
|
client.Write([]byte{255, 250, 31, byte(120 >> 8), byte(120 & 0xFF), byte(128 >> 8), byte(128 & 0xFF), 255, 240})
|
||||||
|
|
||||||
|
c.Client = &client
|
||||||
|
|
||||||
// 排空连接登录的信息
|
// 排空连接登录的信息
|
||||||
c.RunCMD("")
|
c.RunCMD("")
|
||||||
@@ -111,20 +113,9 @@ func (c *ConnTelnet) NewClientSession(cols, rows int) (*TelnetClientSession, err
|
|||||||
if c.Client == nil {
|
if c.Client == nil {
|
||||||
return nil, fmt.Errorf("telnet client not connected")
|
return nil, fmt.Errorf("telnet client not connected")
|
||||||
}
|
}
|
||||||
requestPty(c.Client, cols, rows)
|
s := &TelnetClientSession{
|
||||||
return &TelnetClientSession{
|
|
||||||
Client: *c.Client,
|
Client: *c.Client,
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// requestPty 调整终端窗口大小
|
|
||||||
func requestPty(client *net.Conn, cols, rows int) error {
|
|
||||||
if client == nil {
|
|
||||||
return fmt.Errorf("telnet client not connected")
|
|
||||||
}
|
}
|
||||||
conn := *client
|
s.WindowChange(cols, rows)
|
||||||
// 需要确保接收方理解并正确处理发送窗口大小设置命令
|
return s, nil
|
||||||
conn.Write([]byte{255, 251, 31})
|
|
||||||
conn.Write([]byte{255, 250, 31, byte(cols >> 8), byte(cols & 0xFF), byte(rows >> 8), byte(rows & 0xFF), 255, 240})
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,6 +19,17 @@ func (s *TelnetClientSession) Close() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// WindowChange informs the remote host about a terminal window dimension change to h rows and w columns.
|
||||||
|
func (s *TelnetClientSession) WindowChange(h, w int) error {
|
||||||
|
if s.Client == nil {
|
||||||
|
return fmt.Errorf("client is nil to content write failed")
|
||||||
|
}
|
||||||
|
// 需要确保接收方理解并正确处理发送窗口大小设置命令
|
||||||
|
s.Client.Write([]byte{255, 251, 31})
|
||||||
|
s.Client.Write([]byte{255, 250, 31, byte(w >> 8), byte(w & 0xFF), byte(h >> 8), byte(h & 0xFF), 255, 240})
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// Write 写入命令 不带回车(\n)也会执行根据客户端情况
|
// Write 写入命令 不带回车(\n)也会执行根据客户端情况
|
||||||
func (s *TelnetClientSession) Write(cmd string) (int, error) {
|
func (s *TelnetClientSession) Write(cmd string) (int, error) {
|
||||||
if s.Client == nil {
|
if s.Client == nil {
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ package controller
|
|||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"strconv"
|
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@@ -88,7 +87,7 @@ func (s *WSController) WS(c *gin.Context) {
|
|||||||
|
|
||||||
// Test 测试
|
// Test 测试
|
||||||
//
|
//
|
||||||
// GET /test?clientId=&groupID=
|
// GET /test?clientId=xxx&groupID=xxx
|
||||||
func (s *WSController) Test(c *gin.Context) {
|
func (s *WSController) Test(c *gin.Context) {
|
||||||
language := ctx.AcceptLanguage(c)
|
language := ctx.AcceptLanguage(c)
|
||||||
|
|
||||||
@@ -125,6 +124,21 @@ func (s *WSController) Test(c *gin.Context) {
|
|||||||
// GET /ssh?hostId=1&cols=80&rows=40
|
// GET /ssh?hostId=1&cols=80&rows=40
|
||||||
func (s *WSController) SSH(c *gin.Context) {
|
func (s *WSController) SSH(c *gin.Context) {
|
||||||
language := ctx.AcceptLanguage(c)
|
language := ctx.AcceptLanguage(c)
|
||||||
|
var query struct {
|
||||||
|
HostId string `form:"hostId" binding:"required"` // 连接主机ID
|
||||||
|
Cols int `form:"cols"` // 终端单行字符数
|
||||||
|
Rows int `form:"rows"` // 终端显示行数
|
||||||
|
}
|
||||||
|
if err := c.ShouldBindQuery(&query); err != nil {
|
||||||
|
c.JSON(400, result.CodeMsg(400, i18n.TKey(language, "app.common.err400")))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if query.Cols < 80 || query.Cols > 400 {
|
||||||
|
query.Cols = 80
|
||||||
|
}
|
||||||
|
if query.Rows < 40 || query.Rows > 1200 {
|
||||||
|
query.Rows = 40
|
||||||
|
}
|
||||||
|
|
||||||
// 登录用户信息
|
// 登录用户信息
|
||||||
loginUser, err := ctx.LoginUser(c)
|
loginUser, err := ctx.LoginUser(c)
|
||||||
@@ -133,14 +147,8 @@ func (s *WSController) SSH(c *gin.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// 连接主机ID
|
neHost := s.neHostService.SelectById(query.HostId)
|
||||||
hostId := c.Query("hostId")
|
if neHost.HostID != query.HostId || neHost.HostType != "ssh" {
|
||||||
if hostId == "" {
|
|
||||||
c.JSON(400, result.CodeMsg(400, i18n.TKey(language, "app.common.err400")))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
neHost := s.neHostService.SelectById(hostId)
|
|
||||||
if neHost.HostID != hostId || neHost.HostType != "ssh" {
|
|
||||||
// 没有可访问主机信息数据!
|
// 没有可访问主机信息数据!
|
||||||
c.JSON(200, result.ErrMsg(i18n.TKey(language, "neHost.noData")))
|
c.JSON(200, result.ErrMsg(i18n.TKey(language, "neHost.noData")))
|
||||||
return
|
return
|
||||||
@@ -163,19 +171,8 @@ func (s *WSController) SSH(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
defer client.Close()
|
defer client.Close()
|
||||||
|
|
||||||
// 终端单行字符数
|
|
||||||
cols, err := strconv.Atoi(c.Query("cols"))
|
|
||||||
if err != nil {
|
|
||||||
cols = 80
|
|
||||||
}
|
|
||||||
// 终端显示行数
|
|
||||||
rows, err := strconv.Atoi(c.Query("rows"))
|
|
||||||
if err != nil {
|
|
||||||
rows = 40
|
|
||||||
}
|
|
||||||
|
|
||||||
// 创建SSH客户端会话
|
// 创建SSH客户端会话
|
||||||
clientSession, err := client.NewClientSession(cols, rows)
|
clientSession, err := client.NewClientSession(query.Cols, query.Rows)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// 连接主机失败,请检查连接参数后重试
|
// 连接主机失败,请检查连接参数后重试
|
||||||
c.JSON(200, result.ErrMsg(i18n.TKey(language, "neHost.errByHostInfo")))
|
c.JSON(200, result.ErrMsg(i18n.TKey(language, "neHost.errByHostInfo")))
|
||||||
@@ -204,7 +201,7 @@ func (s *WSController) SSH(c *gin.Context) {
|
|||||||
if len(outputByte) > 0 {
|
if len(outputByte) > 0 {
|
||||||
outputStr := string(outputByte)
|
outputStr := string(outputByte)
|
||||||
msgByte, _ := json.Marshal(result.Ok(map[string]any{
|
msgByte, _ := json.Marshal(result.Ok(map[string]any{
|
||||||
"requestId": fmt.Sprintf("ssh_%s_%d", hostId, ms.UnixMilli()),
|
"requestId": fmt.Sprintf("ssh_%s_%d", neHost.HostID, ms.UnixMilli()),
|
||||||
"data": outputStr,
|
"data": outputStr,
|
||||||
}))
|
}))
|
||||||
wsClient.MsgChan <- msgByte
|
wsClient.MsgChan <- msgByte
|
||||||
@@ -229,6 +226,21 @@ func (s *WSController) SSH(c *gin.Context) {
|
|||||||
// GET /telnet?hostId=1
|
// GET /telnet?hostId=1
|
||||||
func (s *WSController) Telnet(c *gin.Context) {
|
func (s *WSController) Telnet(c *gin.Context) {
|
||||||
language := ctx.AcceptLanguage(c)
|
language := ctx.AcceptLanguage(c)
|
||||||
|
var query struct {
|
||||||
|
HostId string `form:"hostId" binding:"required"` // 连接主机ID
|
||||||
|
Cols int `form:"cols"` // 终端单行字符数
|
||||||
|
Rows int `form:"rows"` // 终端显示行数
|
||||||
|
}
|
||||||
|
if err := c.ShouldBindQuery(&query); err != nil {
|
||||||
|
c.JSON(400, result.CodeMsg(400, i18n.TKey(language, "app.common.err400")))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if query.Cols < 120 || query.Cols > 400 {
|
||||||
|
query.Cols = 120
|
||||||
|
}
|
||||||
|
if query.Rows < 128 || query.Rows > 1200 {
|
||||||
|
query.Rows = 128
|
||||||
|
}
|
||||||
|
|
||||||
// 登录用户信息
|
// 登录用户信息
|
||||||
loginUser, err := ctx.LoginUser(c)
|
loginUser, err := ctx.LoginUser(c)
|
||||||
@@ -237,14 +249,8 @@ func (s *WSController) Telnet(c *gin.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// 连接主机ID
|
neHost := s.neHostService.SelectById(query.HostId)
|
||||||
hostId := c.Query("hostId")
|
if neHost.HostID != query.HostId || neHost.HostType != "telnet" {
|
||||||
if hostId == "" {
|
|
||||||
c.JSON(400, result.CodeMsg(400, i18n.TKey(language, "app.common.err400")))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
neHost := s.neHostService.SelectById(hostId)
|
|
||||||
if neHost.HostID != hostId || neHost.HostType != "telnet" {
|
|
||||||
// 没有可访问主机信息数据!
|
// 没有可访问主机信息数据!
|
||||||
c.JSON(200, result.ErrMsg(i18n.TKey(language, "neHost.noData")))
|
c.JSON(200, result.ErrMsg(i18n.TKey(language, "neHost.noData")))
|
||||||
return
|
return
|
||||||
@@ -260,20 +266,8 @@ func (s *WSController) Telnet(c *gin.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
defer client.Close()
|
defer client.Close()
|
||||||
|
|
||||||
// 终端单行字符数
|
|
||||||
cols, err := strconv.Atoi(c.DefaultQuery("cols", "120"))
|
|
||||||
if err != nil {
|
|
||||||
cols = 120
|
|
||||||
}
|
|
||||||
// 终端显示行数
|
|
||||||
rows, err := strconv.Atoi(c.DefaultQuery("rows", "128"))
|
|
||||||
if err != nil {
|
|
||||||
rows = 128
|
|
||||||
}
|
|
||||||
|
|
||||||
// 创建Telnet客户端会话
|
// 创建Telnet客户端会话
|
||||||
clientSession, err := client.NewClientSession(cols, rows)
|
clientSession, err := client.NewClientSession(query.Cols, query.Rows)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// 连接主机失败,请检查连接参数后重试
|
// 连接主机失败,请检查连接参数后重试
|
||||||
c.JSON(200, result.ErrMsg(i18n.TKey(language, "neHost.errByHostInfo")))
|
c.JSON(200, result.ErrMsg(i18n.TKey(language, "neHost.errByHostInfo")))
|
||||||
@@ -290,7 +284,11 @@ func (s *WSController) Telnet(c *gin.Context) {
|
|||||||
|
|
||||||
wsClient := s.wsService.ClientCreate(loginUser.UserID, nil, wsConn, clientSession)
|
wsClient := s.wsService.ClientCreate(loginUser.UserID, nil, wsConn, clientSession)
|
||||||
go s.wsService.ClientWriteListen(wsClient)
|
go s.wsService.ClientWriteListen(wsClient)
|
||||||
go s.wsService.ClientReadListen(wsClient, service.ReceiveShell)
|
go s.wsService.ClientReadListen(wsClient, service.ReceiveTelnet)
|
||||||
|
|
||||||
|
// 等待1秒,排空首次消息
|
||||||
|
time.Sleep(1 * time.Second)
|
||||||
|
_ = clientSession.Read()
|
||||||
|
|
||||||
// 实时读取Telnet消息直接输出
|
// 实时读取Telnet消息直接输出
|
||||||
msTicker := time.NewTicker(100 * time.Millisecond)
|
msTicker := time.NewTicker(100 * time.Millisecond)
|
||||||
@@ -302,7 +300,7 @@ func (s *WSController) Telnet(c *gin.Context) {
|
|||||||
if len(outputByte) > 0 {
|
if len(outputByte) > 0 {
|
||||||
outputStr := strings.TrimRight(string(outputByte), "\x00")
|
outputStr := strings.TrimRight(string(outputByte), "\x00")
|
||||||
msgByte, _ := json.Marshal(result.Ok(map[string]any{
|
msgByte, _ := json.Marshal(result.Ok(map[string]any{
|
||||||
"requestId": fmt.Sprintf("telnet_%s_%d", hostId, ms.UnixMilli()),
|
"requestId": fmt.Sprintf("telnet_%s_%d", neHost.HostID, ms.UnixMilli()),
|
||||||
"data": outputStr,
|
"data": outputStr,
|
||||||
}))
|
}))
|
||||||
wsClient.MsgChan <- msgByte
|
wsClient.MsgChan <- msgByte
|
||||||
@@ -337,13 +335,6 @@ func (s *WSController) ShellView(c *gin.Context) {
|
|||||||
c.JSON(400, result.CodeMsg(400, i18n.TKey(language, "app.common.err400")))
|
c.JSON(400, result.CodeMsg(400, i18n.TKey(language, "app.common.err400")))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
// 登录用户信息
|
|
||||||
loginUser, err := ctx.LoginUser(c)
|
|
||||||
if err != nil {
|
|
||||||
c.JSON(401, result.CodeMsg(401, i18n.TKey(language, err.Error())))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if query.Cols < 120 || query.Cols > 400 {
|
if query.Cols < 120 || query.Cols > 400 {
|
||||||
query.Cols = 120
|
query.Cols = 120
|
||||||
}
|
}
|
||||||
@@ -351,6 +342,13 @@ func (s *WSController) ShellView(c *gin.Context) {
|
|||||||
query.Rows = 40
|
query.Rows = 40
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 登录用户信息
|
||||||
|
loginUser, err := ctx.LoginUser(c)
|
||||||
|
if err != nil {
|
||||||
|
c.JSON(401, result.CodeMsg(401, i18n.TKey(language, err.Error())))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// 网元主机的SSH客户端
|
// 网元主机的SSH客户端
|
||||||
sshClient, err := s.neInfoService.NeRunSSHClient(query.NeType, query.NeId)
|
sshClient, err := s.neInfoService.NeRunSSHClient(query.NeType, query.NeId)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -377,6 +375,10 @@ func (s *WSController) ShellView(c *gin.Context) {
|
|||||||
go s.wsService.ClientWriteListen(wsClient)
|
go s.wsService.ClientWriteListen(wsClient)
|
||||||
go s.wsService.ClientReadListen(wsClient, service.ReceiveShellView)
|
go s.wsService.ClientReadListen(wsClient, service.ReceiveShellView)
|
||||||
|
|
||||||
|
// 等待1秒,排空首次消息
|
||||||
|
time.Sleep(1 * time.Second)
|
||||||
|
_ = clientSession.Read()
|
||||||
|
|
||||||
// 实时读取SSH消息直接输出
|
// 实时读取SSH消息直接输出
|
||||||
msTicker := time.NewTicker(100 * time.Millisecond)
|
msTicker := time.NewTicker(100 * time.Millisecond)
|
||||||
defer msTicker.Stop()
|
defer msTicker.Stop()
|
||||||
|
|||||||
@@ -181,7 +181,7 @@ func (s *WSImpl) ClientReadListen(wsClient *model.WSClient, receiveType int) {
|
|||||||
if messageType == websocket.TextMessage {
|
if messageType == websocket.TextMessage {
|
||||||
var reqMsg model.WSRequest
|
var reqMsg model.WSRequest
|
||||||
if err := json.Unmarshal(msg, &reqMsg); err != nil {
|
if err := json.Unmarshal(msg, &reqMsg); err != nil {
|
||||||
msgByte, _ := json.Marshal(result.ErrMsg("message format not supported"))
|
msgByte, _ := json.Marshal(result.ErrMsg("message format json error"))
|
||||||
wsClient.MsgChan <- msgByte
|
wsClient.MsgChan <- msgByte
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
@@ -193,6 +193,8 @@ func (s *WSImpl) ClientReadListen(wsClient *model.WSClient, receiveType int) {
|
|||||||
go NewWSReceiveImpl.Shell(wsClient, reqMsg)
|
go NewWSReceiveImpl.Shell(wsClient, reqMsg)
|
||||||
case ReceiveShellView:
|
case ReceiveShellView:
|
||||||
go NewWSReceiveImpl.ShellView(wsClient, reqMsg)
|
go NewWSReceiveImpl.ShellView(wsClient, reqMsg)
|
||||||
|
case ReceiveTelnet:
|
||||||
|
go NewWSReceiveImpl.Telnet(wsClient, reqMsg)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ const (
|
|||||||
ReceiveCommont = iota // Commont 接收通用业务处理
|
ReceiveCommont = iota // Commont 接收通用业务处理
|
||||||
ReceiveShell // Shell 接收终端交互业务处理
|
ReceiveShell // Shell 接收终端交互业务处理
|
||||||
ReceiveShellView // ShellView 接收查看文件终端交互业务处理
|
ReceiveShellView // ShellView 接收查看文件终端交互业务处理
|
||||||
|
ReceiveTelnet // Telnet 接收终端交互业务处理
|
||||||
)
|
)
|
||||||
|
|
||||||
// IWSReceive WebSocket消息接收处理 服务层接口
|
// IWSReceive WebSocket消息接收处理 服务层接口
|
||||||
@@ -18,4 +19,7 @@ type IWSReceive interface {
|
|||||||
|
|
||||||
// ShellView 接收查看文件终端交互业务处理
|
// ShellView 接收查看文件终端交互业务处理
|
||||||
ShellView(client *model.WSClient, reqMsg model.WSRequest)
|
ShellView(client *model.WSClient, reqMsg model.WSRequest)
|
||||||
|
|
||||||
|
// Telnet 接收终端交互业务处理
|
||||||
|
Telnet(client *model.WSClient, reqMsg model.WSRequest)
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user