mirror of
https://github.com/FlourishingWorld/hk4e.git
synced 2026-02-13 01:22:26 +08:00
实现网关服务器不同版本客户端协议代理功能
This commit is contained in:
58
gate/client_proto/client_proto.go
Normal file
58
gate/client_proto/client_proto.go
Normal file
@@ -0,0 +1,58 @@
|
||||
package client_proto
|
||||
|
||||
import (
|
||||
"os"
|
||||
"strconv"
|
||||
"strings"
|
||||
|
||||
"hk4e/pkg/logger"
|
||||
)
|
||||
|
||||
type ClientCmdProtoMap struct {
|
||||
clientCmdIdCmdNameMap map[uint16]string
|
||||
clientCmdNameCmdIdMap map[string]uint16
|
||||
}
|
||||
|
||||
func NewClientCmdProtoMap() (r *ClientCmdProtoMap) {
|
||||
r = new(ClientCmdProtoMap)
|
||||
r.clientCmdIdCmdNameMap = make(map[uint16]string)
|
||||
r.clientCmdNameCmdIdMap = make(map[string]uint16)
|
||||
clientCmdFile, err := os.ReadFile("./client_cmd.csv")
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
clientCmdData := string(clientCmdFile)
|
||||
lineList := strings.Split(clientCmdData, "\n")
|
||||
for _, line := range lineList {
|
||||
item := strings.Split(line, ",")
|
||||
if len(item) != 2 {
|
||||
panic("parse client cmd file error")
|
||||
}
|
||||
cmdName := item[0]
|
||||
cmdId, err := strconv.Atoi(item[1])
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
r.clientCmdIdCmdNameMap[uint16(cmdId)] = cmdName
|
||||
r.clientCmdNameCmdIdMap[cmdName] = uint16(cmdId)
|
||||
}
|
||||
return r
|
||||
}
|
||||
|
||||
func (c *ClientCmdProtoMap) GetClientCmdNameByCmdId(cmdId uint16) string {
|
||||
cmdName, exist := c.clientCmdIdCmdNameMap[cmdId]
|
||||
if !exist {
|
||||
logger.Error("unknown cmd id: %v", cmdId)
|
||||
return ""
|
||||
}
|
||||
return cmdName
|
||||
}
|
||||
|
||||
func (c *ClientCmdProtoMap) GetClientCmdIdByCmdName(cmdName string) uint16 {
|
||||
cmdId, exist := c.clientCmdNameCmdIdMap[cmdName]
|
||||
if !exist {
|
||||
logger.Error("unknown cmd name: %v", cmdName)
|
||||
return 0
|
||||
}
|
||||
return cmdId
|
||||
}
|
||||
33
gate/client_proto/client_proto_gen_test.go
Normal file
33
gate/client_proto/client_proto_gen_test.go
Normal file
@@ -0,0 +1,33 @@
|
||||
package client_proto
|
||||
|
||||
import (
|
||||
"os"
|
||||
"testing"
|
||||
)
|
||||
|
||||
func TestClientProtoGen(t *testing.T) {
|
||||
clientCmdProtoMap := NewClientCmdProtoMap()
|
||||
|
||||
fileData := "package client_proto\n"
|
||||
fileData += "\n"
|
||||
fileData += "import (\n"
|
||||
fileData += "\"hk4e/gate/client_proto/proto\"\n"
|
||||
fileData += "pb \"google.golang.org/protobuf/proto\"\n"
|
||||
fileData += ")\n"
|
||||
fileData += "\n"
|
||||
fileData += "func (c *ClientCmdProtoMap) GetClientProtoObjByCmdName(cmdName string) pb.Message {\n"
|
||||
fileData += "switch cmdName {\n"
|
||||
for cmdName := range clientCmdProtoMap.clientCmdNameCmdIdMap {
|
||||
fileData += "case \"" + cmdName + "\":\nreturn new(proto." + cmdName + ")\n"
|
||||
}
|
||||
fileData += "default:\n"
|
||||
fileData += "return nil\n"
|
||||
fileData += "}\n"
|
||||
fileData += "}\n"
|
||||
fileData += "\n"
|
||||
|
||||
err := os.WriteFile("../client_proto_gen.go", []byte(fileData), 0644)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
}
|
||||
@@ -4,6 +4,7 @@ import (
|
||||
"bytes"
|
||||
"context"
|
||||
"encoding/binary"
|
||||
"reflect"
|
||||
"strconv"
|
||||
"sync"
|
||||
"time"
|
||||
@@ -12,6 +13,7 @@ import (
|
||||
"hk4e/common/mq"
|
||||
"hk4e/common/region"
|
||||
"hk4e/common/rpc"
|
||||
"hk4e/gate/client_proto"
|
||||
"hk4e/gate/kcp"
|
||||
"hk4e/node/api"
|
||||
"hk4e/pkg/logger"
|
||||
@@ -22,18 +24,20 @@ import (
|
||||
const PacketFreqLimit = 1000
|
||||
|
||||
type KcpConnectManager struct {
|
||||
discovery *rpc.DiscoveryClient
|
||||
openState bool
|
||||
sessionConvIdMap map[uint64]*Session
|
||||
sessionUserIdMap map[uint32]*Session
|
||||
sessionMapLock sync.RWMutex
|
||||
kcpEventInput chan *KcpEvent
|
||||
kcpEventOutput chan *KcpEvent
|
||||
cmdProtoMap *cmd.CmdProtoMap
|
||||
messageQueue *mq.MessageQueue
|
||||
localMsgOutput chan *ProtoMsg
|
||||
createSessionChan chan *Session
|
||||
destroySessionChan chan *Session
|
||||
discovery *rpc.DiscoveryClient
|
||||
openState bool
|
||||
sessionConvIdMap map[uint64]*Session
|
||||
sessionUserIdMap map[uint32]*Session
|
||||
sessionMapLock sync.RWMutex
|
||||
kcpEventInput chan *KcpEvent
|
||||
kcpEventOutput chan *KcpEvent
|
||||
serverCmdProtoMap *cmd.CmdProtoMap
|
||||
clientCmdProtoMap *client_proto.ClientCmdProtoMap
|
||||
clientCmdProtoMapRefValue reflect.Value
|
||||
messageQueue *mq.MessageQueue
|
||||
localMsgOutput chan *ProtoMsg
|
||||
createSessionChan chan *Session
|
||||
destroySessionChan chan *Session
|
||||
// 密钥相关
|
||||
dispatchKey []byte
|
||||
signRsaKey []byte
|
||||
@@ -48,7 +52,9 @@ func NewKcpConnectManager(messageQueue *mq.MessageQueue, discovery *rpc.Discover
|
||||
r.sessionUserIdMap = make(map[uint32]*Session)
|
||||
r.kcpEventInput = make(chan *KcpEvent, 1000)
|
||||
r.kcpEventOutput = make(chan *KcpEvent, 1000)
|
||||
r.cmdProtoMap = cmd.NewCmdProtoMap()
|
||||
r.serverCmdProtoMap = cmd.NewCmdProtoMap()
|
||||
r.clientCmdProtoMap = client_proto.NewClientCmdProtoMap()
|
||||
r.clientCmdProtoMapRefValue = reflect.ValueOf(r.clientCmdProtoMap)
|
||||
r.messageQueue = messageQueue
|
||||
r.localMsgOutput = make(chan *ProtoMsg, 1000)
|
||||
r.createSessionChan = make(chan *Session, 1000)
|
||||
|
||||
@@ -1,7 +1,11 @@
|
||||
package net
|
||||
|
||||
import (
|
||||
"reflect"
|
||||
|
||||
"hk4e/common/config"
|
||||
"hk4e/pkg/logger"
|
||||
"hk4e/pkg/object"
|
||||
"hk4e/protocol/cmd"
|
||||
"hk4e/protocol/proto"
|
||||
|
||||
@@ -22,6 +26,33 @@ type ProtoMessage struct {
|
||||
|
||||
func (k *KcpConnectManager) protoDecode(kcpMsg *KcpMsg) (protoMsgList []*ProtoMsg) {
|
||||
protoMsgList = make([]*ProtoMsg, 0)
|
||||
if config.CONF.Hk4e.ClientProtoProxyEnable {
|
||||
clientCmdId := kcpMsg.CmdId
|
||||
clientProtoData := kcpMsg.ProtoData
|
||||
cmdName := k.clientCmdProtoMap.GetClientCmdNameByCmdId(clientCmdId)
|
||||
clientProtoObj := k.clientCmdProtoMapRefValue.MethodByName(
|
||||
"GetClientProtoObjByCmdName",
|
||||
).Call([]reflect.Value{reflect.ValueOf(cmdName)})[0].Interface().(pb.Message)
|
||||
err := pb.Unmarshal(clientProtoData, clientProtoObj)
|
||||
if err != nil {
|
||||
logger.Error("unmarshal client proto error: %v", err)
|
||||
return protoMsgList
|
||||
}
|
||||
serverCmdId := k.serverCmdProtoMap.GetCmdIdByCmdName(cmdName)
|
||||
serverProtoObj := k.serverCmdProtoMap.GetProtoObjByCmdId(serverCmdId)
|
||||
err = object.CopyProtoBufSameField(serverProtoObj, clientProtoObj)
|
||||
if err != nil {
|
||||
logger.Error("copy proto obj error: %v", err)
|
||||
return protoMsgList
|
||||
}
|
||||
serverProtoData, err := pb.Marshal(serverProtoObj)
|
||||
if err != nil {
|
||||
logger.Error("marshal server proto error: %v", err)
|
||||
return protoMsgList
|
||||
}
|
||||
kcpMsg.CmdId = serverCmdId
|
||||
kcpMsg.ProtoData = serverProtoData
|
||||
}
|
||||
protoMsg := new(ProtoMsg)
|
||||
protoMsg.ConvId = kcpMsg.ConvId
|
||||
protoMsg.CmdId = kcpMsg.CmdId
|
||||
@@ -130,11 +161,36 @@ func (k *KcpConnectManager) protoEncode(protoMsg *ProtoMsg) (kcpMsg *KcpMsg) {
|
||||
} else {
|
||||
kcpMsg.ProtoData = nil
|
||||
}
|
||||
if config.CONF.Hk4e.ClientProtoProxyEnable {
|
||||
serverCmdId := kcpMsg.CmdId
|
||||
serverProtoData := kcpMsg.ProtoData
|
||||
serverProtoObj := k.serverCmdProtoMap.GetProtoObjByCmdId(serverCmdId)
|
||||
err := pb.Unmarshal(serverProtoData, serverProtoObj)
|
||||
if err != nil {
|
||||
logger.Error("unmarshal server proto error: %v", err)
|
||||
}
|
||||
cmdName := k.serverCmdProtoMap.GetCmdNameByCmdId(serverCmdId)
|
||||
clientProtoObj := k.clientCmdProtoMapRefValue.MethodByName(
|
||||
"GetClientProtoObjByCmdName",
|
||||
).Call([]reflect.Value{reflect.ValueOf(cmdName)})[0].Interface().(pb.Message)
|
||||
err = object.CopyProtoBufSameField(clientProtoObj, serverProtoObj)
|
||||
if err != nil {
|
||||
logger.Error("copy proto obj error: %v", err)
|
||||
return nil
|
||||
}
|
||||
clientProtoData, err := pb.Marshal(clientProtoObj)
|
||||
if err != nil {
|
||||
logger.Error("marshal client proto error: %v", err)
|
||||
}
|
||||
clientCmdId := k.clientCmdProtoMap.GetClientCmdIdByCmdName(cmdName)
|
||||
kcpMsg.CmdId = clientCmdId
|
||||
kcpMsg.ProtoData = clientProtoData
|
||||
}
|
||||
return kcpMsg
|
||||
}
|
||||
|
||||
func (k *KcpConnectManager) decodePayloadToProto(cmdId uint16, protoData []byte) (protoObj pb.Message) {
|
||||
protoObj = k.cmdProtoMap.GetProtoObjByCmdId(cmdId)
|
||||
protoObj = k.serverCmdProtoMap.GetProtoObjByCmdId(cmdId)
|
||||
if protoObj == nil {
|
||||
logger.Error("get new proto object is nil")
|
||||
return nil
|
||||
@@ -148,7 +204,7 @@ func (k *KcpConnectManager) decodePayloadToProto(cmdId uint16, protoData []byte)
|
||||
}
|
||||
|
||||
func (k *KcpConnectManager) encodeProtoToPayload(protoObj pb.Message) (cmdId uint16, protoData []byte) {
|
||||
cmdId = k.cmdProtoMap.GetCmdIdByProtoObj(protoObj)
|
||||
cmdId = k.serverCmdProtoMap.GetCmdIdByProtoObj(protoObj)
|
||||
var err error = nil
|
||||
protoData, err = pb.Marshal(protoObj)
|
||||
if err != nil {
|
||||
|
||||
Reference in New Issue
Block a user