代码拉取完成,页面将自动刷新
package chanrpc
import (
"errors"
"fmt"
"gitee.com/lichun-123/leaf/conf"
"gitee.com/lichun-123/leaf/log"
"sync"
"time"
"runtime"
)
// one server per goroutine (goroutine not safe)
// one client per goroutine (goroutine not safe)
type Server struct {
// id -> function
//
// function:
// func(args []interface{})
// func(args []interface{}) interface{}
// func(args []interface{}) []interface{}
functions map[interface{}]interface{}
ChanCall chan *CallInfo
}
type CallInfo struct {
f interface{}
args []interface{}
chanRet chan *RetInfo
cb interface{}
}
type RetInfo struct {
// nil
// interface{}
// []interface{}
ret interface{}
err error
// callback:
// func(err error)
// func(ret interface{}, err error)
// func(ret []interface{}, err error)
cb interface{}
}
type Client struct {
s *Server
chanSyncRet chan *RetInfo
ChanAsynRet chan *RetInfo
pendingAsynCall int
}
func NewServer(l int) *Server {
s := new(Server)
s.functions = make(map[interface{}]interface{})
s.ChanCall = make(chan *CallInfo, l)
return s
}
func assert(i interface{}) []interface{} {
if i == nil {
return nil
} else {
return i.([]interface{})
}
}
// you must call the function before calling Open and Go
func (s *Server) Register(id interface{}, f interface{}) {
switch f.(type) {
case func([]interface{}):
case func([]interface{}) interface{}:
case func([]interface{}) []interface{}:
default:
panic(fmt.Sprintf("function id %v: definition of function is invalid", id))
}
if _, ok := s.functions[id]; ok {
panic(fmt.Sprintf("function id %v: already registered", id))
}
s.functions[id] = f
}
func (s *Server) ret(ci *CallInfo, ri *RetInfo) (err error) {
if ci.chanRet == nil {
return
}
defer func() {
if r := recover(); r != nil {
err = r.(error)
}
}()
ri.cb = ci.cb
ci.chanRet <- ri
return
}
func (s *Server) exec(ci *CallInfo) (err error) {
defer func() {
if r := recover(); r != nil {
if conf.LenStackBuf > 0 {
buf := make([]byte, conf.LenStackBuf)
l := runtime.Stack(buf, false)
err = fmt.Errorf("%v: %s", r, buf[:l])
} else {
err = fmt.Errorf("%v", r)
}
log.Fatal("(s *Server) exec recover ")
s.ret(ci, &RetInfo{err: fmt.Errorf("%v", r)})
}
}()
// execute
switch ci.f.(type) {
case func([]interface{}):
var wg sync.WaitGroup
wg.Add(1) // 增加等待组的计数器
go func() {
defer wg.Done() // 在 B 协程完成时减少计数器
ci.f.(func([]interface{}))(ci.args)
}()
wg.Wait() // 等待 B 协程完成
return s.ret(ci, &RetInfo{})
case func([]interface{}) interface{}:
ret := ci.f.(func([]interface{}) interface{})(ci.args)
return s.ret(ci, &RetInfo{ret: ret})
case func([]interface{}) []interface{}:
ret := ci.f.(func([]interface{}) []interface{})(ci.args)
return s.ret(ci, &RetInfo{ret: ret})
}
panic("bug")
}
func (s *Server) Exec(ci *CallInfo) {
tar := time.Now().Unix()
err := s.exec(ci)
if err != nil {
log.Error("%v", err)
}
if time.Now().Unix()-tar >= 2 {
log.Error("(s *Server) exec 处理时间慢")
}
}
// goroutine safe
func (s *Server) Go(id interface{}, args ...interface{}) {
f := s.functions[id]
if f == nil {
return
}
defer func() {
recover()
}()
s.ChanCall <- &CallInfo{
f: f,
args: args,
}
if len(s.ChanCall) > 10000 {
log.Error("通道负载 :", len(s.ChanCall))
}
}
// goroutine safe
func (s *Server) Call0(id interface{}, args ...interface{}) error {
return s.Open(0).Call0(id, args...)
}
// goroutine safe
func (s *Server) Call1(id interface{}, args ...interface{}) (interface{}, error) {
return s.Open(0).Call1(id, args...)
}
// goroutine safe
func (s *Server) CallN(id interface{}, args ...interface{}) ([]interface{}, error) {
return s.Open(0).CallN(id, args...)
}
func (s *Server) Close() {
close(s.ChanCall)
for ci := range s.ChanCall {
s.ret(ci, &RetInfo{
err: errors.New("chanrpc server closed"),
})
}
}
// goroutine safe
func (s *Server) Open(l int) *Client {
c := NewClient(l)
c.Attach(s)
return c
}
func NewClient(l int) *Client {
c := new(Client)
c.chanSyncRet = make(chan *RetInfo, 1)
c.ChanAsynRet = make(chan *RetInfo, l)
return c
}
func (c *Client) Attach(s *Server) {
c.s = s
}
func (c *Client) call(ci *CallInfo, block bool) (err error) {
defer func() {
if r := recover(); r != nil {
err = r.(error)
}
}()
if block {
c.s.ChanCall <- ci
} else {
select {
case c.s.ChanCall <- ci:
default:
err = errors.New("chanrpc channel full")
}
}
return
}
func (c *Client) f(id interface{}, n int) (f interface{}, err error) {
if c.s == nil {
err = errors.New("server not attached")
return
}
f = c.s.functions[id]
if f == nil {
err = fmt.Errorf("function id %v: function not registered", id)
return
}
var ok bool
switch n {
case 0:
_, ok = f.(func([]interface{}))
case 1:
_, ok = f.(func([]interface{}) interface{})
case 2:
_, ok = f.(func([]interface{}) []interface{})
default:
panic("bug")
}
if !ok {
err = fmt.Errorf("function id %v: return type mismatch", id)
}
return
}
func (c *Client) Call0(id interface{}, args ...interface{}) error {
f, err := c.f(id, 0)
if err != nil {
return err
}
err = c.call(&CallInfo{
f: f,
args: args,
chanRet: c.chanSyncRet,
}, true)
if err != nil {
return err
}
ri := <-c.chanSyncRet
return ri.err
}
func (c *Client) Call1(id interface{}, args ...interface{}) (interface{}, error) {
f, err := c.f(id, 1)
if err != nil {
return nil, err
}
err = c.call(&CallInfo{
f: f,
args: args,
chanRet: c.chanSyncRet,
}, true)
if err != nil {
return nil, err
}
ri := <-c.chanSyncRet
return ri.ret, ri.err
}
func (c *Client) CallN(id interface{}, args ...interface{}) ([]interface{}, error) {
f, err := c.f(id, 2)
if err != nil {
return nil, err
}
err = c.call(&CallInfo{
f: f,
args: args,
chanRet: c.chanSyncRet,
}, true)
if err != nil {
return nil, err
}
ri := <-c.chanSyncRet
return assert(ri.ret), ri.err
}
func (c *Client) asynCall(id interface{}, args []interface{}, cb interface{}, n int) {
f, err := c.f(id, n)
if err != nil {
c.ChanAsynRet <- &RetInfo{err: err, cb: cb}
return
}
err = c.call(&CallInfo{
f: f,
args: args,
chanRet: c.ChanAsynRet,
cb: cb,
}, false)
if err != nil {
c.ChanAsynRet <- &RetInfo{err: err, cb: cb}
return
}
}
func (c *Client) AsynCall(id interface{}, _args ...interface{}) {
if len(_args) < 1 {
panic("callback function not found")
}
args := _args[:len(_args)-1]
cb := _args[len(_args)-1]
var n int
switch cb.(type) {
case func(error):
n = 0
case func(interface{}, error):
n = 1
case func([]interface{}, error):
n = 2
default:
panic("definition of callback function is invalid")
}
// too many calls
if c.pendingAsynCall >= cap(c.ChanAsynRet) {
execCb(&RetInfo{err: errors.New("too many calls"), cb: cb})
return
}
c.asynCall(id, args, cb, n)
c.pendingAsynCall++
}
func execCb(ri *RetInfo) {
defer func() {
if r := recover(); r != nil {
if conf.LenStackBuf > 0 {
buf := make([]byte, conf.LenStackBuf)
l := runtime.Stack(buf, false)
log.Error("%v: %s", r, buf[:l])
} else {
log.Error("%v", r)
}
}
}()
// execute
switch ri.cb.(type) {
case func(error):
ri.cb.(func(error))(ri.err)
case func(interface{}, error):
ri.cb.(func(interface{}, error))(ri.ret, ri.err)
case func([]interface{}, error):
ri.cb.(func([]interface{}, error))(assert(ri.ret), ri.err)
default:
panic("bug")
}
return
}
func (c *Client) Cb(ri *RetInfo) {
c.pendingAsynCall--
execCb(ri)
}
func (c *Client) Close() {
for c.pendingAsynCall > 0 {
c.Cb(<-c.ChanAsynRet)
}
}
func (c *Client) Idle() bool {
return c.pendingAsynCall == 0
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。