2025-06-20 17:00:56 +08:00
|
|
|
package codec
|
|
|
|
|
|
|
|
|
|
import (
|
2025-06-22 00:50:40 +00:00
|
|
|
"blazing/common/data/entity"
|
2025-06-20 17:00:56 +08:00
|
|
|
"encoding/binary"
|
|
|
|
|
"errors"
|
|
|
|
|
"io"
|
2025-06-22 00:50:40 +00:00
|
|
|
"log"
|
2025-06-20 17:00:56 +08:00
|
|
|
|
|
|
|
|
"github.com/panjf2000/gnet/v2"
|
|
|
|
|
)
|
|
|
|
|
|
2025-06-22 00:50:40 +00:00
|
|
|
// CROSS_DOMAIN 定义跨域策略文件内容
|
|
|
|
|
const CROSS_DOMAIN = "<?xml version=\"1.0\"?><!DOCTYPE cross-domain-policy><cross-domain-policy><allow-access-from domain=\"*\" to-ports=\"*\" /></cross-domain-policy>\x00"
|
|
|
|
|
|
|
|
|
|
// TEXT 定义跨域请求的文本格式
|
|
|
|
|
const TEXT = "<policy-file-request/>\x00"
|
|
|
|
|
|
2025-06-20 17:00:56 +08:00
|
|
|
var ErrIncompletePacket = errors.New("incomplete packet")
|
|
|
|
|
|
|
|
|
|
// TomeeSocketCodec 协议格式:
|
|
|
|
|
//
|
|
|
|
|
// * 0 4
|
|
|
|
|
// * +-----------+
|
|
|
|
|
// * | body len |
|
|
|
|
|
// * +-----------+
|
|
|
|
|
// * | |
|
|
|
|
|
// * + +
|
|
|
|
|
// * | body bytes|
|
|
|
|
|
// * + +
|
|
|
|
|
// * | ... ... |
|
|
|
|
|
// * +-----------+
|
|
|
|
|
type TomeeSocketCodec struct{}
|
|
|
|
|
|
2025-06-22 00:50:40 +00:00
|
|
|
var _ SocketCodec = (*TomeeSocketCodec)(nil)
|
2025-06-20 17:00:56 +08:00
|
|
|
|
|
|
|
|
func NewTomeeSocketCodec() *TomeeSocketCodec {
|
|
|
|
|
return &TomeeSocketCodec{}
|
|
|
|
|
|
|
|
|
|
}
|
2025-06-22 00:50:40 +00:00
|
|
|
|
|
|
|
|
func handle(c gnet.Conn) {
|
|
|
|
|
clientdata:=c.Context().(*entity.ClientData)
|
|
|
|
|
if(clientdata.IsCrossDomain){
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
// 读取数据并检查是否为跨域请求
|
|
|
|
|
data, err := c.Peek(len(TEXT))
|
|
|
|
|
if err != nil {
|
|
|
|
|
log.Printf("Error reading cross-domain request: %v", err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if string(data) == TEXT { //判断是否是跨域请求
|
|
|
|
|
log.Printf("Received cross-domain request from %s", c.RemoteAddr())
|
|
|
|
|
// 处理跨域请求
|
|
|
|
|
c.Write([]byte(CROSS_DOMAIN))
|
|
|
|
|
c.Discard(len(TEXT))
|
|
|
|
|
|
|
|
|
|
clientdata.IsCrossDomain=true
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return
|
|
|
|
|
}
|
2025-06-20 17:00:56 +08:00
|
|
|
func (codec TomeeSocketCodec) Encode(buf []byte) ([]byte, error) {
|
|
|
|
|
bodyLen := len(buf)
|
|
|
|
|
data := make([]byte, 4+bodyLen)
|
2025-06-22 00:50:40 +00:00
|
|
|
|
2025-06-20 17:00:56 +08:00
|
|
|
// 写入4字节的包长度
|
|
|
|
|
binary.BigEndian.PutUint32(data[:4], uint32(bodyLen))
|
|
|
|
|
// 写入包体
|
|
|
|
|
copy(data[4:], buf)
|
2025-06-22 00:50:40 +00:00
|
|
|
|
2025-06-20 17:00:56 +08:00
|
|
|
return data, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (codec TomeeSocketCodec) Decode(c gnet.Conn) ([]byte, error) {
|
2025-06-22 00:50:40 +00:00
|
|
|
|
|
|
|
|
handle(c)
|
2025-06-20 17:00:56 +08:00
|
|
|
// 先读取4字节的包长度
|
|
|
|
|
lenBuf, err := c.Peek(4)
|
|
|
|
|
if err != nil {
|
|
|
|
|
if errors.Is(err, io.ErrShortBuffer) {
|
|
|
|
|
return nil, ErrIncompletePacket
|
|
|
|
|
}
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
2025-06-22 00:50:40 +00:00
|
|
|
|
2025-06-20 17:00:56 +08:00
|
|
|
bodyLen := binary.BigEndian.Uint32(lenBuf)
|
|
|
|
|
totalLen := 4 + int(bodyLen)
|
2025-06-22 00:50:40 +00:00
|
|
|
|
2025-06-20 17:00:56 +08:00
|
|
|
// 检查整个包是否完整
|
|
|
|
|
buf, err := c.Peek(totalLen)
|
|
|
|
|
if err != nil {
|
|
|
|
|
if errors.Is(err, io.ErrShortBuffer) {
|
|
|
|
|
return nil, ErrIncompletePacket
|
|
|
|
|
}
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
2025-06-22 00:50:40 +00:00
|
|
|
|
2025-06-20 17:00:56 +08:00
|
|
|
// 提取包体
|
|
|
|
|
body := make([]byte, bodyLen)
|
|
|
|
|
copy(body, buf[4:totalLen])
|
2025-06-22 00:50:40 +00:00
|
|
|
|
2025-06-20 17:00:56 +08:00
|
|
|
// 从缓冲区中丢弃已读取的数据
|
|
|
|
|
_, _ = c.Discard(totalLen)
|
2025-06-22 00:50:40 +00:00
|
|
|
|
2025-06-20 17:00:56 +08:00
|
|
|
return body, nil
|
|
|
|
|
}
|