gf/os/gproc/gproc_comm_receive.go

128 lines
3.3 KiB
Go

// Copyright 2018 gf Author(https://github.com/gogf/gf). All Rights Reserved.
//
// This Source Code Form is subject to the terms of the MIT License.
// If a copy of the MIT was not distributed with this file,
// You can obtain one at https://github.com/gogf/gf.
package gproc
import (
"encoding/json"
"fmt"
"net"
"github.com/gogf/gf/container/gqueue"
"github.com/gogf/gf/container/gtype"
"github.com/gogf/gf/net/gtcp"
"github.com/gogf/gf/os/gfile"
"github.com/gogf/gf/os/glog"
"github.com/gogf/gf/util/gconv"
)
var (
// tcpListened marks whether the receiving listening service started.
tcpListened = gtype.NewBool()
)
// Receive blocks and receives message from other process using local TCP listening.
// Note that, it only enables the TCP listening service when this function called.
func Receive(group ...string) *MsgRequest {
// Use atomic operations to guarantee only one receiver goroutine listening.
if tcpListened.Cas(false, true) {
go receiveTcpListening()
}
var groupName string
if len(group) > 0 {
groupName = group[0]
} else {
groupName = gPROC_COMM_DEFAULT_GRUOP_NAME
}
queue := commReceiveQueues.GetOrSetFuncLock(groupName, func() interface{} {
return gqueue.New(gPROC_MSG_QUEUE_MAX_LENGTH)
}).(*gqueue.Queue)
// Blocking receiving.
if v := queue.Pop(); v != nil {
return v.(*MsgRequest)
}
return nil
}
// receiveTcpListening scans local for available port and starts listening.
func receiveTcpListening() {
var listen *net.TCPListener
// Scan the available port for listening.
for i := gPROC_DEFAULT_TCP_PORT; ; i++ {
addr, err := net.ResolveTCPAddr("tcp", fmt.Sprintf("127.0.0.1:%d", i))
if err != nil {
continue
}
listen, err = net.ListenTCP("tcp", addr)
if err != nil {
continue
}
// Save the port to the pid file.
if err := gfile.PutContents(getCommFilePath(Pid()), gconv.String(i)); err != nil {
panic(err)
}
break
}
// Start listening.
for {
if conn, err := listen.Accept(); err != nil {
glog.Error(err)
} else if conn != nil {
go receiveTcpHandler(gtcp.NewConnByNetConn(conn))
}
}
}
// receiveTcpHandler is the connection handler for receiving data.
func receiveTcpHandler(conn *gtcp.Conn) {
var result []byte
var response MsgResponse
for {
response.Code = 0
response.Message = ""
response.Data = nil
buffer, err := conn.RecvPkg()
if len(buffer) > 0 {
// Package decoding.
msg := new(MsgRequest)
if err := json.Unmarshal(buffer, msg); err != nil {
glog.Error(err)
continue
}
if msg.RecvPid != Pid() {
// Not mine package.
response.Message = fmt.Sprintf("receiver pid not match, target: %d, current: %d", msg.RecvPid, Pid())
} else if v := commReceiveQueues.Get(msg.Group); v == nil {
// Group check.
response.Message = fmt.Sprintf("group [%s] does not exist", msg.Group)
} else {
// Push to buffer queue.
response.Code = 1
v.(*gqueue.Queue).Push(msg)
}
} else {
// Empty package.
response.Message = "empty package"
}
if err == nil {
result, err = json.Marshal(response)
if err != nil {
glog.Error(err)
}
if err := conn.SendPkg(result); err != nil {
glog.Error(err)
}
} else {
// Just close the connection if any error occurs.
if err := conn.Close(); err != nil {
glog.Error(err)
}
break
}
}
}