Merge pull request #2893 from cfromknecht/no-want-zombie

channeldb/graph: filter zombie channels in FilterKnownChanIDs
This commit is contained in:
Conner Fromknecht 2019-04-05 14:45:11 -07:00 committed by GitHub
commit a52f013161
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 45 additions and 6 deletions

@ -1513,10 +1513,10 @@ func (c *ChannelGraph) NodeUpdatesInHorizon(startTime, endTime time.Time) ([]Lig
} }
// FilterKnownChanIDs takes a set of channel IDs and return the subset of chan // FilterKnownChanIDs takes a set of channel IDs and return the subset of chan
// ID's that we don't know of in the passed set. In other words, we perform a // ID's that we don't know and are not known zombies of the passed set. In other
// set difference of our set of chan ID's and the ones passed in. This method // words, we perform a set difference of our set of chan ID's and the ones
// can be used by callers to determine the set of channels ta peer knows of // passed in. This method can be used by callers to determine the set of
// that we don't. // channels another peer knows of that we don't.
func (c *ChannelGraph) FilterKnownChanIDs(chanIDs []uint64) ([]uint64, error) { func (c *ChannelGraph) FilterKnownChanIDs(chanIDs []uint64) ([]uint64, error) {
var newChanIDs []uint64 var newChanIDs []uint64
@ -1530,15 +1530,31 @@ func (c *ChannelGraph) FilterKnownChanIDs(chanIDs []uint64) ([]uint64, error) {
return ErrGraphNoEdgesFound return ErrGraphNoEdgesFound
} }
// Fetch the zombie index, it may not exist if no edges have
// ever been marked as zombies. If the index has been
// initialized, we will use it later to skip known zombie edges.
zombieIndex := edges.Bucket(zombieBucket)
// We'll run through the set of chanIDs and collate only the // We'll run through the set of chanIDs and collate only the
// set of channel that are unable to be found within our db. // set of channel that are unable to be found within our db.
var cidBytes [8]byte var cidBytes [8]byte
for _, cid := range chanIDs { for _, cid := range chanIDs {
byteOrder.PutUint64(cidBytes[:], cid) byteOrder.PutUint64(cidBytes[:], cid)
if v := edgeIndex.Get(cidBytes[:]); v == nil { // If the edge is already known, skip it.
newChanIDs = append(newChanIDs, cid) if v := edgeIndex.Get(cidBytes[:]); v != nil {
continue
} }
// If the edge is a known zombie, skip it.
if zombieIndex != nil {
isZombie, _, _ := isZombieEdge(zombieIndex, cid)
if isZombie {
continue
}
}
newChanIDs = append(newChanIDs, cid)
} }
return nil return nil

@ -1741,6 +1741,24 @@ func TestFilterKnownChanIDs(t *testing.T) {
chanIDs = append(chanIDs, chanID.ToUint64()) chanIDs = append(chanIDs, chanID.ToUint64())
} }
const numZombies = 5
zombieIDs := make([]uint64, 0, numZombies)
for i := 0; i < numZombies; i++ {
channel, chanID := createEdge(
uint32(i*10+1), 0, 0, 0, node1, node2,
)
if err := graph.AddChannelEdge(&channel); err != nil {
t.Fatalf("unable to create channel edge: %v", err)
}
if err := graph.MarkEdgeZombie(
chanID.ToUint64(), node1.PubKeyBytes, node2.PubKeyBytes,
); err != nil {
t.Fatalf("unable to mark edge zombie: %v", err)
}
zombieIDs = append(zombieIDs, chanID.ToUint64())
}
queryCases := []struct { queryCases := []struct {
queryIDs []uint64 queryIDs []uint64
@ -1751,6 +1769,11 @@ func TestFilterKnownChanIDs(t *testing.T) {
{ {
queryIDs: chanIDs, queryIDs: chanIDs,
}, },
// If we attempt to filter out all zombies that we know of, the
// response should be the empty set.
{
queryIDs: zombieIDs,
},
// If we query for a set of ID's that we didn't insert, we // If we query for a set of ID's that we didn't insert, we
// should get the same set back. // should get the same set back.