In Zinx v0.5, the routing mode functionality has been configured, but it can only bind a single routing handler method. Obviously, this cannot meet the basic requirements of a server. Now, we need to add a multi-router mode to Zinx based on the previous implementation.

What is the multi-router mode? It means that we need to associate MsgIDs with their corresponding processing logic. This association needs to be stored in a map data structure, defined as follows:

Apis map[uint32] ziface.IRouter

The key of the map is of type uint32 and stores the IDs of each type of message. The value is an abstraction layer of the IRouter routing business, which should be the Handle method overridden by the user. It is important to note that it is still not recommended to store specific implementation layer Router types here. The reason is that the design module is still based on abstract layer design. The name of this map is Apis.

6.1 Creating the Message Management Module

In this section, we will define a message management module to maintain the binding relationship in the Apis map.

6.1.1 Creating the Abstract Class for the Message Management Module

Create a file named imsghandler.go in the zinx/ziface directory. This file defines the abstract layer interface for message management. The interface for message management is named IMsgHandle and is defined as follows:

//zinx/ziface/imsghandler.go

package ziface
/*
Abstract layer for message management
*/
type IMsgHandle interface {
DoMsgHandler(request IRequest) // Process messages immediately in a non-blocking manner
AddRouter(msgId uint32, router IRouter) // Add specific handling logic for a message
}

There are two methods inside this interface. AddRouter() is used to add a MsgID and a routing relationship to the Apis map. DoMsgHandler() is an interface that calls the specific Handle() method in the Router. The parameter is of type IRequest because Zinx has already put all client message requests into an IRequest with all the relevant message properties.

Translated into English, using the original Markdown format.

6.1.2 Implementation of the Message Management Module

Create a file named msghandler.go in the zinx/znet directory. This file contains the implementation code for the IMsgHandle interface. The specific implementation is as follows:

//zinx/znet/msghandler.go

package znet

import (
"fmt"
"strconv"
"zinx/ziface"
)

type MsgHandle struct {
Apis map[uint32]ziface.IRouter // Map to store the handler methods for each MsgID
}

The MsgHandle struct has an Apis attribute, which is a map that binds MsgIDs to Routers. Next, we provide the constructor method for MsgHandle. The implementation code is as follows:

//zinx/znet/msghandler.go

func NewMsgHandle() *MsgHandle {
return &MsgHandle{
Apis: make(map[uint32]ziface.IRouter),
}
}

In Golang, initializing a map requires using the make keyword to allocate space. Please note this. The MsgHandle struct needs to implement the two interface methods of IMsgHandle: DoMsgHandler() and AddRouter(). The specific implementation is as follows:

//zinx/znet/msghandler.go

// Process messages in a non-blocking manner
func (mh *MsgHandle) DoMsgHandler(request ziface.IRequest) {
handler, ok := mh.Apis[request.GetMsgID()]
if !ok {
fmt.Println("api msgId =", request.GetMsgID(), "is not FOUND!")
return
}
// Execute the corresponding handler methods
handler.PreHandle(request)
handler.Handle(request)
handler.PostHandle(request)
}
// Add specific handling logic for a message
func (mh *MsgHandle) AddRouter(msgId uint32, router ziface.IRouter) {
// 1. Check if the current msg's API handler method already exists
if _, ok := mh.Apis[msgId]; ok {
panic("repeated api, msgId = " + strconv.Itoa(int(msgId)))
}
// 2. Add the binding relationship between msg and api
mh.Apis[msgId] = router
fmt.Println("Add api msgId =", msgId)
}

The DoMsgHandler() method consists of two steps. First, it retrieves the MsgID from the input parameter request and uses the Apis map to get the corresponding Router. If it cannot find a match, it indicates an unrecognized message, and the developer needs to register the callback business Router for that type of message in advance. Second, after obtaining the Router, it sequentially executes the registered PreHandle(), Handle(), and PostHandle() methods in the template order. Once these three methods are executed, the message processing is completed.

