Files
bl/logic/main.go
昔念 ba46330056 build: 更新 go.mod 并添加新依赖
- 在 go.mod 中添加了 github.com/ECUST-XX/xml v1.20.2 依赖
- 更新了 go.sum 文件以包含新依赖的哈希值
- 在 ServerEvent.go
2025-06-26 19:22:06 +08:00

209 lines
5.5 KiB
Go

package main
import (
"bytes"
_ "github.com/gogf/gf/contrib/nosql/redis/v2"
"github.com/lunixbochs/struc"
"github.com/panjf2000/gnet/v2"
"blazing/common/core/info"
"blazing/common/data/entity"
"blazing/common/socket"
"blazing/common/socket/cmd"
"blazing/common/socket/handler"
"blazing/cool"
"blazing/logic/controller"
"fmt"
"reflect"
"strconv"
"sync"
"github.com/gogf/gf/v2/os/gctx"
)
var (
maininfocodec = info.NewInfoCodec() //创建一个InfoCodec实例
maincontroller = controller.NewController() //注入service
mainplayer sync.Map //玩家数据
maincmdcache sync.Map //命令缓存
)
func main() {
if cool.IsRedisMode {
go cool.ListenFunc(gctx.New())
}
Start(cool.Config.PortBL) //注入service
}
func Start(port string) {
head := handler.NewTomeeHandler()
head.Callback = recv
socket.NewServer(socket.WithPort(port), socket.WithSocketHandler(head)).Start()
}
func recv(c gnet.Conn, data handler.TomeeHeader) {
// 处理接收到的TomeeHeader数据
// fmt.Println("收到数据:", data)
processWithReflection(c, data)
}
func getplayer(c gnet.Conn, userid uint32) *entity.Player { //TODO 这里待优化,可能存在内存泄漏问题
clientdata := c.Context().(*entity.ClientData)
if clientdata != nil && clientdata.Player != nil {
return clientdata.Player
}
var player *entity.Player
if player1, ok := mainplayer.Load((userid)); !ok {
player = entity.NewPlayer(
entity.WithUserID(userid), //注入ID
entity.WithConn(c), //注入conn
)
mainplayer.Store(userid, player)
} else {
player = player1.(*entity.Player) //取成功,否则创建
}
clientdata.Player = player
return player
}
// 遍历结构体方法并执行RECV_cmd
func processWithReflection(c gnet.Conn, pp handler.TomeeHeader) {
data := pp.Data
player := getplayer(c, pp.UserID) //获取player实例
cmdlister, ok := maincmdcache.Load(pp.CMDID) //TODO 待实现对不同用户初始化方法以取消全局cmdcache
if ok {
callhandler(cmdlister.(reflect.Value), data, player, cmd.EnumCommandID(pp.CMDID))
// return
} else {
// 获取对象的反射值和类型
value := reflect.ValueOf(maincontroller)
// 如果传入的是指针,获取其指向的值
if value.Kind() == reflect.Ptr {
if value.IsNil() {
fmt.Println("错误: 传入的是nil指针")
return
}
value = value.Elem()
}
// 获取类型
typ := value.Type()
// 遍历所有方法
fmt.Printf("检查结构体 %s 的方法...\n", typ.Name(), typ.NumMethod())
for i := 0; i < typ.NumMethod(); i++ {
method := typ.Method(i)
if method.Type.NumIn() == 4 { //&& // TODO 接收者 + 2个参数 一个是类型,一个是player
//method.Type.In(1) == reflect.TypeOf([]byte{}
// 获取方法值
methodValue := value.MethodByName(method.Name)
// 准备参数
cmd1 := method.Name[len("Recv_"):len(method.Name)]
cmdint, _ := strconv.Atoi(cmd1)
if cmdint == int(pp.CMDID) {
if !cmd.Isexist(cmd.CommandID, cmd.EnumCommandID(pp.CMDID)) {
fmt.Println("ID方法存在cmd未注册", pp.CMDID, cmd.CommandID)
}
maincmdcache.Store(pp.CMDID, methodValue) //TODO 待实现对不同用户初始化方法以取消全局cmdcache
callhandler(methodValue, data, player, cmd.EnumCommandID(pp.CMDID))
}
//return
}
//}
}
}
}
func callhandler(cmdlister reflect.Value, data []byte, player *entity.Player, cmd cmd.EnumCommandID) {
//TODO 待实现返回参数返回客户端???实现直接返回对象进行序列化 传递chan待返回
//TODO 已实现,待测试结构体序列化
recvchan := make(chan any, 1) //传递自定义chan
go func() { //TODO 待实现ants线程池,以及确认是否存在顺序混乱问题https://github.com/lxzan/gws/issues/3
retv := <-recvchan
switch ttt := retv.(type) {
case []byte:
player.SendPackBytes(cmd, ttt)
default:
var data1 bytes.Buffer
struc.Unpack(&data1, &ttt)
player.SendPackBytes(cmd, data1.Bytes())
}
}()
cmdlister.Call([]reflect.Value{
reflect.ValueOf(data),
reflect.ValueOf(player),
reflect.ValueOf(recvchan),
})
}
func getincodec(cmdid cmd.EnumCommandID, data []byte) reflect.Value {
// 获取对象的反射值和类型
value := reflect.ValueOf(maininfocodec)
// 如果传入的是指针,获取其指向的值
if value.Kind() == reflect.Ptr {
if value.IsNil() {
fmt.Println("错误: 传入的是nil指针")
return reflect.ValueOf(nil)
}
value = value.Elem()
}
// 获取类型
typ := value.Type()
// 遍历所有方法
fmt.Printf("检查结构体 %s 的方法...\n", typ.Name(), typ.NumMethod())
for i := 0; i < typ.NumMethod(); i++ {
method := typ.Method(i)
if method.Type.NumIn() == 2 && // 接收者 + 2个参数
method.Type.In(1) == reflect.TypeOf([]byte{}) {
// 获取方法值
methodValue := value.MethodByName(method.Name)
// 准备参数
cmd1 := method.Name[len("In_"):len(method.Name)]
cmdint, _ := strconv.Atoi(cmd1)
if cmdint == int(cmdid) {
if !cmd.Isexist(cmd.CommandID, cmd.EnumCommandID(cmdid)) {
fmt.Println("ID方法存在cmd未注册", cmdid, cmd.CommandID)
}
//data := pp.Data
// 调用方法
tt := methodValue.Call([]reflect.Value{
// reflect.ValueOf(cmdint),
reflect.ValueOf(data),
})
return tt[0]
}
//return
}
//}
}
return reflect.ValueOf(nil)
}