Files
gf/os/gproc/gproc_comm_receive.go

136 lines
3.5 KiB
Go
Raw Normal View History

2021-01-17 21:46:25 +08:00
// Copyright GoFrame Author(https://goframe.org). 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.
2019-02-20 16:07:11 +08:00
package gproc
import (
2021-09-27 21:27:24 +08:00
"context"
"fmt"
"net"
2019-07-29 21:01:19 +08:00
2021-10-11 21:41:56 +08:00
"github.com/gogf/gf/v2/container/gqueue"
"github.com/gogf/gf/v2/container/gtype"
2022-02-16 20:51:39 +08:00
"github.com/gogf/gf/v2/errors/gerror"
2021-11-15 20:49:02 +08:00
"github.com/gogf/gf/v2/internal/json"
2021-10-11 21:41:56 +08:00
"github.com/gogf/gf/v2/net/gtcp"
"github.com/gogf/gf/v2/os/gfile"
"github.com/gogf/gf/v2/os/glog"
"github.com/gogf/gf/v2/util/gconv"
)
var (
// tcpListened marks whether the receiving listening service started.
2019-06-19 09:06:52 +08:00
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()
2019-06-19 09:06:52 +08:00
}
var groupName string
2019-06-19 09:06:52 +08:00
if len(group) > 0 {
groupName = group[0]
} else {
groupName = defaultGroupNameForProcComm
2019-06-19 09:06:52 +08:00
}
queue := commReceiveQueues.GetOrSetFuncLock(groupName, func() interface{} {
return gqueue.New(maxLengthForProcMsgQueue)
}).(*gqueue.Queue)
2019-02-20 16:07:11 +08:00
// Blocking receiving.
2019-06-19 09:06:52 +08:00
if v := queue.Pop(); v != nil {
return v.(*MsgRequest)
2019-06-19 09:06:52 +08:00
}
return nil
2019-02-20 16:07:11 +08:00
}
// receiveTcpListening scans local for available port and starts listening.
func receiveTcpListening() {
2022-02-16 20:51:39 +08:00
var (
listen *net.TCPListener
conn net.Conn
port = gtcp.MustGetFreePort()
address = fmt.Sprintf("127.0.0.1:%d", port)
)
tcpAddress, err := net.ResolveTCPAddr("tcp", address)
if err != nil {
panic(gerror.Wrap(err, `net.ResolveTCPAddr failed`))
}
listen, err = net.ListenTCP("tcp", tcpAddress)
if err != nil {
panic(gerror.Wrapf(err, `net.ListenTCP failed for address "%s"`, address))
}
// Save the port to the pid file.
if err = gfile.PutContents(getCommFilePath(Pid()), gconv.String(port)); err != nil {
panic(err)
2019-06-19 09:06:52 +08:00
}
// Start listening.
2019-06-19 09:06:52 +08:00
for {
2022-02-16 20:51:39 +08:00
if conn, err = listen.Accept(); err != nil {
2021-09-27 21:27:24 +08:00
glog.Error(context.TODO(), err)
2019-06-19 09:06:52 +08:00
} else if conn != nil {
go receiveTcpHandler(gtcp.NewConnByNetConn(conn))
2019-06-19 09:06:52 +08:00
}
}
}
// receiveTcpHandler is the connection handler for receiving data.
func receiveTcpHandler(conn *gtcp.Conn) {
var (
2021-09-27 21:27:24 +08:00
ctx = context.TODO()
result []byte
response MsgResponse
)
2019-06-19 09:06:52 +08:00
for {
response.Code = 0
response.Message = ""
response.Data = nil
buffer, err := conn.RecvPkg()
2019-06-19 09:06:52 +08:00
if len(buffer) > 0 {
// Package decoding.
msg := new(MsgRequest)
if err = json.UnmarshalUseNumber(buffer, msg); err != nil {
2019-06-19 09:06:52 +08:00
continue
}
2022-02-16 20:51:39 +08:00
if msg.ReceiverPid != Pid() {
// Not mine package.
2022-02-16 20:51:39 +08:00
response.Message = fmt.Sprintf(
"receiver pid not match, target: %d, current: %d",
msg.ReceiverPid, Pid(),
)
} else if v := commReceiveQueues.Get(msg.Group); v == nil {
// Group check.
response.Message = fmt.Sprintf("group [%s] does not exist", msg.Group)
2019-06-19 09:06:52 +08:00
} else {
// Push to buffer queue.
response.Code = 1
v.(*gqueue.Queue).Push(msg)
2019-06-19 09:06:52 +08:00
}
} else {
// Empty package.
response.Message = "empty package"
2019-06-19 09:06:52 +08:00
}
if err == nil {
result, err = json.Marshal(response)
if err != nil {
2021-09-27 21:27:24 +08:00
glog.Error(ctx, err)
}
if err = conn.SendPkg(result); err != nil {
2021-09-27 21:27:24 +08:00
glog.Error(ctx, err)
2019-06-19 09:06:52 +08:00
}
} else {
// Just close the connection if any error occurs.
if err = conn.Close(); err != nil {
2021-09-27 21:27:24 +08:00
glog.Error(ctx, err)
2019-06-19 09:06:52 +08:00
}
break
2019-06-19 09:06:52 +08:00
}
}
}