The message management module has now been designed. The next step is to integrate this module into the Zinx framework and upgrade it to Zinx v0.6.

Translated into English, using the original Markdown format.

6.2 Implementation of Zinx-V0.6 Code

First, the AddRouter() interface in the IServer abstract layer needs to be modified to include the MsgID parameter, as now we have added MsgID differentiation. The modification is as follows:

//zinx/ziface/iserver.go

package ziface
// Server interface definition
type IServer interface {
// Start the server
Start()
// Stop the server
Stop()
// Start the business service
Serve()
// Route function: Register a route business method for the current server to handle client connections
AddRouter(msgId uint32, router IRouter)
}

Second, in the Server struct of the Server class, the previous Router member, representing the unique message handling business method, should be replaced with the MsgHandler member. After the modification, it should look like this:

//zinx/znet/server.go

type Server struct {
// Server name
Name string
// IP version, e.g., "tcp4" or "tcp6"
IPVersion string
// IP address the server is bound to
IP string
// Port the server is bound to
Port int
// Message handler module of the current server, used to bind MsgIDs with corresponding handling methods
msgHandler ziface.IMsgHandle
}

The constructor function for initializing the Server also needs to be modified to include the initialization of the msgHandler object:

//zinx/znet/server.go

/*
Create a server handler
*/
func NewServer () ziface.IServer {
utils.GlobalObject.Reload()
s := &Server {
Name: utils.GlobalObject.Name,
IPVersion: "tcp4",
IP: utils.GlobalObject.Host,
Port: utils.GlobalObject.TcpPort,
msgHandler: NewMsgHandle(), // Initialize msgHandler
}
return s
}

When the Server is handling connection requests, the creation of a connection also needs to pass the msgHandler as a parameter to the Connection object. The relevant code modification is as follows:

//zinx/znet/server.go

// ... (omitted code)
dealConn := NewConntion(conn, cid, s.msgHandler)
// ... (omitted code)

Next, let’s move on to the Connection object. The Connection object should have a MsgHandler member to look up the callback route method for a message. The modified code is as follows:

//zinx/znet/connection.go

type Connection struct {
// Current socket TCP connection
Conn *net.TCPConn
// Current connection ID, also known as SessionID, which is globally unique
ConnID uint32
// Current connection closed state
isClosed bool
// Message handler, which manages MsgIDs and corresponding handling methods
MsgHandler ziface.IMsgHandle
// Channel to notify that the connection has exited/stopped
ExitBuffChan chan bool
}

The constructor method for creating a connection (NewConntion()) also needs to pass the MsgHandler as a parameter to assign it to the member:

//zinx/znet/connection.go

func NewConntion(conn *net.TCPConn, connID uint32, msgHandler ziface.IMsgHandle) *Connection {
c := &Connection{
Conn: conn,
ConnID: connID,
isClosed: false,
MsgHandler: msgHandler,
ExitBuffChan: make(chan bool, 1),
}
return c
}

Finally, after reading the complete Message data from the conn, when encapsulating it in a Request, and when it's necessary to invoke the routing business, we only need to call the DoMsgHandler() method of MsgHandler from the conn. The relevant code modification is as follows:

//zinx/znet/connection.go

func (c *Connection) StartReader() {
// ... (omitted code)
for {
// ... (omitted code)
// Get the Request data for the current client request
req := Request{
conn: c,
msg: msg,
}
// Execute the corresponding Handle method from the bound message and its handling method
go c.MsgHandler.DoMsgHandler(&req)
}
}

By starting a new Goroutine to handle the DoMsgHandler() method, messages with different MsgIDs will match different processing business flows.

Translated into English, using the original Markdown format.

6.3 Developing an Application using Zinx-V0.6

To test the development using Zinx-V0.6, we will create a server-side application. The code is as follows:

// Server.go
package main

