mirror of
https://github.com/duanhf2012/origin.git
synced 2026-02-04 06:54:45 +08:00
163 lines
3.3 KiB
Go
163 lines
3.3 KiB
Go
package network
|
|
|
|
import (
|
|
"errors"
|
|
"github.com/duanhf2012/origin/log"
|
|
"net"
|
|
"sync"
|
|
"sync/atomic"
|
|
"time"
|
|
)
|
|
|
|
type ConnSet map[net.Conn]struct{}
|
|
|
|
type TCPConn struct {
|
|
sync.Mutex
|
|
conn net.Conn
|
|
writeChan chan []byte
|
|
closeFlag int32
|
|
msgParser *MsgParser
|
|
}
|
|
|
|
func freeChannel(conn *TCPConn){
|
|
for;len(conn.writeChan)>0;{
|
|
byteBuff := <- conn.writeChan
|
|
if byteBuff != nil {
|
|
conn.ReleaseReadMsg(byteBuff)
|
|
}
|
|
}
|
|
}
|
|
|
|
func newTCPConn(conn net.Conn, pendingWriteNum int, msgParser *MsgParser,writeDeadline time.Duration) *TCPConn {
|
|
tcpConn := new(TCPConn)
|
|
tcpConn.conn = conn
|
|
tcpConn.writeChan = make(chan []byte, pendingWriteNum)
|
|
tcpConn.msgParser = msgParser
|
|
go func() {
|
|
for b := range tcpConn.writeChan {
|
|
if b == nil {
|
|
break
|
|
}
|
|
|
|
conn.SetWriteDeadline(time.Now().Add(writeDeadline))
|
|
_, err := conn.Write(b)
|
|
tcpConn.msgParser.ReleaseByteSlice(b)
|
|
|
|
if err != nil {
|
|
break
|
|
}
|
|
}
|
|
conn.Close()
|
|
tcpConn.Lock()
|
|
freeChannel(tcpConn)
|
|
atomic.StoreInt32(&tcpConn.closeFlag,1)
|
|
tcpConn.Unlock()
|
|
}()
|
|
|
|
return tcpConn
|
|
}
|
|
|
|
func (tcpConn *TCPConn) doDestroy() {
|
|
tcpConn.conn.(*net.TCPConn).SetLinger(0)
|
|
tcpConn.conn.Close()
|
|
|
|
if atomic.LoadInt32(&tcpConn.closeFlag)==0 {
|
|
close(tcpConn.writeChan)
|
|
atomic.StoreInt32(&tcpConn.closeFlag,1)
|
|
}
|
|
}
|
|
|
|
func (tcpConn *TCPConn) Destroy() {
|
|
tcpConn.Lock()
|
|
defer tcpConn.Unlock()
|
|
|
|
tcpConn.doDestroy()
|
|
}
|
|
|
|
func (tcpConn *TCPConn) Close() {
|
|
tcpConn.Lock()
|
|
defer tcpConn.Unlock()
|
|
if atomic.LoadInt32(&tcpConn.closeFlag)==1 {
|
|
return
|
|
}
|
|
|
|
tcpConn.doWrite(nil)
|
|
atomic.StoreInt32(&tcpConn.closeFlag,1)
|
|
}
|
|
|
|
func (tcpConn *TCPConn) GetRemoteIp() string {
|
|
return tcpConn.conn.RemoteAddr().String()
|
|
}
|
|
|
|
func (tcpConn *TCPConn) doWrite(b []byte) error{
|
|
if len(tcpConn.writeChan) == cap(tcpConn.writeChan) {
|
|
tcpConn.ReleaseReadMsg(b)
|
|
log.Error("close conn: channel full")
|
|
tcpConn.doDestroy()
|
|
return errors.New("close conn: channel full")
|
|
}
|
|
|
|
tcpConn.writeChan <- b
|
|
return nil
|
|
}
|
|
|
|
// b must not be modified by the others goroutines
|
|
func (tcpConn *TCPConn) Write(b []byte) error{
|
|
tcpConn.Lock()
|
|
defer tcpConn.Unlock()
|
|
if atomic.LoadInt32(&tcpConn.closeFlag)==1 || b == nil {
|
|
tcpConn.ReleaseReadMsg(b)
|
|
return errors.New("conn is close")
|
|
}
|
|
|
|
return tcpConn.doWrite(b)
|
|
}
|
|
|
|
func (tcpConn *TCPConn) Read(b []byte) (int, error) {
|
|
return tcpConn.conn.Read(b)
|
|
}
|
|
|
|
func (tcpConn *TCPConn) LocalAddr() net.Addr {
|
|
return tcpConn.conn.LocalAddr()
|
|
}
|
|
|
|
func (tcpConn *TCPConn) RemoteAddr() net.Addr {
|
|
return tcpConn.conn.RemoteAddr()
|
|
}
|
|
|
|
func (tcpConn *TCPConn) ReadMsg() ([]byte, error) {
|
|
return tcpConn.msgParser.Read(tcpConn)
|
|
}
|
|
|
|
func (tcpConn *TCPConn) ReleaseReadMsg(byteBuff []byte){
|
|
tcpConn.msgParser.ReleaseByteSlice(byteBuff)
|
|
}
|
|
|
|
func (tcpConn *TCPConn) WriteMsg(args ...[]byte) error {
|
|
if atomic.LoadInt32(&tcpConn.closeFlag) == 1 {
|
|
return errors.New("conn is close")
|
|
}
|
|
return tcpConn.msgParser.Write(tcpConn, args...)
|
|
}
|
|
|
|
func (tcpConn *TCPConn) WriteRawMsg(args []byte) error {
|
|
if atomic.LoadInt32(&tcpConn.closeFlag) == 1 {
|
|
return errors.New("conn is close")
|
|
}
|
|
|
|
return tcpConn.Write(args)
|
|
}
|
|
|
|
|
|
func (tcpConn *TCPConn) IsConnected() bool {
|
|
return atomic.LoadInt32(&tcpConn.closeFlag) == 0
|
|
}
|
|
|
|
func (tcpConn *TCPConn) SetReadDeadline(d time.Duration) {
|
|
tcpConn.conn.SetReadDeadline(time.Now().Add(d))
|
|
}
|
|
|
|
func (tcpConn *TCPConn) SetWriteDeadline(d time.Duration) {
|
|
tcpConn.conn.SetWriteDeadline(time.Now().Add(d))
|
|
}
|