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.
notify/client.go

288 lines
6.7 KiB
Go

5 years ago
package notify
import (
4 years ago
"context"
4 years ago
"errors"
"fmt"
"math/rand"
5 years ago
"net"
"strings"
"time"
5 years ago
"b612.me/starainrt"
)
5 years ago
5 years ago
// StarNotifyC 为Client端
type StarNotifyC struct {
4 years ago
Connc net.Conn
5 years ago
clientSign map[string]chan string
// FuncLists 当不使用channel时使用此记录调用函数
4 years ago
FuncLists map[string]func(CMsg)
stopSign context.Context
cancel context.CancelFunc
defaultFunc func(CMsg)
4 years ago
// Stop 停止信 号
5 years ago
Stop chan int
// UseChannel 是否使用channel作为信息传递
UseChannel bool
5 years ago
isUDP bool
5 years ago
// Queue 是用来处理收发信息的简单消息队列
Queue *starainrt.StarQueue
5 years ago
// Online 当前链接是否处于活跃状态
4 years ago
Online bool
lockPool map[string]CMsg
5 years ago
}
5 years ago
5 years ago
// CMsg 指明当前客户端被通知的关键字
type CMsg struct {
Key string
Value string
4 years ago
mode string
wait chan int
5 years ago
}
5 years ago
func (star *StarNotifyC) starinitc() {
4 years ago
star.stopSign, star.cancel = context.WithCancel(context.Background())
5 years ago
star.Queue = starainrt.NewQueue()
5 years ago
star.FuncLists = make(map[string]func(CMsg))
star.UseChannel = false
4 years ago
star.Stop = make(chan int, 5)
5 years ago
star.clientSign = make(map[string]chan string)
5 years ago
star.Online = false
4 years ago
star.lockPool = make(map[string]CMsg)
4 years ago
star.Queue.RestoreDuration(time.Second * 2)
5 years ago
}
5 years ago
// Notify 用于获取一个通知
5 years ago
func (star *StarNotifyC) Notify(key string) chan string {
if _, ok := star.clientSign[key]; !ok {
5 years ago
ch := make(chan string, 20)
5 years ago
star.clientSign[key] = ch
5 years ago
}
5 years ago
return star.clientSign[key]
5 years ago
}
5 years ago
func (star *StarNotifyC) store(key, value string) {
if _, ok := star.clientSign[key]; !ok {
5 years ago
ch := make(chan string, 20)
5 years ago
ch <- value
5 years ago
star.clientSign[key] = ch
5 years ago
return
}
5 years ago
star.clientSign[key] <- value
5 years ago
}
5 years ago
// NewNotifyC 用于新建一个Client端进程
5 years ago
func NewNotifyC(netype, value string) (*StarNotifyC, error) {
5 years ago
var err error
5 years ago
var star StarNotifyC
star.starinitc()
5 years ago
star.isUDP = false
if strings.Index(netype, "udp") >= 0 {
star.isUDP = true
}
4 years ago
star.Connc, err = net.Dial(netype, value)
5 years ago
if err != nil {
5 years ago
return nil, err
5 years ago
}
5 years ago
go star.cnotify()
4 years ago
go func() {
4 years ago
<-star.stopSign.Done()
4 years ago
star.Connc.Close()
star.Online = false
return
}()
5 years ago
go func() {
for {
buf := make([]byte, 8192)
4 years ago
n, err := star.Connc.Read(buf)
if n != 0 {
star.Queue.ParseMessage(buf[0:n], star.Connc)
}
5 years ago
if err != nil {
4 years ago
star.Connc.Close()
4 years ago
star.ClientStop()
5 years ago
//star, _ = NewNotifyC(netype, value)
5 years ago
star.Online = false
5 years ago
return
5 years ago
}
}
}()
5 years ago
star.Online = true
5 years ago
return &star, nil
5 years ago
}
5 years ago
// Send 用于向Server端发送数据
5 years ago
func (star *StarNotifyC) Send(name string) error {
4 years ago
return star.SendValue(name, "")
5 years ago
}
func (star *StarNotifyC) SendValueRaw(key string, msg interface{}) error {
encodeData, err := encode(msg)
if err != nil {
return err
}
return star.SendValue(key, string(encodeData))
}
5 years ago
// SendValue 用于向Server端发送key-value类型数据
func (star *StarNotifyC) SendValue(name, value string) error {
4 years ago
var err error
var key []byte
for _, v := range []byte(name) {
if v == byte(124) || v == byte(92) {
key = append(key, byte(92))
}
key = append(key, v)
}
_, err = star.Connc.Write(star.Queue.BuildMessage([]byte("pa" + "||" + string(key) + "||" + value)))
return err
}
func (star *StarNotifyC) trim(name string) string {
var slash bool = false
var key []byte
for _, v := range []byte(name) {
if v == byte(92) && !slash {
slash = true
continue
}
slash = false
key = append(key, v)
}
return string(key)
}
func (star *StarNotifyC) SendValueWaitRaw(key string, msg interface{}, tmout time.Duration) (CMsg, error) {
encodeData, err := encode(msg)
if err != nil {
return CMsg{}, err
}
return star.SendValueWait(key, string(encodeData), tmout)
}
4 years ago
// SendValueWait 用于向Server端发送key-value类型数据并等待结果返回此结果不会通过标准返回流程处理
func (star *StarNotifyC) SendValueWait(name, value string, tmout time.Duration) (CMsg, error) {
var err error
var tmceed <-chan time.Time
if star.UseChannel {
return CMsg{}, errors.New("Do Not Use UseChannel Mode!")
}
rand.Seed(time.Now().UnixNano())
mode := "cr" + fmt.Sprintf("%05d", rand.Intn(99999))
var key []byte
for _, v := range []byte(name) {
if v == byte(124) || v == byte(92) {
key = append(key, byte(92))
}
key = append(key, v)
}
_, err = star.Connc.Write(star.Queue.BuildMessage([]byte(mode + "||" + string(key) + "||" + value)))
if err != nil {
return CMsg{}, err
}
if int64(tmout) > 0 {
tmceed = time.After(tmout)
}
var source CMsg
source.wait = make(chan int, 2)
star.lockPool[mode] = source
select {
case <-source.wait:
res := star.lockPool[mode]
delete(star.lockPool, mode)
return res, nil
case <-tmceed:
return CMsg{}, errors.New("Time Exceed")
}
}
// ReplyMsg 用于向Server端Reply信息
func (star *StarNotifyC) ReplyMsg(data CMsg, name, value string) error {
var err error
var key []byte
for _, v := range []byte(name) {
if v == byte(124) || v == byte(92) {
key = append(key, byte(92))
}
key = append(key, v)
}
_, err = star.Connc.Write(star.Queue.BuildMessage([]byte(data.mode + "||" + string(key) + "||" + value)))
5 years ago
return err
}
5 years ago
func (star *StarNotifyC) cnotify() {
5 years ago
for {
select {
4 years ago
case <-star.stopSign.Done():
5 years ago
return
5 years ago
default:
}
5 years ago
data, err := star.Queue.RestoreOne()
5 years ago
if err != nil {
time.Sleep(time.Millisecond * 20)
continue
}
4 years ago
if string(data.Msg) == "b612ryzstop" {
star.ClientStop()
5 years ago
star.Online = false
5 years ago
return
}
4 years ago
strs := strings.SplitN(string(data.Msg), "||", 3)
if len(strs) < 3 {
5 years ago
continue
}
4 years ago
strs[1] = star.trim(strs[1])
5 years ago
if star.UseChannel {
4 years ago
go star.store(strs[1], strs[2])
5 years ago
} else {
4 years ago
mode, key, value := strs[0], strs[1], strs[2]
if mode[0:2] != "cr" {
if msg, ok := star.FuncLists[key]; ok {
go msg(CMsg{key, value, mode, nil})
} else {
if star.defaultFunc != nil {
go star.defaultFunc(CMsg{key, value, mode, nil})
}
}
5 years ago
} else {
4 years ago
if sa, ok := star.lockPool[mode]; ok {
sa.Key = key
sa.Value = value
sa.mode = mode
star.lockPool[mode] = sa
sa.wait <- 1
} else {
if msg, ok := star.FuncLists[key]; ok {
go msg(CMsg{key, value, mode, nil})
} else {
if star.defaultFunc != nil {
go star.defaultFunc(CMsg{key, value, mode, nil})
}
}
5 years ago
}
}
}
5 years ago
}
}
5 years ago
// ClientStop 终止client端运行
5 years ago
func (star *StarNotifyC) ClientStop() {
5 years ago
if star.isUDP {
star.Send("b612ryzstop")
}
4 years ago
star.cancel()
star.Stop <- 1
4 years ago
star.Stop <- 1
star.Stop <- 1
5 years ago
}
5 years ago
// SetNotify 用于设置关键词的调用函数
func (star *StarNotifyC) SetNotify(name string, data func(CMsg)) {
star.FuncLists[name] = data
}
// SetDefaultNotify 用于设置默认关键词的调用函数
4 years ago
func (star *StarNotifyC) SetDefaultNotify(data func(CMsg)) {
5 years ago
star.defaultFunc = data
}