import (
"fmt"
"zinx/ziface"
"zinx/znet"
)

// Ping test custom router
type PingRouter struct {
znet.BaseRouter
}

// Ping Handle
func (this *PingRouter) Handle(request ziface.IRequest) {
fmt.Println("Call PingRouter Handle")
// Read client data first, then write back ping...ping...ping
fmt.Println("recv from client : msgId=", request.GetMsgID(), ", data=", string(request.GetData()))
err := request.GetConnection().SendMsg(0, []byte("ping...ping...ping"))
if err != nil {
fmt.Println(err)
}
}

// HelloZinxRouter Handle
type HelloZinxRouter struct {
znet.BaseRouter
}

func (this *HelloZinxRouter) Handle(request ziface.IRequest) {
fmt.Println("Call HelloZinxRouter Handle")
// Read client data first, then write back ping...ping...ping
fmt.Println("recv from client : msgId=", request.GetMsgID(), ", data=", string(request.GetData()))
err := request.GetConnection().SendMsg(1, []byte("Hello Zinx Router V0.6"))
if err != nil {
fmt.Println(err)
}
}

func main() {
// Create a server handle
s := znet.NewServer()
// Configure routers
s.AddRouter(0, &PingRouter{})
s.AddRouter(1, &HelloZinxRouter{})
// Start the server
s.Serve()
}

The server sets up two routers: one for messages with MsgID 0, which will execute the Handle() method overridden in PingRouter{}, and another for messages with MsgID 1, which will execute the Handle() method overridden in HelloZinxRouter{}.

Next, we’ll create two clients that will send messages with MsgID 0 and MsgID 1 to test if Zinx can handle these different message types.

The first client will be implemented in Client0.go, with the following code:

//Client0.go
package main

import (
"fmt"
"io"
"net"
"time"
"zinx/znet"
)
/*
Simulate client
*/
func main() {
fmt.Println("Client Test ... start")
// Wait for 3 seconds before sending the test request to give the server a chance to start
time.Sleep(3 * time.Second)
conn, err := net.Dial("tcp", "127.0.0.1:7777")
if err != nil {
fmt.Println("client start err, exit!")
return
}
for {
// Pack the message
dp := znet.NewDataPack()
msg, _ := dp.Pack(znet.NewMsgPackage(0, []byte("Zinx V0.6 Client0 Test Message")))
_, err := conn.Write(msg)
if err != nil {
fmt.Println("write error err ", err)
return
}
// Read the head part from the stream
headData := make([]byte, dp.GetHeadLen())
_, err = io.ReadFull(conn, headData) // ReadFull fills the buffer until it's full
if err != nil {
fmt.Println("read head error")
break
}
// Unpack the headData into a message
msgHead, err := dp.Unpack(headData)
if err != nil {
fmt.Println("server unpack err:", err)
return
}
if msgHead.GetDataLen() > 0 {
// The message has data, so we need to read the data part
msg := msgHead.(*znet.Message)
msg.Data = make([]byte, msg.GetDataLen())
// Read the data bytes from the stream based on the dataLen
_, err := io.ReadFull(conn, msg.Data)
if err != nil {
fmt.Println("server unpack data err:", err)
return
}
fmt.Println("==> Recv Msg: ID=", msg.Id, ", len=", msg.DataLen, ", data=", string(msg.Data))
}
time.Sleep(1 * time.Second)
}
}

The Client0 sends a message with MsgID 0 and the content “Zinx V0.6 Client0 Test Message”.

The second client will be implemented in Client1.go, with the following code:

//Client1.go
package main

