Skip to content

Commit 15f1763

Browse files
committed
channeldb: FetchPermAndTempPeers to load access perms on startup
We introduce a new func FetchPermAndTempPeers that returns two maps. The first map indicates the nodes that will have "protected" access to the server. The second map indicates the nodes that have "temporary" access to the server. This will be used in a future commit in the server.go code.
1 parent 5d8309e commit 15f1763

File tree

3 files changed

+303
-0
lines changed

3 files changed

+303
-0
lines changed

channeldb/channel_test.go

+30
Original file line numberDiff line numberDiff line change
@@ -107,6 +107,10 @@ type testChannelParams struct {
107107
// openChannel is set to true if the channel should be fully marked as
108108
// open if this is false, the channel will be left in pending state.
109109
openChannel bool
110+
111+
// closedChannel is set to true if the channel should be marked as
112+
// closed after opening it.
113+
closedChannel bool
110114
}
111115

112116
// testChannelOption is a functional option which can be used to alter the
@@ -129,6 +133,21 @@ func openChannelOption() testChannelOption {
129133
}
130134
}
131135

136+
// closedChannelOption is an option which can be used to create a test channel
137+
// that is closed.
138+
func closedChannelOption() testChannelOption {
139+
return func(params *testChannelParams) {
140+
params.closedChannel = true
141+
}
142+
}
143+
144+
// pubKeyOption is an option which can be used to set the remote's pubkey.
145+
func pubKeyOption(pubKey *btcec.PublicKey) testChannelOption {
146+
return func(params *testChannelParams) {
147+
params.channel.IdentityPub = pubKey
148+
}
149+
}
150+
132151
// localHtlcsOption is an option which allows setting of htlcs on the local
133152
// commitment.
134153
func localHtlcsOption(htlcs []HTLC) testChannelOption {
@@ -231,6 +250,17 @@ func createTestChannel(t *testing.T, cdb *ChannelStateDB,
231250
err = params.channel.MarkAsOpen(params.channel.ShortChannelID)
232251
require.NoError(t, err, "unable to mark channel open")
233252

253+
if params.closedChannel {
254+
// Set the other public keys so that serialization doesn't
255+
// panic.
256+
err = params.channel.CloseChannel(&ChannelCloseSummary{
257+
RemotePub: params.channel.IdentityPub,
258+
RemoteCurrentRevocation: params.channel.IdentityPub,
259+
RemoteNextRevocation: params.channel.IdentityPub,
260+
})
261+
require.NoError(t, err, "unable to close channel")
262+
}
263+
234264
return params.channel
235265
}
236266

channeldb/db.go

+188
Original file line numberDiff line numberDiff line change
@@ -730,6 +730,194 @@ func (c *ChannelStateDB) FetchChannelByID(tx kvdb.RTx, id lnwire.ChannelID) (
730730
return c.channelScanner(tx, selector)
731731
}
732732

733+
// ChanCount is used by the server in determining access control.
734+
type ChanCount struct {
735+
HasOpenOrClosedChan bool
736+
PendingOpenCount uint64
737+
}
738+
739+
// FetchPermAndTempPeers returns a map where the key is the remote node's
740+
// public key and the value is a struct that has a tally of the pending-open
741+
// channels and whether the peer has an open or closed channel with us.
742+
func (c *ChannelStateDB) FetchPermAndTempPeers(
743+
chainHash []byte) (map[string]ChanCount, error) {
744+
745+
peerCounts := make(map[string]ChanCount)
746+
747+
err := kvdb.View(c.backend, func(tx kvdb.RTx) error {
748+
openChanBucket := tx.ReadBucket(openChannelBucket)
749+
if openChanBucket == nil {
750+
return ErrNoChanDBExists
751+
}
752+
753+
openChanErr := openChanBucket.ForEach(func(nodePub,
754+
v []byte) error {
755+
756+
// If there is a value, this is not a bucket.
757+
if v != nil {
758+
return nil
759+
}
760+
761+
nodeChanBucket := openChanBucket.NestedReadBucket(
762+
nodePub,
763+
)
764+
if nodeChanBucket == nil {
765+
return nil
766+
}
767+
768+
chainBucket := nodeChanBucket.NestedReadBucket(
769+
chainHash,
770+
)
771+
if chainBucket == nil {
772+
return fmt.Errorf("no chain bucket exists")
773+
}
774+
775+
var isPermPeer bool
776+
var pendingOpenCount uint64
777+
778+
internalErr := chainBucket.ForEach(func(chanPoint,
779+
val []byte) error {
780+
781+
// If there is a value, this is not a bucket.
782+
if val != nil {
783+
return nil
784+
}
785+
786+
chanBucket := chainBucket.NestedReadBucket(
787+
chanPoint,
788+
)
789+
if chanBucket == nil {
790+
return nil
791+
}
792+
793+
var op wire.OutPoint
794+
readErr := graphdb.ReadOutpoint(
795+
bytes.NewReader(chanPoint), &op,
796+
)
797+
if readErr != nil {
798+
return readErr
799+
}
800+
801+
// We need to go through each channel and look
802+
// at the IsPending status.
803+
openChan, err := fetchOpenChannel(
804+
chanBucket, &op,
805+
)
806+
if err != nil {
807+
return err
808+
}
809+
810+
if openChan.IsPending {
811+
// Add to the pending-open count since
812+
// this is a temp peer.
813+
pendingOpenCount++
814+
return nil
815+
}
816+
817+
// Since IsPending is false, this is a perm
818+
// peer.
819+
isPermPeer = true
820+
821+
return nil
822+
})
823+
if internalErr != nil {
824+
return internalErr
825+
}
826+
827+
peerCount := ChanCount{
828+
HasOpenOrClosedChan: isPermPeer,
829+
PendingOpenCount: pendingOpenCount,
830+
}
831+
peerCounts[string(nodePub)] = peerCount
832+
833+
return nil
834+
})
835+
if openChanErr != nil {
836+
return openChanErr
837+
}
838+
839+
// Now check the closed channel bucket.
840+
historicalChanBucket := tx.ReadBucket(historicalChannelBucket)
841+
if historicalChanBucket == nil {
842+
return ErrNoHistoricalBucket
843+
}
844+
845+
historicalErr := historicalChanBucket.ForEach(func(chanPoint,
846+
v []byte) error {
847+
// Parse each nested bucket and the chanInfoKey to get
848+
// the IsPending bool. This determines whether the
849+
// peer is protected or not.
850+
if v != nil {
851+
// This is not a bucket. This is currently not
852+
// possible.
853+
return nil
854+
}
855+
856+
chanBucket := historicalChanBucket.NestedReadBucket(
857+
chanPoint,
858+
)
859+
if chanBucket == nil {
860+
// This is not possible.
861+
return fmt.Errorf("no historical channel " +
862+
"bucket exists")
863+
}
864+
865+
var op wire.OutPoint
866+
readErr := graphdb.ReadOutpoint(
867+
bytes.NewReader(chanPoint), &op,
868+
)
869+
if readErr != nil {
870+
return readErr
871+
}
872+
873+
// This channel is closed, but the structure of the
874+
// historical bucket is the same. This is by design,
875+
// which means we can call fetchOpenChannel.
876+
channel, fetchErr := fetchOpenChannel(chanBucket, &op)
877+
if fetchErr != nil {
878+
return fetchErr
879+
}
880+
881+
// Only include this peer in the protected class if
882+
// the closing transaction confirmed. Note that
883+
// CloseChannel can be called in the funding manager
884+
// while IsPending is true which is why we need this
885+
// special-casing to not count premature funding
886+
// manager calls to CloseChannel.
887+
if !channel.IsPending {
888+
// Fetch the public key of the remote node. We
889+
// need to use the string-ified serialized,
890+
// compressed bytes as the key.
891+
remotePub := channel.IdentityPub
892+
remoteSer := remotePub.SerializeCompressed()
893+
remoteKey := string(remoteSer)
894+
895+
count, exists := peerCounts[remoteKey]
896+
if exists {
897+
count.HasOpenOrClosedChan = true
898+
peerCounts[remoteKey] = count
899+
} else {
900+
peerCount := ChanCount{
901+
HasOpenOrClosedChan: true,
902+
}
903+
peerCounts[remoteKey] = peerCount
904+
}
905+
}
906+
907+
return nil
908+
})
909+
if historicalErr != nil {
910+
return historicalErr
911+
}
912+
913+
return nil
914+
}, func() {
915+
clear(peerCounts)
916+
})
917+
918+
return peerCounts, err
919+
}
920+
733921
// channelSelector describes a function that takes a chain-hash bucket from
734922
// within the open-channel DB and returns the wanted channel point bytes, and
735923
// channel point. It must return the ErrChannelNotFound error if the wanted

channeldb/db_test.go

+85
Original file line numberDiff line numberDiff line change
@@ -721,6 +721,91 @@ func TestFetchHistoricalChannel(t *testing.T) {
721721
}
722722
}
723723

