mirror of
https://github.com/status-im/go-waku.git
synced 2025-01-15 00:04:36 +00:00
3226def4cf
* refactor discovery and common service to separate package to remove package inter-dependencies * relay on-demand discovery ,use proto to enr field mapping * chore: no need to dial discovered peers as peermanager already does that * on demand discovery for service peers during peer selection * identify supported protocols for discovered peers and add to service slots * fix: tests to use proper static sharding topics * fix: random selection with default pubsubTopic --------- Co-authored-by: richΛrd <info@richardramos.me>
74 lines
1.7 KiB
Go
74 lines
1.7 KiB
Go
package service
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"sync"
|
|
)
|
|
|
|
// this is common layout for all the services that require mutex protection and a guarantee that all running goroutines will be finished before stop finishes execution. This guarantee comes from waitGroup all one has to use CommonService.WaitGroup() in the goroutines that should finish by the end of stop function.
|
|
type CommonService struct {
|
|
sync.RWMutex
|
|
cancel context.CancelFunc
|
|
ctx context.Context
|
|
wg sync.WaitGroup
|
|
started bool
|
|
}
|
|
|
|
func NewCommonService() *CommonService {
|
|
return &CommonService{
|
|
wg: sync.WaitGroup{},
|
|
RWMutex: sync.RWMutex{},
|
|
}
|
|
}
|
|
|
|
// mutex protected start function
|
|
// creates internal context over provided context and runs fn safely
|
|
// fn is excerpt to be executed to start the protocol
|
|
func (sp *CommonService) Start(ctx context.Context, fn func() error) error {
|
|
sp.Lock()
|
|
defer sp.Unlock()
|
|
if sp.started {
|
|
return ErrAlreadyStarted
|
|
}
|
|
sp.started = true
|
|
sp.ctx, sp.cancel = context.WithCancel(ctx)
|
|
if err := fn(); err != nil {
|
|
sp.started = false
|
|
sp.cancel()
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
var ErrAlreadyStarted = errors.New("already started")
|
|
var ErrNotStarted = errors.New("not started")
|
|
|
|
// mutex protected stop function
|
|
func (sp *CommonService) Stop(fn func()) {
|
|
sp.Lock()
|
|
defer sp.Unlock()
|
|
if !sp.started {
|
|
return
|
|
}
|
|
sp.cancel()
|
|
fn()
|
|
sp.wg.Wait()
|
|
sp.started = false
|
|
}
|
|
|
|
// This is not a mutex protected function, it is up to the caller to use it in a mutex protected context
|
|
func (sp *CommonService) ErrOnNotRunning() error {
|
|
if !sp.started {
|
|
return ErrNotStarted
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (sp *CommonService) Context() context.Context {
|
|
return sp.ctx
|
|
}
|
|
func (sp *CommonService) WaitGroup() *sync.WaitGroup {
|
|
return &sp.wg
|
|
}
|