channeldb: fix bug when writing revocation log states

This commit fixes a bug that was introduced when we moved to using
64-bit integers for storing the revocation log state. When we made this
change, we forgot to increase the size of the buffer which stores the
key for the particular channel state from 40 to 44 bytes to account for
the 4 additional bytes in the new 64-bit integer.

This bug has been fixed by properly sizing the key buffer. We’ve also
added an additional test to ensure that we retrieve the proper state
after multiple state updates.
This commit is contained in:
Olaoluwa Osuntokun 2017-02-20 21:58:34 -08:00
parent f2e077c10c
commit bcde70e74a
No known key found for this signature in database
GPG Key ID: 9CC5B105D03521A2
2 changed files with 39 additions and 6 deletions

@ -1817,18 +1817,20 @@ func deserializeChannelDelta(r io.Reader) (*ChannelDelta, error) {
return delta, nil return delta, nil
} }
func makeLogKey(o *wire.OutPoint, updateNum uint64) [40]byte { func makeLogKey(o *wire.OutPoint, updateNum uint64) [44]byte {
var ( var (
scratch [8]byte scratch [8]byte
n int n int
k [40]byte
// txid (32) || index (4) || update_num (8)
// 32 + 4 + 8 = 44
k [44]byte
) )
n += copy(k[:], o.Hash[:]) n += copy(k[:], o.Hash[:])
byteOrder.PutUint32(scratch[:4], o.Index) byteOrder.PutUint32(scratch[:4], o.Index)
copy(k[n:], scratch[:4]) n += copy(k[n:], scratch[:4])
n += 4
byteOrder.PutUint64(scratch[:], updateNum) byteOrder.PutUint64(scratch[:], updateNum)
copy(k[n:], scratch[:]) copy(k[n:], scratch[:])

@ -389,7 +389,10 @@ func TestChannelStateTransition(t *testing.T) {
// Add some HTLCs which were added during this new state transition. // Add some HTLCs which were added during this new state transition.
// Half of the HTLCs are incoming, while the other half are outgoing. // Half of the HTLCs are incoming, while the other half are outgoing.
var htlcs []*HTLC var (
htlcs []*HTLC
htlcAmt btcutil.Amount
)
for i := uint32(0); i < 10; i++ { for i := uint32(0); i < 10; i++ {
var incoming bool var incoming bool
if i > 5 { if i > 5 {
@ -397,13 +400,14 @@ func TestChannelStateTransition(t *testing.T) {
} }
htlc := &HTLC{ htlc := &HTLC{
Incoming: incoming, Incoming: incoming,
Amt: 50000, Amt: 10,
RHash: key, RHash: key,
RefundTimeout: i, RefundTimeout: i,
RevocationDelay: i + 2, RevocationDelay: i + 2,
OutputIndex: uint16(i * 3), OutputIndex: uint16(i * 3),
} }
htlcs = append(htlcs, htlc) htlcs = append(htlcs, htlc)
htlcAmt += htlc.Amt
} }
// Create a new channel delta which includes the above HTLCs, some // Create a new channel delta which includes the above HTLCs, some
@ -508,6 +512,33 @@ func TestChannelStateTransition(t *testing.T) {
} }
} }
// Next modify the delta slightly, then create a new entry within the
// revocation log.
delta.UpdateNum = 2
delta.LocalBalance -= htlcAmt
delta.RemoteBalance += htlcAmt
delta.Htlcs = nil
if err := channel.AppendToRevocationLog(delta); err != nil {
t.Fatalf("unable to append to revocation log: %v", err)
}
// Once again, fetch the state and ensure it has been properly updated.
diskDelta, err = channel.FindPreviousState(uint64(delta.UpdateNum))
if err != nil {
t.Fatalf("unable to fetch past delta: %v", err)
}
if len(diskDelta.Htlcs) != 0 {
t.Fatalf("expected %v htlcs, got %v", 0, len(diskDelta.Htlcs))
}
if delta.LocalBalance != 1e8-htlcAmt {
t.Fatalf("mismatched balances, expected %v got %v", 1e8-htlcAmt,
delta.LocalBalance)
}
if delta.RemoteBalance != 1e8+htlcAmt {
t.Fatalf("mismatched balances, expected %v got %v", 1e8+htlcAmt,
delta.RemoteBalance)
}
// The revocation state stored on-disk should now also be identical. // The revocation state stored on-disk should now also be identical.
updatedChannel, err = cdb.FetchOpenChannels(channel.IdentityPub) updatedChannel, err = cdb.FetchOpenChannels(channel.IdentityPub)
if err != nil { if err != nil {