feat: server/client 新增 Unix Domain Socket 客户端

This commit is contained in:
kercylan98 2023-08-22 15:20:03 +08:00
parent dc761964b9
commit 3de1f6b9d3
2 changed files with 205 additions and 0 deletions

157
server/client/unix.go Normal file
View File

@ -0,0 +1,157 @@
package client
import (
"bufio"
"github.com/kercylan98/minotaur/server"
"github.com/kercylan98/minotaur/utils/concurrent"
"net"
"sync"
"time"
)
// NewUnixDomainSocket 创建 unix domain socket 客户端
func NewUnixDomainSocket(addr string) *UnixDomainSocket {
return &UnixDomainSocket{
udsEvents: new(udsEvents),
addr: addr,
data: map[string]any{},
}
}
// UnixDomainSocket unix domain socket 客户端
type UnixDomainSocket struct {
*udsEvents
conn net.Conn
addr string
data map[string]any
mutex sync.Mutex
packetPool *concurrent.Pool[*Packet]
packets []*Packet
accumulate []server.Packet
}
// Run 启动
func (slf *UnixDomainSocket) Run() error {
c, err := net.Dial("unix", slf.addr)
if err != nil {
return err
}
slf.conn = c
var wait = new(sync.WaitGroup)
wait.Add(1)
go slf.writeLoop(wait)
wait.Wait()
go func() {
defer func() {
if err := recover(); err != nil {
slf.Close()
slf.OnUDSConnectionClosedEvent(slf, err)
}
}()
slf.OnUDSConnectionOpenedEvent(slf)
for slf.packetPool != nil {
reader := bufio.NewReader(slf.conn)
packet, readErr := reader.ReadBytes('\n')
if readErr != nil {
panic(readErr)
}
slf.OnUDSConnectionReceivePacketEvent(slf, server.NewPacket(packet))
}
}()
return nil
}
// Close 关闭
func (slf *UnixDomainSocket) Close() {
if slf.packetPool != nil {
slf.packetPool.Close()
slf.packetPool = nil
}
slf.packets = nil
}
// IsConnected 是否已连接
func (slf *UnixDomainSocket) IsConnected() bool {
return slf.packetPool != nil
}
// GetData 获取数据
func (slf *UnixDomainSocket) GetData(key string) any {
return slf.data[key]
}
// SetData 设置数据
func (slf *UnixDomainSocket) SetData(key string, value any) {
slf.data[key] = value
}
// Write 向连接中写入数据
func (slf *UnixDomainSocket) Write(packet server.Packet) {
if slf.packetPool == nil {
slf.accumulate = append(slf.accumulate, packet)
return
}
cp := slf.packetPool.Get()
cp.websocketMessageType = packet.WebsocketType
cp.packet = packet.Data
slf.mutex.Lock()
slf.packets = append(slf.packets, cp)
slf.mutex.Unlock()
}
// writeLoop 写循环
func (slf *UnixDomainSocket) writeLoop(wait *sync.WaitGroup) {
slf.packetPool = concurrent.NewPool[*Packet](10*1024,
func() *Packet {
return &Packet{}
}, func(data *Packet) {
data.packet = nil
data.websocketMessageType = 0
data.callback = nil
},
)
slf.mutex.Lock()
for _, packet := range slf.accumulate {
cp := slf.packetPool.Get()
cp.websocketMessageType = packet.WebsocketType
cp.packet = packet.Data
slf.packets = append(slf.packets, cp)
}
slf.accumulate = nil
slf.mutex.Unlock()
defer func() {
if err := recover(); err != nil {
slf.Close()
}
}()
wait.Done()
for {
slf.mutex.Lock()
if slf.packetPool == nil {
slf.mutex.Unlock()
return
}
if len(slf.packets) == 0 {
slf.mutex.Unlock()
time.Sleep(50 * time.Millisecond)
continue
}
packets := slf.packets[0:]
slf.packets = slf.packets[0:0]
slf.mutex.Unlock()
for i := 0; i < len(packets); i++ {
data := packets[i]
var _, err = slf.conn.Write(data.packet)
callback := data.callback
slf.packetPool.Release(data)
if callback != nil {
callback(err)
}
if err != nil {
panic(err)
}
}
}
}

View File

@ -0,0 +1,48 @@
package client
import "github.com/kercylan98/minotaur/server"
type (
UDSConnectionClosedEventHandle func(conn *UnixDomainSocket, err any)
UDSConnectionOpenedEventHandle func(conn *UnixDomainSocket)
UDSConnectionReceivePacketEventHandle func(conn *UnixDomainSocket, packet server.Packet)
)
type udsEvents struct {
UDSConnectionClosedEventHandles []UDSConnectionClosedEventHandle
UDSConnectionOpenedEventHandles []UDSConnectionOpenedEventHandle
UDSConnectionReceivePacketEventHandles []UDSConnectionReceivePacketEventHandle
}
// RegUDSConnectionClosedEvent 注册连接关闭事件
func (slf *udsEvents) RegUDSConnectionClosedEvent(handle UDSConnectionClosedEventHandle) {
slf.UDSConnectionClosedEventHandles = append(slf.UDSConnectionClosedEventHandles, handle)
}
func (slf *udsEvents) OnUDSConnectionClosedEvent(conn *UnixDomainSocket, err any) {
for _, handle := range slf.UDSConnectionClosedEventHandles {
handle(conn, err)
}
}
// RegUDSConnectionOpenedEvent 注册连接打开事件
func (slf *udsEvents) RegUDSConnectionOpenedEvent(handle UDSConnectionOpenedEventHandle) {
slf.UDSConnectionOpenedEventHandles = append(slf.UDSConnectionOpenedEventHandles, handle)
}
func (slf *udsEvents) OnUDSConnectionOpenedEvent(conn *UnixDomainSocket) {
for _, handle := range slf.UDSConnectionOpenedEventHandles {
handle(conn)
}
}
// RegUDSConnectionReceivePacketEvent 注册连接接收数据包事件
func (slf *udsEvents) RegUDSConnectionReceivePacketEvent(handle UDSConnectionReceivePacketEventHandle) {
slf.UDSConnectionReceivePacketEventHandles = append(slf.UDSConnectionReceivePacketEventHandles, handle)
}
func (slf *udsEvents) OnUDSConnectionReceivePacketEvent(conn *UnixDomainSocket, packet server.Packet) {
for _, handle := range slf.UDSConnectionReceivePacketEventHandles {
handle(conn, packet)
}
}