lnd/watchtower/wtclient/addr_iterator.go

419 lines
12 KiB
Go

package wtclient
import (
"container/list"
"errors"
"fmt"
"net"
"sync"
"github.com/lightningnetwork/lnd/watchtower/wtdb"
)
var (
// ErrAddressesExhausted signals that a addressIterator has cycled
// through all available addresses.
ErrAddressesExhausted = errors.New("exhausted all addresses")
// ErrAddrInUse indicates that an address is locked and cannot be
// removed from the addressIterator.
ErrAddrInUse = errors.New("address in use")
)
// AddressIterator handles iteration over a list of addresses. It strictly
// disallows the list of addresses it holds to be empty. It also allows callers
// to place locks on certain addresses in order to prevent other callers from
// removing the addresses in question from the iterator.
type AddressIterator interface {
// Next returns the next candidate address. This iterator will always
// return candidates in the order given when the iterator was
// instantiated. If no more candidates are available,
// ErrAddressesExhausted is returned.
Next() (net.Addr, error)
// NextAndLock does the same as described for Next, and it also places a
// lock on the returned address so that the address can not be removed
// until the lock on it has been released via ReleaseLock.
NextAndLock() (net.Addr, error)
// Peek returns the currently selected address in the iterator. If the
// end of the iterator has been reached then it is reset and the first
// item in the iterator is returned. Since the AddressIterator will
// never have an empty address list, this function will never return a
// nil value.
Peek() net.Addr
// PeekAndLock does the same as described for Peek, and it also places
// a lock on the returned address so that the address can not be removed
// until the lock on it has been released via ReleaseLock.
PeekAndLock() net.Addr
// ReleaseLock releases the lock held on the given address.
ReleaseLock(addr net.Addr)
// Add adds a new address to the iterator.
Add(addr net.Addr)
// Remove removes an existing address from the iterator. It disallows
// the address from being removed if it is the last address in the
// iterator or if there is currently a lock on the address.
Remove(addr net.Addr) error
// HasLocked returns true if the addressIterator has any locked
// addresses.
HasLocked() bool
// GetAll returns a copy of all the addresses in the iterator.
GetAll() []net.Addr
// Reset clears the iterators state, and makes the address at the front
// of the list the next item to be returned.
Reset()
// Copy constructs a new AddressIterator that has the same addresses
// as this iterator.
//
// NOTE that the address locks are not expected to be copied.
Copy() AddressIterator
}
// A compile-time check to ensure that addressIterator implements the
// AddressIterator interface.
var _ AddressIterator = (*addressIterator)(nil)
// addressIterator is a linked-list implementation of an AddressIterator.
type addressIterator struct {
mu sync.Mutex
addrList *list.List
currentTopAddr *list.Element
candidates map[string]*candidateAddr
totalLockCount int
}
type candidateAddr struct {
addr net.Addr
numLocks int
}
// newAddressIterator constructs a new addressIterator.
func newAddressIterator(addrs ...net.Addr) (*addressIterator, error) {
if len(addrs) == 0 {
return nil, fmt.Errorf("must have at least one address")
}
iter := &addressIterator{
addrList: list.New(),
candidates: make(map[string]*candidateAddr),
}
for _, addr := range addrs {
addrID := addr.String()
iter.addrList.PushBack(addrID)
iter.candidates[addrID] = &candidateAddr{addr: addr}
}
iter.Reset()
return iter, nil
}
// Reset clears the iterators state, and makes the address at the front of the
// list the next item to be returned.
//
// NOTE: This is part of the AddressIterator interface.
func (a *addressIterator) Reset() {
a.mu.Lock()
defer a.mu.Unlock()
a.unsafeReset()
}
// unsafeReset clears the iterator state and makes the address at the front of
// the list the next item to be returned.
//
// NOTE: this method is not thread safe and so should only be called if the
// appropriate mutex is being held.
func (a *addressIterator) unsafeReset() {
// Reset the next candidate to the front of the linked-list.
a.currentTopAddr = a.addrList.Front()
}
// Next returns the next candidate address. This iterator will always return
// candidates in the order given when the iterator was instantiated. If no more
// candidates are available, ErrAddressesExhausted is returned.
//
// NOTE: This is part of the AddressIterator interface.
func (a *addressIterator) Next() (net.Addr, error) {
return a.next(false)
}
// NextAndLock does the same as described for Next, and it also places a lock on
// the returned address so that the address can not be removed until the lock on
// it has been released via ReleaseLock.
//
// NOTE: This is part of the AddressIterator interface.
func (a *addressIterator) NextAndLock() (net.Addr, error) {
return a.next(true)
}
// next returns the next candidate address. This iterator will always return
// candidates in the order given when the iterator was instantiated. If no more
// candidates are available, ErrAddressesExhausted is returned.
func (a *addressIterator) next(lock bool) (net.Addr, error) {
a.mu.Lock()
defer a.mu.Unlock()
// In-case currentTopAddr is nil (meaning that Reset has not yet been
// called), return an error indicating this.
if a.currentTopAddr == nil {
return nil, ErrAddressesExhausted
}
// Set the next candidate to the subsequent element. If we are at the
// end of the address list, this could mean setting currentTopAddr to
// nil.
a.currentTopAddr = a.currentTopAddr.Next()
for a.currentTopAddr != nil {
// Propose the address at the front of the list.
addrID := a.currentTopAddr.Value.(string)
// Check whether this address is still considered a candidate.
// If it's not, we'll proceed to the next.
candidate, ok := a.candidates[addrID]
// If the address cannot be found in the candidate set, then
// this must mean that the Remove method was called for the
// address. The Remove method would have checked that the
// address is not the last one in the iterator and that it has
// no locks on it. It is therefor safe to remove.
if !ok {
// Grab the next address candidate. This might be nil
// if the iterator is on the last item in the list.
nextCandidate := a.currentTopAddr.Next()
// Remove the address from the list that is no longer
// in the candidate set.
a.addrList.Remove(a.currentTopAddr)
// Set the current top to the next candidate. This might
// mean setting it to nil if the iterator is on its last
// item in which case the loop will be exited and an
// ErrAddressesExhausted exhausted error will be
// returned.
a.currentTopAddr = nextCandidate
continue
}
if lock {
candidate.numLocks++
a.totalLockCount++
}
return candidate.addr, nil
}
return nil, ErrAddressesExhausted
}
// Peek returns the currently selected address in the iterator. If the end of
// the list has been reached then the iterator is reset and the first item in
// the list is returned. Since the addressIterator will never have an empty
// address list, this function will never return a nil value.
//
// NOTE: This is part of the AddressIterator interface.
func (a *addressIterator) Peek() net.Addr {
return a.peek(false)
}
// PeekAndLock does the same as described for Peek, and it also places a lock on
// the returned address so that the address can not be removed until the lock
// on it has been released via ReleaseLock.
//
// NOTE: This is part of the AddressIterator interface.
func (a *addressIterator) PeekAndLock() net.Addr {
return a.peek(true)
}
// peek returns the currently selected address in the iterator. If the end of
// the list has been reached then the iterator is reset and the first item in
// the list is returned. Since the addressIterator will never have an empty
// address list, this function will never return a nil value. If lock is set to
// true, the address will be locked for removal until ReleaseLock has been
// called for the address.
func (a *addressIterator) peek(lock bool) net.Addr {
a.mu.Lock()
defer a.mu.Unlock()
for {
// If currentTopAddr is nil, it means we have reached the end of
// the list, so we reset it here. The iterator always has at
// least one address, so we can be sure that currentTopAddr will
// be non-nil after calling reset here.
if a.currentTopAddr == nil {
a.unsafeReset()
}
addrID := a.currentTopAddr.Value.(string)
candidate, ok := a.candidates[addrID]
// If the address cannot be found in the candidate set, then
// this must mean that the Remove method was called for the
// address. The Remove method would have checked that the
// address is not the last one in the iterator and that it has
// no locks on it. It is therefor safe to remove.
if !ok {
// Grab the next address candidate. This might be nil
// if the iterator is on the last item in the list.
nextCandidate := a.currentTopAddr.Next()
// Remove the address from the list that is no longer
// in the candidate set.
a.addrList.Remove(a.currentTopAddr)
// Set the current top to the next candidate. This might
// mean setting it to nil if the iterator is on its last
// item but this will be reset at the top of the for
// loop.
a.currentTopAddr = nextCandidate
continue
}
if lock {
candidate.numLocks++
a.totalLockCount++
}
return candidate.addr
}
}
// ReleaseLock releases the lock held on the given address.
//
// NOTE: This is part of the AddressIterator interface.
func (a *addressIterator) ReleaseLock(addr net.Addr) {
a.mu.Lock()
defer a.mu.Unlock()
candidateAddr, ok := a.candidates[addr.String()]
if !ok {
return
}
if candidateAddr.numLocks == 0 {
return
}
candidateAddr.numLocks--
a.totalLockCount--
}
// Add adds a new address to the iterator.
//
// NOTE: This is part of the AddressIterator interface.
func (a *addressIterator) Add(addr net.Addr) {
a.mu.Lock()
defer a.mu.Unlock()
if _, ok := a.candidates[addr.String()]; ok {
return
}
a.addrList.PushBack(addr.String())
a.candidates[addr.String()] = &candidateAddr{addr: addr}
// If we've reached the end of our queue, then this candidate
// will become the next.
if a.currentTopAddr == nil {
a.currentTopAddr = a.addrList.Back()
}
}
// Remove removes an existing address from the iterator. It disallows the
// address from being removed if it is the last address in the iterator or if
// there is currently a lock on the address.
//
// NOTE: This is part of the AddressIterator interface.
func (a *addressIterator) Remove(addr net.Addr) error {
a.mu.Lock()
defer a.mu.Unlock()
candidate, ok := a.candidates[addr.String()]
if !ok {
return nil
}
if len(a.candidates) == 1 {
return wtdb.ErrLastTowerAddr
}
if candidate.numLocks > 0 {
return ErrAddrInUse
}
delete(a.candidates, addr.String())
return nil
}
// HasLocked returns true if the addressIterator has any locked addresses.
//
// NOTE: This is part of the AddressIterator interface.
func (a *addressIterator) HasLocked() bool {
a.mu.Lock()
defer a.mu.Unlock()
return a.totalLockCount > 0
}
// GetAll returns a copy of all the addresses in the iterator.
//
// NOTE: This is part of the AddressIterator interface.
func (a *addressIterator) GetAll() []net.Addr {
a.mu.Lock()
defer a.mu.Unlock()
return a.getAllUnsafe()
}
// Copy constructs a new AddressIterator that has the same addresses
// as this iterator.
//
// NOTE that the address locks will not be copied.
func (a *addressIterator) Copy() AddressIterator {
a.mu.Lock()
defer a.mu.Unlock()
addrs := a.getAllUnsafe()
// Since newAddressIterator will only ever return an error if it is
// initialised with zero addresses, we can ignore the error here since
// we are initialising it with the set of addresses of this
// addressIterator which is by definition a non-empty list.
iter, _ := newAddressIterator(addrs...)
return iter
}
// getAllUnsafe returns a copy of all the addresses in the iterator.
//
// NOTE: this method is not thread safe and so must only be called once the
// addressIterator mutex is already being held.
func (a *addressIterator) getAllUnsafe() []net.Addr {
var addrs []net.Addr
cursor := a.addrList.Front()
for cursor != nil {
addrID := cursor.Value.(string)
addr, ok := a.candidates[addrID]
if !ok {
cursor = cursor.Next()
continue
}
addrs = append(addrs, addr.addr)
cursor = cursor.Next()
}
return addrs
}