mirror of
https://github.com/duanhf2012/origin.git
synced 2026-02-03 22:45:13 +08:00
390 lines
9.4 KiB
Go
390 lines
9.4 KiB
Go
package service
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"github.com/duanhf2012/origin/event"
|
|
"github.com/duanhf2012/origin/log"
|
|
"github.com/duanhf2012/origin/profiler"
|
|
"github.com/duanhf2012/origin/rpc"
|
|
"github.com/duanhf2012/origin/util/timer"
|
|
"reflect"
|
|
"runtime"
|
|
"strconv"
|
|
"sync"
|
|
"sync/atomic"
|
|
"github.com/duanhf2012/origin/concurrent"
|
|
)
|
|
|
|
var timerDispatcherLen = 100000
|
|
var maxServiceEventChannelNum = 2000000
|
|
|
|
type IService interface {
|
|
concurrent.IConcurrent
|
|
Init(iService IService,getClientFun rpc.FuncRpcClient,getServerFun rpc.FuncRpcServer,serviceCfg interface{})
|
|
Stop()
|
|
Start()
|
|
|
|
OnSetup(iService IService)
|
|
OnInit() error
|
|
OnStart()
|
|
OnRelease()
|
|
|
|
SetName(serviceName string)
|
|
GetName() string
|
|
GetRpcHandler() rpc.IRpcHandler
|
|
GetServiceCfg()interface{}
|
|
GetProfiler() *profiler.Profiler
|
|
GetServiceEventChannelNum() int
|
|
GetServiceTimerChannelNum() int
|
|
|
|
SetEventChannelNum(num int)
|
|
OpenProfiler()
|
|
}
|
|
|
|
type Service struct {
|
|
Module
|
|
|
|
rpcHandler rpc.RpcHandler //rpc
|
|
name string //service name
|
|
wg sync.WaitGroup
|
|
serviceCfg interface{}
|
|
goroutineNum int32
|
|
startStatus bool
|
|
eventProcessor event.IEventProcessor
|
|
profiler *profiler.Profiler //性能分析器
|
|
nodeEventLister rpc.INodeListener
|
|
discoveryServiceLister rpc.IDiscoveryServiceListener
|
|
chanEvent chan event.IEvent
|
|
closeSig chan struct{}
|
|
}
|
|
|
|
// RpcConnEvent Node结点连接事件
|
|
type RpcConnEvent struct{
|
|
IsConnect bool
|
|
NodeId int
|
|
}
|
|
|
|
// DiscoveryServiceEvent 发现服务结点
|
|
type DiscoveryServiceEvent struct{
|
|
IsDiscovery bool
|
|
ServiceName []string
|
|
NodeId int
|
|
}
|
|
|
|
func SetMaxServiceChannel(maxEventChannel int){
|
|
maxServiceEventChannelNum = maxEventChannel
|
|
}
|
|
|
|
func (rpcEventData *DiscoveryServiceEvent) GetEventType() event.EventType{
|
|
return event.Sys_Event_DiscoverService
|
|
}
|
|
|
|
func (rpcEventData *RpcConnEvent) GetEventType() event.EventType{
|
|
return event.Sys_Event_Node_Event
|
|
}
|
|
|
|
func (s *Service) OnSetup(iService IService){
|
|
if iService.GetName() == "" {
|
|
s.name = reflect.Indirect(reflect.ValueOf(iService)).Type().Name()
|
|
}
|
|
}
|
|
|
|
func (s *Service) OpenProfiler() {
|
|
s.profiler = profiler.RegProfiler(s.GetName())
|
|
if s.profiler==nil {
|
|
log.SFatal("rofiler.RegProfiler ",s.GetName()," fail.")
|
|
}
|
|
}
|
|
|
|
func (s *Service) Init(iService IService,getClientFun rpc.FuncRpcClient,getServerFun rpc.FuncRpcServer,serviceCfg interface{}) {
|
|
s.closeSig = make(chan struct{})
|
|
s.dispatcher =timer.NewDispatcher(timerDispatcherLen)
|
|
if s.chanEvent == nil {
|
|
s.chanEvent = make(chan event.IEvent,maxServiceEventChannelNum)
|
|
}
|
|
|
|
s.rpcHandler.InitRpcHandler(iService.(rpc.IRpcHandler),getClientFun,getServerFun,iService.(rpc.IRpcHandlerChannel))
|
|
s.IRpcHandler = &s.rpcHandler
|
|
s.self = iService.(IModule)
|
|
//初始化祖先
|
|
s.ancestor = iService.(IModule)
|
|
s.seedModuleId =InitModuleId
|
|
s.descendants = map[uint32]IModule{}
|
|
s.serviceCfg = serviceCfg
|
|
s.goroutineNum = 1
|
|
s.eventProcessor = event.NewEventProcessor()
|
|
s.eventProcessor.Init(s)
|
|
s.eventHandler = event.NewEventHandler()
|
|
s.eventHandler.Init(s.eventProcessor)
|
|
s.Module.IConcurrent = &concurrent.Concurrent{}
|
|
}
|
|
|
|
func (s *Service) Start() {
|
|
s.startStatus = true
|
|
var waitRun sync.WaitGroup
|
|
|
|
for i:=int32(0);i< s.goroutineNum;i++{
|
|
s.wg.Add(1)
|
|
waitRun.Add(1)
|
|
go func(){
|
|
log.SRelease(s.GetName()," service is running",)
|
|
waitRun.Done()
|
|
s.Run()
|
|
}()
|
|
}
|
|
|
|
waitRun.Wait()
|
|
}
|
|
|
|
func (s *Service) Run() {
|
|
defer s.wg.Done()
|
|
var bStop = false
|
|
|
|
concurrent := s.IConcurrent.(*concurrent.Concurrent)
|
|
concurrentCBChannel := concurrent.GetCallBackChannel()
|
|
|
|
s.self.(IService).OnStart()
|
|
for{
|
|
var analyzer *profiler.Analyzer
|
|
select {
|
|
case <- s.closeSig:
|
|
bStop = true
|
|
concurrent.Close()
|
|
case cb:=<-concurrentCBChannel:
|
|
concurrent.DoCallback(cb)
|
|
case ev := <- s.chanEvent:
|
|
switch ev.GetEventType() {
|
|
case event.ServiceRpcRequestEvent:
|
|
cEvent,ok := ev.(*event.Event)
|
|
if ok == false {
|
|
log.SError("Type event conversion error")
|
|
break
|
|
}
|
|
rpcRequest,ok := cEvent.Data.(*rpc.RpcRequest)
|
|
if ok == false {
|
|
log.SError("Type *rpc.RpcRequest conversion error")
|
|
break
|
|
}
|
|
if s.profiler!=nil {
|
|
analyzer = s.profiler.Push("[Req]"+rpcRequest.RpcRequestData.GetServiceMethod())
|
|
}
|
|
|
|
s.GetRpcHandler().HandlerRpcRequest(rpcRequest)
|
|
if analyzer!=nil {
|
|
analyzer.Pop()
|
|
analyzer = nil
|
|
}
|
|
event.DeleteEvent(cEvent)
|
|
case event.ServiceRpcResponseEvent:
|
|
cEvent,ok := ev.(*event.Event)
|
|
if ok == false {
|
|
log.SError("Type event conversion error")
|
|
break
|
|
}
|
|
rpcResponseCB,ok := cEvent.Data.(*rpc.Call)
|
|
if ok == false {
|
|
log.SError("Type *rpc.Call conversion error")
|
|
break
|
|
}
|
|
if s.profiler!=nil {
|
|
analyzer = s.profiler.Push("[Res]" + rpcResponseCB.ServiceMethod)
|
|
}
|
|
s.GetRpcHandler().HandlerRpcResponseCB(rpcResponseCB)
|
|
if analyzer!=nil {
|
|
analyzer.Pop()
|
|
analyzer = nil
|
|
}
|
|
event.DeleteEvent(cEvent)
|
|
default:
|
|
if s.profiler!=nil {
|
|
analyzer = s.profiler.Push("[SEvent]"+strconv.Itoa(int(ev.GetEventType())))
|
|
}
|
|
s.eventProcessor.EventHandler(ev)
|
|
if analyzer!=nil {
|
|
analyzer.Pop()
|
|
analyzer = nil
|
|
}
|
|
}
|
|
|
|
case t := <- s.dispatcher.ChanTimer:
|
|
if s.profiler != nil {
|
|
analyzer = s.profiler.Push("[timer]"+t.GetName())
|
|
}
|
|
t.Do()
|
|
if analyzer != nil {
|
|
analyzer.Pop()
|
|
analyzer = nil
|
|
}
|
|
}
|
|
|
|
if bStop == true {
|
|
if atomic.AddInt32(&s.goroutineNum,-1)<=0 {
|
|
s.startStatus = false
|
|
s.Release()
|
|
}
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s *Service) GetName() string{
|
|
return s.name
|
|
}
|
|
|
|
func (s *Service) SetName(serviceName string) {
|
|
s.name = serviceName
|
|
}
|
|
|
|
func (s *Service) Release(){
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
buf := make([]byte, 4096)
|
|
l := runtime.Stack(buf, false)
|
|
errString := fmt.Sprint(r)
|
|
log.SError("core dump info[",errString,"]\n",string(buf[:l]))
|
|
}
|
|
}()
|
|
|
|
s.self.OnRelease()
|
|
}
|
|
|
|
func (s *Service) OnRelease(){
|
|
}
|
|
|
|
func (s *Service) OnInit() error {
|
|
return nil
|
|
}
|
|
|
|
func (s *Service) Stop(){
|
|
log.SRelease("stop ",s.GetName()," service ")
|
|
close(s.closeSig)
|
|
s.wg.Wait()
|
|
log.SRelease(s.GetName()," service has been stopped")
|
|
}
|
|
|
|
func (s *Service) GetServiceCfg()interface{}{
|
|
return s.serviceCfg
|
|
}
|
|
|
|
func (s *Service) GetProfiler() *profiler.Profiler{
|
|
return s.profiler
|
|
}
|
|
|
|
func (s *Service) RegEventReceiverFunc(eventType event.EventType, receiver event.IEventHandler,callback event.EventCallBack){
|
|
s.eventProcessor.RegEventReceiverFunc(eventType, receiver,callback)
|
|
}
|
|
|
|
func (s *Service) UnRegEventReceiverFunc(eventType event.EventType, receiver event.IEventHandler){
|
|
s.eventProcessor.UnRegEventReceiverFun(eventType, receiver)
|
|
}
|
|
|
|
func (s *Service) IsSingleCoroutine() bool {
|
|
return s.goroutineNum == 1
|
|
}
|
|
|
|
func (s *Service) RegRawRpc(rpcMethodId uint32,rawRpcCB rpc.RawRpcCallBack){
|
|
s.rpcHandler.RegRawRpc(rpcMethodId,rawRpcCB)
|
|
}
|
|
|
|
func (s *Service) OnStart(){
|
|
}
|
|
|
|
func (s *Service) OnNodeEvent(ev event.IEvent){
|
|
event := ev.(*RpcConnEvent)
|
|
if event.IsConnect {
|
|
s.nodeEventLister.OnNodeConnected(event.NodeId)
|
|
}else{
|
|
s.nodeEventLister.OnNodeDisconnect(event.NodeId)
|
|
}
|
|
}
|
|
|
|
func (s *Service) OnDiscoverServiceEvent(ev event.IEvent){
|
|
event := ev.(*DiscoveryServiceEvent)
|
|
if event.IsDiscovery {
|
|
s.discoveryServiceLister.OnDiscoveryService(event.NodeId,event.ServiceName)
|
|
}else{
|
|
s.discoveryServiceLister.OnUnDiscoveryService(event.NodeId,event.ServiceName)
|
|
}
|
|
}
|
|
|
|
func (s *Service) RegRpcListener(rpcEventLister rpc.INodeListener) {
|
|
s.nodeEventLister = rpcEventLister
|
|
s.RegEventReceiverFunc(event.Sys_Event_Node_Event,s.GetEventHandler(),s.OnNodeEvent)
|
|
RegRpcEventFun(s.GetName())
|
|
}
|
|
|
|
func (s *Service) UnRegRpcListener(rpcLister rpc.INodeListener) {
|
|
s.UnRegEventReceiverFunc(event.Sys_Event_Node_Event,s.GetEventHandler())
|
|
UnRegRpcEventFun(s.GetName())
|
|
}
|
|
|
|
func (s *Service) RegDiscoverListener(discoveryServiceListener rpc.IDiscoveryServiceListener) {
|
|
s.discoveryServiceLister = discoveryServiceListener
|
|
s.RegEventReceiverFunc(event.Sys_Event_DiscoverService,s.GetEventHandler(),s.OnDiscoverServiceEvent)
|
|
RegDiscoveryServiceEventFun(s.GetName())
|
|
}
|
|
|
|
func (s *Service) UnRegDiscoverListener(rpcLister rpc.INodeListener) {
|
|
s.UnRegEventReceiverFunc(event.Sys_Event_DiscoverService,s.GetEventHandler())
|
|
UnRegDiscoveryServiceEventFun(s.GetName())
|
|
}
|
|
|
|
func (s *Service) PushRpcRequest(rpcRequest *rpc.RpcRequest) error{
|
|
ev := event.NewEvent()
|
|
ev.Type = event.ServiceRpcRequestEvent
|
|
ev.Data = rpcRequest
|
|
|
|
return s.pushEvent(ev)
|
|
}
|
|
|
|
func (s *Service) PushRpcResponse(call *rpc.Call) error{
|
|
ev := event.NewEvent()
|
|
ev.Type = event.ServiceRpcResponseEvent
|
|
ev.Data = call
|
|
|
|
return s.pushEvent(ev)
|
|
}
|
|
|
|
func (s *Service) PushEvent(ev event.IEvent) error{
|
|
return s.pushEvent(ev)
|
|
}
|
|
|
|
func (s *Service) pushEvent(ev event.IEvent) error{
|
|
if len(s.chanEvent) >= maxServiceEventChannelNum {
|
|
err := errors.New("The event channel in the service is full")
|
|
log.SError(err.Error())
|
|
return err
|
|
}
|
|
|
|
s.chanEvent <- ev
|
|
return nil
|
|
}
|
|
|
|
func (s *Service) GetServiceEventChannelNum() int{
|
|
return len(s.chanEvent)
|
|
}
|
|
|
|
func (s *Service) GetServiceTimerChannelNum() int{
|
|
return len(s.dispatcher.ChanTimer)
|
|
}
|
|
|
|
func (s *Service) SetEventChannelNum(num int){
|
|
if s.chanEvent == nil {
|
|
s.chanEvent = make(chan event.IEvent,num)
|
|
}else {
|
|
panic("this stage cannot be set")
|
|
}
|
|
}
|
|
|
|
func (s *Service) SetGoRoutineNum(goroutineNum int32) bool {
|
|
//已经开始状态不允许修改协程数量,打开性能分析器不允许开多线程
|
|
if s.startStatus == true || s.profiler!=nil {
|
|
log.SError("open profiler mode is not allowed to set Multi-coroutine.")
|
|
return false
|
|
}
|
|
|
|
s.goroutineNum = goroutineNum
|
|
return true
|
|
}
|