mirror of
https://github.com/kaspanet/kaspad.git
synced 2025-11-24 14:35:53 +00:00
[NOD-1404] Remove most of the notification manager to fix a deadlock (#935)
* [NOD-1404] Remove most of the notification manager to fix a deadlock. * [NOD-1404] Rename a couple of fields.
This commit is contained in:
parent
8d38a28b82
commit
3d8c131fa8
@ -40,16 +40,12 @@ func (m *Manager) routerInitializer(router *router.Router, netConnection *netada
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
spawn("routerInitializer-handleIncomingMessages", func() {
|
m.context.NotificationManager.AddListener(router)
|
||||||
err := m.handleIncomingMessages(router, incomingRoute)
|
|
||||||
m.handleError(err, netConnection)
|
|
||||||
})
|
|
||||||
|
|
||||||
notificationListener := m.context.NotificationManager.AddListener(router)
|
spawn("routerInitializer-handleIncomingMessages", func() {
|
||||||
spawn("routerInitializer-handleOutgoingNotifications", func() {
|
|
||||||
defer m.context.NotificationManager.RemoveListener(router)
|
defer m.context.NotificationManager.RemoveListener(router)
|
||||||
|
|
||||||
err := m.handleOutgoingNotifications(notificationListener)
|
err := m.handleIncomingMessages(router, incomingRoute)
|
||||||
m.handleError(err, netConnection)
|
m.handleError(err, netConnection)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@ -76,15 +72,6 @@ func (m *Manager) handleIncomingMessages(router *router.Router, incomingRoute *r
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *Manager) handleOutgoingNotifications(notificationListener *rpccontext.NotificationListener) error {
|
|
||||||
for {
|
|
||||||
err := notificationListener.ProcessNextNotification()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *Manager) handleError(err error, netConnection *netadapter.NetConnection) {
|
func (m *Manager) handleError(err error, netConnection *netadapter.NetConnection) {
|
||||||
if errors.Is(err, router.ErrTimeout) {
|
if errors.Is(err, router.ErrTimeout) {
|
||||||
log.Warnf("Got timeout from %s. Disconnecting...", netConnection)
|
log.Warnf("Got timeout from %s. Disconnecting...", netConnection)
|
||||||
|
|||||||
@ -2,7 +2,7 @@ package rpccontext
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/kaspanet/kaspad/app/appmessage"
|
"github.com/kaspanet/kaspad/app/appmessage"
|
||||||
"github.com/kaspanet/kaspad/infrastructure/network/netadapter/router"
|
routerpkg "github.com/kaspanet/kaspad/infrastructure/network/netadapter/router"
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
"sync"
|
"sync"
|
||||||
)
|
)
|
||||||
@ -10,58 +10,41 @@ import (
|
|||||||
// NotificationManager manages notifications for the RPC
|
// NotificationManager manages notifications for the RPC
|
||||||
type NotificationManager struct {
|
type NotificationManager struct {
|
||||||
sync.RWMutex
|
sync.RWMutex
|
||||||
listeners map[*router.Router]*NotificationListener
|
listeners map[*routerpkg.Router]*NotificationListener
|
||||||
}
|
}
|
||||||
|
|
||||||
// OnBlockAddedListener is a listener function for when a block is added to the DAG
|
|
||||||
type OnBlockAddedListener func(notification *appmessage.BlockAddedNotificationMessage) error
|
|
||||||
|
|
||||||
// OnChainChangedListener is a listener function for when the DAG's selected parent chain changes
|
|
||||||
type OnChainChangedListener func(notification *appmessage.ChainChangedNotificationMessage) error
|
|
||||||
|
|
||||||
// NotificationListener represents a registered RPC notification listener
|
// NotificationListener represents a registered RPC notification listener
|
||||||
type NotificationListener struct {
|
type NotificationListener struct {
|
||||||
onBlockAddedListener OnBlockAddedListener
|
propagateBlockAddedNotifications bool
|
||||||
onBlockAddedNotificationChan chan *appmessage.BlockAddedNotificationMessage
|
propagateChainChangedNotifications bool
|
||||||
onChainChangedListener OnChainChangedListener
|
|
||||||
onChainChangedNotificationChan chan *appmessage.ChainChangedNotificationMessage
|
|
||||||
|
|
||||||
closeChan chan struct{}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewNotificationManager creates a new NotificationManager
|
// NewNotificationManager creates a new NotificationManager
|
||||||
func NewNotificationManager() *NotificationManager {
|
func NewNotificationManager() *NotificationManager {
|
||||||
return &NotificationManager{
|
return &NotificationManager{
|
||||||
listeners: make(map[*router.Router]*NotificationListener),
|
listeners: make(map[*routerpkg.Router]*NotificationListener),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// AddListener registers a listener with the given router
|
// AddListener registers a listener with the given router
|
||||||
func (nm *NotificationManager) AddListener(router *router.Router) *NotificationListener {
|
func (nm *NotificationManager) AddListener(router *routerpkg.Router) {
|
||||||
nm.Lock()
|
nm.Lock()
|
||||||
defer nm.Unlock()
|
defer nm.Unlock()
|
||||||
|
|
||||||
listener := newNotificationListener()
|
listener := newNotificationListener()
|
||||||
nm.listeners[router] = listener
|
nm.listeners[router] = listener
|
||||||
return listener
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// RemoveListener unregisters the given router
|
// RemoveListener unregisters the given router
|
||||||
func (nm *NotificationManager) RemoveListener(router *router.Router) {
|
func (nm *NotificationManager) RemoveListener(router *routerpkg.Router) {
|
||||||
nm.Lock()
|
nm.Lock()
|
||||||
defer nm.Unlock()
|
defer nm.Unlock()
|
||||||
|
|
||||||
listener, ok := nm.listeners[router]
|
|
||||||
if !ok {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
listener.close()
|
|
||||||
|
|
||||||
delete(nm.listeners, router)
|
delete(nm.listeners, router)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Listener retrieves the listener registered with the given router
|
// Listener retrieves the listener registered with the given router
|
||||||
func (nm *NotificationManager) Listener(router *router.Router) (*NotificationListener, error) {
|
func (nm *NotificationManager) Listener(router *routerpkg.Router) (*NotificationListener, error) {
|
||||||
nm.RLock()
|
nm.RLock()
|
||||||
defer nm.RUnlock()
|
defer nm.RUnlock()
|
||||||
|
|
||||||
@ -73,67 +56,52 @@ func (nm *NotificationManager) Listener(router *router.Router) (*NotificationLis
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NotifyBlockAdded notifies the notification manager that a block has been added to the DAG
|
// NotifyBlockAdded notifies the notification manager that a block has been added to the DAG
|
||||||
func (nm *NotificationManager) NotifyBlockAdded(notification *appmessage.BlockAddedNotificationMessage) {
|
func (nm *NotificationManager) NotifyBlockAdded(notification *appmessage.BlockAddedNotificationMessage) error {
|
||||||
nm.RLock()
|
nm.RLock()
|
||||||
defer nm.RUnlock()
|
defer nm.RUnlock()
|
||||||
|
|
||||||
for _, listener := range nm.listeners {
|
for router, listener := range nm.listeners {
|
||||||
if listener.onBlockAddedListener != nil {
|
if listener.propagateBlockAddedNotifications {
|
||||||
select {
|
err := router.OutgoingRoute().Enqueue(notification)
|
||||||
case listener.onBlockAddedNotificationChan <- notification:
|
if err != nil {
|
||||||
case <-listener.closeChan:
|
return err
|
||||||
continue
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// NotifyChainChanged notifies the notification manager that the DAG's selected parent chain has changed
|
// NotifyChainChanged notifies the notification manager that the DAG's selected parent chain has changed
|
||||||
func (nm *NotificationManager) NotifyChainChanged(message *appmessage.ChainChangedNotificationMessage) {
|
func (nm *NotificationManager) NotifyChainChanged(notification *appmessage.ChainChangedNotificationMessage) error {
|
||||||
nm.RLock()
|
nm.RLock()
|
||||||
defer nm.RUnlock()
|
defer nm.RUnlock()
|
||||||
|
|
||||||
for _, listener := range nm.listeners {
|
for router, listener := range nm.listeners {
|
||||||
if listener.onChainChangedListener != nil {
|
if listener.propagateChainChangedNotifications {
|
||||||
select {
|
err := router.OutgoingRoute().Enqueue(notification)
|
||||||
case listener.onChainChangedNotificationChan <- message:
|
if err != nil {
|
||||||
case <-listener.closeChan:
|
return err
|
||||||
continue
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func newNotificationListener() *NotificationListener {
|
func newNotificationListener() *NotificationListener {
|
||||||
return &NotificationListener{
|
return &NotificationListener{
|
||||||
onBlockAddedNotificationChan: make(chan *appmessage.BlockAddedNotificationMessage),
|
propagateBlockAddedNotifications: false,
|
||||||
onChainChangedNotificationChan: make(chan *appmessage.ChainChangedNotificationMessage),
|
propagateChainChangedNotifications: false,
|
||||||
closeChan: make(chan struct{}, 1),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetOnBlockAddedListener sets the onBlockAddedListener handler for this listener
|
// PropagateBlockAddedNotifications instructs the listener to send block added notifications
|
||||||
func (nl *NotificationListener) SetOnBlockAddedListener(onBlockAddedListener OnBlockAddedListener) {
|
// to the remote listener
|
||||||
nl.onBlockAddedListener = onBlockAddedListener
|
func (nl *NotificationListener) PropagateBlockAddedNotifications() {
|
||||||
|
nl.propagateBlockAddedNotifications = true
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetOnChainChangedListener sets the onChainChangedListener handler for this listener
|
// PropagateChainChangedNotifications instructs the listener to send chain changed notifications
|
||||||
func (nl *NotificationListener) SetOnChainChangedListener(onChainChangedListener OnChainChangedListener) {
|
// to the remote listener
|
||||||
nl.onChainChangedListener = onChainChangedListener
|
func (nl *NotificationListener) PropagateChainChangedNotifications() {
|
||||||
}
|
nl.propagateChainChangedNotifications = true
|
||||||
|
|
||||||
// ProcessNextNotification waits until a notification arrives and processes it
|
|
||||||
func (nl *NotificationListener) ProcessNextNotification() error {
|
|
||||||
select {
|
|
||||||
case block := <-nl.onBlockAddedNotificationChan:
|
|
||||||
return nl.onBlockAddedListener(block)
|
|
||||||
case notification := <-nl.onChainChangedNotificationChan:
|
|
||||||
return nl.onChainChangedListener(notification)
|
|
||||||
case <-nl.closeChan:
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (nl *NotificationListener) close() {
|
|
||||||
nl.closeChan <- struct{}{}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -12,9 +12,7 @@ func HandleNotifyBlockAdded(context *rpccontext.Context, router *router.Router,
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
listener.SetOnBlockAddedListener(func(notification *appmessage.BlockAddedNotificationMessage) error {
|
listener.PropagateBlockAddedNotifications()
|
||||||
return router.OutgoingRoute().Enqueue(notification)
|
|
||||||
})
|
|
||||||
|
|
||||||
response := appmessage.NewNotifyBlockAddedResponseMessage()
|
response := appmessage.NewNotifyBlockAddedResponseMessage()
|
||||||
return response, nil
|
return response, nil
|
||||||
|
|||||||
@ -18,9 +18,7 @@ func HandleNotifyChainChanged(context *rpccontext.Context, router *router.Router
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
listener.SetOnChainChangedListener(func(message *appmessage.ChainChangedNotificationMessage) error {
|
listener.PropagateChainChangedNotifications()
|
||||||
return router.OutgoingRoute().Enqueue(message)
|
|
||||||
})
|
|
||||||
|
|
||||||
response := appmessage.NewNotifyChainChangedResponseMessage()
|
response := appmessage.NewNotifyChainChangedResponseMessage()
|
||||||
return response, nil
|
return response, nil
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user