aboutsummaryrefslogblamecommitdiffstats
path: root/dex/nodetable.go
blob: cc1de160fe3048ca7988a6460672813cdfb51025 (plain) (tree)
1
2
3
4
5
6
7
8
9


           

              
                                                 
                                               
                                                     
                                                   

 
                                                    


                          
                                      




                                
                                                      


         
                                                      




                            
                                                       


                           








                                                                             
                                                                                                
                                                                                               

                                




                                                                                       
         


                                                           

 
                                             

                            


                                                        
         
                      

 

                                                       

                                   
package dex

import (
    "sync"

    "github.com/dexon-foundation/dexon/event"
    "github.com/dexon-foundation/dexon/log"
    "github.com/dexon-foundation/dexon/p2p/enode"
    "github.com/dexon-foundation/dexon/p2p/enr"
)

type newRecordsEvent struct{ Records []*enr.Record }

type nodeTable struct {
    mu    sync.RWMutex
    entry map[enode.ID]*enode.Node
    feed  event.Feed
}

func newNodeTable() *nodeTable {
    return &nodeTable{
        entry: make(map[enode.ID]*enode.Node),
    }
}

func (t *nodeTable) GetNode(id enode.ID) *enode.Node {
    t.mu.RLock()
    defer t.mu.RUnlock()
    return t.entry[id]
}

func (t *nodeTable) AddRecords(records []*enr.Record) {
    t.mu.Lock()
    defer t.mu.Unlock()

    var newRecords []*enr.Record
    for _, record := range records {
        node, err := enode.New(enode.ValidSchemes, record)
        if err != nil {
            log.Error("invalid node record", "err", err)
            return
        }

        if n, ok := t.entry[node.ID()]; ok && n.Seq() >= node.Seq() {
            log.Trace("Ignore new record, already exists", "id", node.ID().String(),
                "ip", node.IP().String(), "udp", node.UDP(), "tcp", node.TCP())
            continue
        }

        t.entry[node.ID()] = node
        newRecords = append(newRecords, record)
        log.Debug("Add new record to node table", "id", node.ID().String(),
            "ip", node.IP().String(), "udp", node.UDP(), "tcp", node.TCP())
    }
    if len(newRecords) > 0 {
        go t.feed.Send(newRecordsEvent{newRecords})
    }
}

func (t *nodeTable) Records() []*enr.Record {
    t.mu.RLock()
    defer t.mu.RUnlock()
    records := make([]*enr.Record, 0, len(t.entry))
    for _, node := range t.entry {
        records = append(records, node.Record())
    }
    return records
}

func (t *nodeTable) SubscribeNewRecordsEvent(
    ch chan<- newRecordsEvent) event.Subscription {
    return t.feed.Subscribe(ch)
}