2020-07-26 17:09:05 +08:00
|
|
|
package service
|
|
|
|
|
|
|
|
import (
|
|
|
|
"log"
|
|
|
|
|
2020-08-08 16:40:10 +08:00
|
|
|
"github.com/tal-tech/go-zero/core/proc"
|
|
|
|
"github.com/tal-tech/go-zero/core/syncx"
|
|
|
|
"github.com/tal-tech/go-zero/core/threading"
|
2020-07-26 17:09:05 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
2021-02-22 22:43:24 +08:00
|
|
|
// Starter is the interface wraps the Start method.
|
2020-07-26 17:09:05 +08:00
|
|
|
Starter interface {
|
|
|
|
Start()
|
|
|
|
}
|
|
|
|
|
2021-02-22 22:43:24 +08:00
|
|
|
// Stopper is the interface wraps the Stop method.
|
2020-07-26 17:09:05 +08:00
|
|
|
Stopper interface {
|
|
|
|
Stop()
|
|
|
|
}
|
|
|
|
|
2021-02-22 22:43:24 +08:00
|
|
|
// Service is the interface that groups Start and Stop methods.
|
2020-07-26 17:09:05 +08:00
|
|
|
Service interface {
|
|
|
|
Starter
|
|
|
|
Stopper
|
|
|
|
}
|
|
|
|
|
2021-02-22 22:43:24 +08:00
|
|
|
// A Group is a group of services.
|
|
|
|
Group struct {
|
2020-07-26 17:09:05 +08:00
|
|
|
services []Service
|
|
|
|
stopOnce func()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2021-02-22 22:43:24 +08:00
|
|
|
// NewGroup returns a Group.
|
|
|
|
func NewGroup() *Group {
|
|
|
|
sg := new(Group)
|
2020-07-26 17:09:05 +08:00
|
|
|
sg.stopOnce = syncx.Once(sg.doStop)
|
|
|
|
return sg
|
|
|
|
}
|
|
|
|
|
2021-02-22 22:43:24 +08:00
|
|
|
// Add adds service into sg.
|
|
|
|
func (sg *Group) Add(service Service) {
|
2020-07-26 17:09:05 +08:00
|
|
|
sg.services = append(sg.services, service)
|
|
|
|
}
|
|
|
|
|
2021-02-22 22:43:24 +08:00
|
|
|
// Start starts the Group.
|
2020-07-26 17:09:05 +08:00
|
|
|
// There should not be any logic code after calling this method, because this method is a blocking one.
|
|
|
|
// Also, quitting this method will close the logx output.
|
2021-02-22 22:43:24 +08:00
|
|
|
func (sg *Group) Start() {
|
2020-07-26 17:09:05 +08:00
|
|
|
proc.AddShutdownListener(func() {
|
|
|
|
log.Println("Shutting down...")
|
|
|
|
sg.stopOnce()
|
|
|
|
})
|
|
|
|
|
|
|
|
sg.doStart()
|
|
|
|
}
|
|
|
|
|
2021-02-22 22:43:24 +08:00
|
|
|
// Stop stops the Group.
|
|
|
|
func (sg *Group) Stop() {
|
2020-07-26 17:09:05 +08:00
|
|
|
sg.stopOnce()
|
|
|
|
}
|
|
|
|
|
2021-02-22 22:43:24 +08:00
|
|
|
func (sg *Group) doStart() {
|
2020-07-26 17:09:05 +08:00
|
|
|
routineGroup := threading.NewRoutineGroup()
|
|
|
|
|
|
|
|
for i := range sg.services {
|
|
|
|
service := sg.services[i]
|
|
|
|
routineGroup.RunSafe(func() {
|
|
|
|
service.Start()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
routineGroup.Wait()
|
|
|
|
}
|
|
|
|
|
2021-02-22 22:43:24 +08:00
|
|
|
func (sg *Group) doStop() {
|
2020-07-26 17:09:05 +08:00
|
|
|
for _, service := range sg.services {
|
|
|
|
service.Stop()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-22 22:43:24 +08:00
|
|
|
// WithStart wraps a start func as a Service.
|
2020-07-26 17:09:05 +08:00
|
|
|
func WithStart(start func()) Service {
|
|
|
|
return startOnlyService{
|
|
|
|
start: start,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-22 22:43:24 +08:00
|
|
|
// WithStarter wraps a Starter as a Service.
|
2020-07-26 17:09:05 +08:00
|
|
|
func WithStarter(start Starter) Service {
|
|
|
|
return starterOnlyService{
|
|
|
|
Starter: start,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type (
|
|
|
|
stopper struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
startOnlyService struct {
|
|
|
|
start func()
|
|
|
|
stopper
|
|
|
|
}
|
|
|
|
|
|
|
|
starterOnlyService struct {
|
|
|
|
Starter
|
|
|
|
stopper
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
func (s stopper) Stop() {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s startOnlyService) Start() {
|
|
|
|
s.start()
|
|
|
|
}
|