123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160 |
- package linker
-
- import (
- "errors"
- "fmt"
- "io"
- "net"
- "runtime"
- "time"
-
- "github.com/wpajqz/linker/utils/convert"
- )
-
- func (s *Server) handleTCPConnection(conn *net.TCPConn) error {
- var ctx Context = &ContextTcp{Conn: conn}
- if s.constructHandler != nil {
- s.constructHandler.Handle(ctx)
- }
-
- defer func() {
- if s.destructHandler != nil {
- s.destructHandler.Handle(ctx)
- }
-
- conn.Close()
- }()
-
- if s.config.ReadBufferSize > 0 {
- conn.SetReadBuffer(s.config.ReadBufferSize)
- }
-
- if s.config.WriteBufferSize > 0 {
- conn.SetWriteBuffer(s.config.WriteBufferSize)
- }
-
- var (
- bType = make([]byte, 4)
- bSequence = make([]byte, 8)
- bHeaderLength = make([]byte, 4)
- bBodyLength = make([]byte, 4)
- sequence int64
- headerLength uint32
- bodyLength uint32
- )
-
- for {
- if s.config.Timeout != 0 {
- conn.SetDeadline(time.Now().Add(s.config.Timeout))
- }
-
- if _, err := io.ReadFull(conn, bType); err != nil {
- return err
- }
-
- if _, err := io.ReadFull(conn, bSequence); err != nil {
- return err
- }
-
- if _, err := io.ReadFull(conn, bHeaderLength); err != nil {
- return err
- }
-
- if _, err := io.ReadFull(conn, bBodyLength); err != nil {
- return err
- }
-
- sequence = convert.BytesToInt64(bSequence)
- headerLength = convert.BytesToUint32(bHeaderLength)
- bodyLength = convert.BytesToUint32(bBodyLength)
- pacLen := headerLength + bodyLength + uint32(20)
-
- if pacLen > s.config.MaxPayload {
- _, file, line, _ := runtime.Caller(1)
- return SystemError{time.Now(), file, line, "packet larger than MaxPayload"}
- }
-
- header := make([]byte, headerLength)
- if _, err := io.ReadFull(conn, header); err != nil {
- return err
- }
-
- body := make([]byte, bodyLength)
- if _, err := io.ReadFull(conn, body); err != nil {
- return err
- }
-
- rp, err := NewPacket(convert.BytesToUint32(bType), sequence, header, body, s.config.PluginForPacketReceiver)
-
- if err != nil {
- return err
- }
-
- ctx = NewContextTcp(conn, rp.Operator, rp.Sequence, rp.Header, rp.Body, s.config)
- go s.handleTCPPacket(ctx, conn, rp)
- }
- }
-
- func (s *Server) handleTCPPacket(ctx Context, conn net.Conn, rp Packet) {
- defer func() {
- if r := recover(); r != nil {
- if s.errorHandler != nil {
- buf := make([]byte, 1<<12)
- n := runtime.Stack(buf, false)
- s.errorHandler(errors.New(string(buf[:n])))
- }
- }
- }()
-
- if rp.Operator == OPERATOR_HEARTBEAT && s.pingHandler != nil {
- s.pingHandler.Handle(ctx)
- ctx.Success(nil)
- }
-
- handler, ok := s.router.handlerContainer[rp.Operator]
- if !ok {
- ctx.Error(StatusInternalServerError, "server don't register your request.")
- }
-
- if rm, ok := s.router.routerMiddleware[rp.Operator]; ok {
- for _, v := range rm {
- ctx = v.Handle(ctx)
- }
- }
-
- for _, v := range s.router.middleware {
- ctx = v.Handle(ctx)
- if tm, ok := v.(TerminateMiddleware); ok {
- tm.Terminate(ctx)
- }
- }
-
- handler.Handle(ctx)
- ctx.Success(nil) // If it don't call the function of Success or Error, deal it by default
- }
-
- // 开始运行Tcp服务
- func (s *Server) RunTCP(name, address string) error {
- tcpAddr, err := net.ResolveTCPAddr(name, address)
- if err != nil {
- return err
- }
-
- listener, err := net.ListenTCP(name, tcpAddr)
- if err != nil {
- return err
- }
-
- defer listener.Close()
-
- fmt.Printf("tcp server running on %s\n", address)
- for {
- conn, err := listener.AcceptTCP()
- if err != nil {
- continue
- }
-
- go s.handleTCPConnection(conn)
- }
- }
|