status-go/services/wallet/walletevent/transmitter.go

72 lines
1.5 KiB
Go
Raw Normal View History

2022-12-01 09:19:32 +00:00
package walletevent
import (
"sync"
"go.uber.org/zap"
"github.com/ethereum/go-ethereum/event"
2020-01-02 09:10:19 +00:00
gocommon "github.com/status-im/status-go/common"
"github.com/status-im/status-go/logutils"
"github.com/status-im/status-go/signal"
)
2022-12-01 09:19:32 +00:00
type Publisher interface {
Subscribe(interface{}) event.Subscription
}
// SignalsTransmitter transmits received events as wallet signals.
type SignalsTransmitter struct {
2022-12-01 09:19:32 +00:00
Publisher
wg sync.WaitGroup
quit chan struct{}
}
// Start runs loop in background.
func (tmr *SignalsTransmitter) Start() error {
if tmr.quit != nil {
// already running, nothing to do
return nil
}
tmr.quit = make(chan struct{})
2022-12-01 09:19:32 +00:00
events := make(chan Event, 10)
sub := tmr.Publisher.Subscribe(events)
tmr.wg.Add(1)
go func() {
defer gocommon.LogOnPanic()
defer tmr.wg.Done()
for {
select {
case <-tmr.quit:
sub.Unsubscribe()
return
case err := <-sub.Err():
// technically event.Feed cannot send an error to subscription.Err channel.
// the only time we will get an event is when that channel is closed.
if err != nil {
logutils.ZapLogger().Error("wallet signals transmitter failed with", zap.Error(err))
}
return
case event := <-events:
if !event.Type.IsInternal() {
signal.SendWalletEvent(signal.Wallet, event)
}
}
}
}()
return nil
}
// Stop stops the loop and waits till it exits.
func (tmr *SignalsTransmitter) Stop() {
if tmr.quit == nil {
return
}
close(tmr.quit)
tmr.wg.Wait()
tmr.quit = nil
}