724+
// TestFetchPermTempPeer tests that we're able to call FetchPermAndTempPeers
725+
// successfully.
726+
func TestFetchPermTempPeer(t *testing.T) {
727+
t.Parallel()
728+
729+
fullDB, err := MakeTestDB(t)
730+
require.NoError(t, err, "unable to make test database")
731+
732+
cdb := fullDB.ChannelStateDB()
733+
734+
// Create an open channel.
735+
privKey1, err := btcec.NewPrivateKey()
736+
require.NoError(t, err, "unable to generate new private key")
737+
738+
pubKey1 := privKey1.PubKey()
739+
740+
channelState1 := createTestChannel(
741+
t, cdb, openChannelOption(), pubKeyOption(pubKey1),
742+
)
743+
744+
// Next, assert that the channel exists in the database.
745+
_, err = cdb.FetchChannel(channelState1.FundingOutpoint)
746+
require.NoError(t, err, "unable to fetch channel")
747+
748+
// Create a pending channel.
749+
privKey2, err := btcec.NewPrivateKey()
750+
require.NoError(t, err, "unable to generate private key")
751+
752+
pubKey2 := privKey2.PubKey()
753+
channelState2 := createTestChannel(t, cdb, pubKeyOption(pubKey2))
754+
755+
// Assert that the channel exists in the database.
756+
_, err = cdb.FetchChannel(channelState2.FundingOutpoint)
757+
require.NoError(t, err, "unable to fetch channel")
758+
759+
// Create a closed channel.
760+
privKey3, err := btcec.NewPrivateKey()
761+
require.NoError(t, err, "unable to generate new private key")
762+
763+
pubKey3 := privKey3.PubKey()
764+
765+
_ = createTestChannel(
766+
t, cdb, pubKeyOption(pubKey3), openChannelOption(),
767+
closedChannelOption(),
768+
)
769+
770+
// Fetch the ChanCount for our peers.
771+
peerCounts, err := cdb.FetchPermAndTempPeers(key[:])
772+
require.NoError(t, err, "unable to fetch perm and temp peers")
773+
774+
// There should only be three entries.
775+
require.Len(t, peerCounts, 3)
776+
777+
// The first entry should have OpenClosed set to true and Pending set
778+
// to 0.
779+
count1, found := peerCounts[string(pubKey1.SerializeCompressed())]
780+
require.True(t, found, "unable to find peer 1 in peerCounts")
781+
require.True(
782+
t, count1.HasOpenOrClosedChan,
783+
"couldn't find peer 1's channels",
784+
)
785+
require.Zero(
786+
t, count1.PendingOpenCount,
787+
"peer 1 doesn't have 0 pending-open",
788+
)
789+
790+
count2, found := peerCounts[string(pubKey2.SerializeCompressed())]
791+
require.True(t, found, "unable to find peer 2 in peerCounts")
792+
require.False(
793+
t, count2.HasOpenOrClosedChan, "found erroneous channels",
794+
)
795+
require.Equal(t, uint64(1), count2.PendingOpenCount)
796+
797+
count3, found := peerCounts[string(pubKey3.SerializeCompressed())]
798+
require.True(t, found, "unable to find peer 3 in peerCounts")
799+
require.True(
800+
t, count3.HasOpenOrClosedChan,
801+
"couldn't find peer 3's channels",
802+
)
803+
require.Zero(
804+
t, count3.PendingOpenCount,
805+
"peer 3 doesn't have 0 pending-open",
806+
)
807+
}
808+
724809
func createLightningNode(priv *btcec.PrivateKey) *models.LightningNode {
725810
updateTime := rand.Int63()
726811

0 commit comments

Comments
 (0)