import (
"fmt"
"io"
"net"
"time"
"zinx/znet"
)
/*
Simulate client
*/
func main() {
fmt.Println("Client Test ... start")
// Wait for 3 seconds before sending the test request to give the server a chance to start
time.Sleep(3 * time.Second)
conn, err := net.Dial("tcp", "127.0.0.1:7777")
if err != nil {
fmt.Println("client start err, exit!")
return
}
for {
// Pack the message
dp := znet.NewDataPack()
msg, _ := dp.Pack(znet.NewMsgPackage(1, []byte("Zinx V0.6 Client1 Test Message")))
_, err := conn.Write(msg)
if err != nil {
fmt.Println("write error err ", err)
return
}
// Read the head part from the stream
headData := make([]byte, dp.GetHeadLen())
_, err = io.ReadFull(conn, headData) // ReadFull fills the buffer until it's full
if err != nil {
fmt.Println("read head error")
break
}
// Unpack the headData into a message
msgHead, err := dp.Unpack(headData)
if err != nil {
fmt.Println("server unpack err:", err)
return
}
if msgHead.GetDataLen() > 0 {
// The message has data, so we need to read the data part
msg := msgHead.(*znet.Message)
msg.Data = make([]byte, msg.GetDataLen())
// Read the data bytes from the stream based on the dataLen
_, err := io.ReadFull(conn, msg.Data)
if err != nil {
fmt.Println("server unpack data err:", err)
return
}
fmt.Println("==> Recv Msg: ID=", msg.Id, ", len=", msg.DataLen, ", data=", string(msg.Data))
}
time.Sleep(1 * time.Second)
}
}

The Client1 sends a message with MsgID 1 and the content "Zinx V0.6 Client1 Test Message".

To run the server and the two clients, execute the following commands in three different terminals:

$ go run Server.go
$ go run Client0.go
$ go run Client1.go

The server output will be as follows:

$ go run Server.go 
Add api msgId = 0
Add api msgId = 1
[START] Server name: zinx v-0.6 demoApp, listenner at IP: 127.0.0.1, Port 7777 is starting
[Zinx] Version: V0.4, MaxConn: 3, MaxPacketSize: 4096
start Zinx server zinx v-0.6 demoApp succ, now listenning...
Reader Goroutine is running
Call PingRouter Handle
recv from client: msgId= 0, data= Zinx V0.6 Client0 Test Message
Reader Goroutine is running
Call HelloZinxRouter Handle
recv from client: msgId= 1, data= Zinx V0.6 Client1 Test Message
Call PingRouter Handle
recv from client: msgId= 0, data= Zinx V0.6 Client0 Test Message
Call HelloZinxRouter Handle
recv from client: msgId= 1, data= Zinx V0.6 Client1 Test Message
Call PingRouter Handle
recv from client: msgId= 0, data= Zinx V0.6 Client0 Test Message
Call HelloZinxRouter Handle
recv from client: msgId= 1, data= Zinx V0.6 Client1 Test Message
// ...

The Client0 output will be as follows:

$ go run Client0.go 
Client Test ... start
==> Recv Msg: ID= 0, len= 18, data= ping...ping...ping
==> Recv Msg: ID= 0, len= 18, data= ping...ping...ping
==> Recv Msg: ID= 0, len= 18, data= ping...ping...ping
// ...

The Client1 output will be as follows:

$ go run Client1.go 
Client Test ... start
==> Recv Msg: ID= 1, len= 22, data= Hello Zinx Router V0.6
==> Recv Msg: ID= 1, len= 22, data= Hello Zinx Router V0.6
==> Recv Msg: ID= 1, len= 22, data= Hello Zinx Router V0.6
// ...

From the results, it can be observed that the server code is now able to handle different message types and perform different logic based on the message ID. Client0 receives only “ping…ping…ping” as a reply, while Client1 receives only “Hello Zinx Router V0.6” as a reply.

7.4 Summary

In conclusion, up to Zinx V0.6, it is possible to handle different business logic based on different message IDs. Zinx provides a basic framework for server-side network communication, allowing developers to define message types, register different handle functions based on the message types, and add them to the server service object using AddRouter(), thus enabling server-side business development capabilities. The next step would be to further upgrade the internal module structure handling in Zinx.

--

--

No responses yet