You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			67 lines
		
	
	
		
			1.8 KiB
		
	
	
	
		
			Go
		
	
			
		
		
	
	
			67 lines
		
	
	
		
			1.8 KiB
		
	
	
	
		
			Go
		
	
| package protocol
 | |
| 
 | |
| import (
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"git.noahlan.cn/noahlan/nnet/core"
 | |
| 	"git.noahlan.cn/noahlan/nnet/entity"
 | |
| 	"git.noahlan.cn/noahlan/ntools-go/core/nlog"
 | |
| )
 | |
| 
 | |
| type (
 | |
| 	HandshakeValidatorFunc  func([]byte) error
 | |
| 	HandshakeAckBuilderFunc func() (interface{}, error)
 | |
| )
 | |
| 
 | |
| func WithNNetPipeline(ackDataBuilder HandshakeAckBuilderFunc, validator HandshakeValidatorFunc) core.RunOption {
 | |
| 	packer := NewNNetPacker()
 | |
| 	return func(server *core.Server) {
 | |
| 		server.Pipeline().Inbound().PushFront(func(entity entity.NetworkEntity, v interface{}) error {
 | |
| 			pkg, ok := v.(*NNetPacket)
 | |
| 			if !ok {
 | |
| 				return ErrWrongPacketType
 | |
| 			}
 | |
| 			conn, _ := entity.Conn()
 | |
| 
 | |
| 			switch pkg.PacketType {
 | |
| 			case Handshake:
 | |
| 				if err := validator(pkg.Data); err != nil {
 | |
| 					return err
 | |
| 				}
 | |
| 				data, err := ackDataBuilder()
 | |
| 				nlog.Must(err)
 | |
| 
 | |
| 				hrd, _ := packer.Pack(Handshake, data)
 | |
| 				if err := entity.SendBytes(hrd); err != nil {
 | |
| 					return err
 | |
| 				}
 | |
| 				entity.SetStatus(core.StatusPrepare)
 | |
| 				nlog.Debugf("connection handshake Id=%d, Remote=%s", entity.Session().ID(), conn.RemoteAddr())
 | |
| 			case HandshakeAck:
 | |
| 				entity.SetStatus(core.StatusPending)
 | |
| 				nlog.Debugf("receive handshake ACK Id=%d, Remote=%s", entity.Session().ID(), conn.RemoteAddr())
 | |
| 			case Heartbeat:
 | |
| 				// Expected
 | |
| 			case Data:
 | |
| 				if entity.Status() < core.StatusPending {
 | |
| 					return errors.New(fmt.Sprintf("receive data on socket which not yet ACK, session will be closed immediately, remote=%s",
 | |
| 						conn.RemoteAddr()))
 | |
| 				}
 | |
| 				entity.SetStatus(core.StatusWorking)
 | |
| 
 | |
| 				var lastMid uint64
 | |
| 				switch pkg.MsgType {
 | |
| 				case Request:
 | |
| 					lastMid = pkg.ID
 | |
| 				case Notify:
 | |
| 					lastMid = 0
 | |
| 				default:
 | |
| 					return fmt.Errorf("Invalid message type: %s ", pkg.MsgType.String())
 | |
| 				}
 | |
| 				entity.SetLastMID(lastMid)
 | |
| 			}
 | |
| 			return nil
 | |
| 		})
 | |
| 	}
 | |
| }
 |