Holy shit pointer crap was annoying! Serialize/deserialize works now.
Running "go test -v" will show the serialization and deserialization. Doing the rest of the wire stuff should be *much* faster since I figured everything out...
This commit is contained in:
parent
0c0900006d
commit
f51a5a6458
130
lnwire/lnwire.go
130
lnwire/lnwire.go
@ -1,13 +1,13 @@
|
|||||||
package lnwire
|
package lnwire
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/btcsuite/btcd/btcec"
|
"github.com/btcsuite/btcd/btcec"
|
||||||
"github.com/btcsuite/btcd/wire"
|
"github.com/btcsuite/btcd/wire"
|
||||||
"github.com/btcsuite/btcutil"
|
"github.com/btcsuite/btcutil"
|
||||||
"io"
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
)
|
)
|
||||||
|
|
||||||
//Actual pkScript, not redeemScript
|
//Actual pkScript, not redeemScript
|
||||||
@ -42,11 +42,35 @@ type CreateChannel struct {
|
|||||||
Inputs []*wire.TxIn
|
Inputs []*wire.TxIn
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *CreateChannel) String() string {
|
||||||
|
|
||||||
|
var inputs string
|
||||||
|
for i, in := range c.Inputs {
|
||||||
|
inputs += fmt.Sprintf("\n Slice\t%d\n", i)
|
||||||
|
inputs += fmt.Sprintf("\tHash\t%s\n", in.PreviousOutPoint.Hash)
|
||||||
|
inputs += fmt.Sprintf("\tIndex\t%d\n", in.PreviousOutPoint.Index)
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("\n--- Begin CreateChannel ---\n") +
|
||||||
|
fmt.Sprintf("ChannelType:\t\t%x\n", c.ChannelType) +
|
||||||
|
fmt.Sprintf("FundingAmount:\t\t%s\n", c.FundingAmount.String()) +
|
||||||
|
fmt.Sprintf("ReserveAmount:\t\t%s\n", c.ReserveAmount.String()) +
|
||||||
|
fmt.Sprintf("MinFeePerKb:\t\t%s\n", c.MinFeePerKb.String()) +
|
||||||
|
fmt.Sprintf("MinTotalFundingAmount\t%s\n", c.MinTotalFundingAmount.String()) +
|
||||||
|
fmt.Sprintf("LockTime\t\t%d\n", c.LockTime) +
|
||||||
|
fmt.Sprintf("FeePayer\t\t%x\n", c.FeePayer) +
|
||||||
|
fmt.Sprintf("RevocationHash\t\t%x\n", c.RevocationHash) +
|
||||||
|
fmt.Sprintf("Pubkey\t\t\t%x\n", c.Pubkey.SerializeCompressed()) +
|
||||||
|
fmt.Sprintf("DeliveryPkScript\t%x\n", c.DeliveryPkScript) +
|
||||||
|
fmt.Sprintf("Inputs:") +
|
||||||
|
inputs +
|
||||||
|
fmt.Sprintf("--- End CreateChannel ---\n")
|
||||||
|
}
|
||||||
|
|
||||||
//Writes the big endian representation of element
|
//Writes the big endian representation of element
|
||||||
//Unified function to call when writing different types
|
//Unified function to call when writing different types
|
||||||
//Pre-allocate a byte-array of the correct size for cargo-cult security
|
//Pre-allocate a byte-array of the correct size for cargo-cult security
|
||||||
//More copies but whatever...
|
//More copies but whatever...
|
||||||
func writeElement(w *bytes.Buffer, element interface{}) error {
|
func writeElement(w io.Writer, element interface{}) error {
|
||||||
var err error
|
var err error
|
||||||
switch e := element.(type) {
|
switch e := element.(type) {
|
||||||
case uint8:
|
case uint8:
|
||||||
@ -64,9 +88,7 @@ func writeElement(w *bytes.Buffer, element interface{}) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
case btcutil.Amount:
|
case btcutil.Amount:
|
||||||
var b [8]byte
|
err = binary.Write(w, binary.BigEndian, int64(e))
|
||||||
binary.BigEndian.PutUint64(b[:], uint64(e))
|
|
||||||
_, err = w.Write(b[:])
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -137,14 +159,16 @@ func writeElement(w *bytes.Buffer, element interface{}) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func readElement(r *bytes.Buffer, element interface{}) error {
|
func readElement(r io.Reader, element interface{}) error {
|
||||||
var err error
|
var err error
|
||||||
switch e := element.(type) {
|
switch e := element.(type) {
|
||||||
case *uint8:
|
case *uint8:
|
||||||
err = binary.Read(r, binary.BigEndian, *e)
|
var b [1]uint8
|
||||||
|
_, err = r.Read(b[:])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
*e = b[0]
|
||||||
case *uint32:
|
case *uint32:
|
||||||
var b [4]byte
|
var b [4]byte
|
||||||
_, err = io.ReadFull(r, b[:])
|
_, err = io.ReadFull(r, b[:])
|
||||||
@ -158,14 +182,15 @@ func readElement(r *bytes.Buffer, element interface{}) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
*e = btcutil.Amount(binary.BigEndian.Uint64(b[:]))
|
*e = btcutil.Amount(int64(binary.BigEndian.Uint64(b[:])))
|
||||||
case *btcec.PublicKey:
|
case **btcec.PublicKey:
|
||||||
var b [33]byte
|
var b [33]byte
|
||||||
_, err = io.ReadFull(r, b[:])
|
_, err = io.ReadFull(r, b[:])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
e, err = btcec.ParsePubKey(b[:], btcec.S256())
|
x, err := btcec.ParsePubKey(b[:], btcec.S256())
|
||||||
|
*e = &*x
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -176,35 +201,36 @@ func readElement(r *bytes.Buffer, element interface{}) error {
|
|||||||
}
|
}
|
||||||
case *PkScript:
|
case *PkScript:
|
||||||
//Get the script length first
|
//Get the script length first
|
||||||
var scriptLength uint8
|
var scriptLength [1]uint8
|
||||||
err = binary.Read(r, binary.BigEndian, scriptLength)
|
_, err = r.Read(scriptLength[:])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if scriptLength > 25 {
|
|
||||||
|
if scriptLength[0] > 25 {
|
||||||
return fmt.Errorf("PkScript too long!")
|
return fmt.Errorf("PkScript too long!")
|
||||||
}
|
}
|
||||||
|
|
||||||
//Read the script length
|
//Read the script length
|
||||||
l := io.LimitReader(r, int64(scriptLength))
|
l := io.LimitReader(r, int64(scriptLength[0]))
|
||||||
|
*e, _ = ioutil.ReadAll(l)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
l.Read(*e)
|
case *[]*wire.TxIn:
|
||||||
case []*wire.TxIn:
|
|
||||||
//Read the size (1-byte number of txins)
|
//Read the size (1-byte number of txins)
|
||||||
var numScripts uint8
|
var numScripts [1]uint8
|
||||||
err = binary.Read(r, binary.BigEndian, numScripts)
|
_, err = r.Read(numScripts[:])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if numScripts > 127 {
|
if numScripts[0] > 127 {
|
||||||
return fmt.Errorf("Too many txins")
|
return fmt.Errorf("Too many txins")
|
||||||
}
|
}
|
||||||
|
|
||||||
//Append the actual TxIns
|
//Append the actual TxIns
|
||||||
var txins []*wire.TxIn
|
var txins []*wire.TxIn
|
||||||
for i := uint8(0); i < numScripts; i++ {
|
for i := uint8(0); i < numScripts[0]; i++ {
|
||||||
//Hash
|
//Hash
|
||||||
var h [32]byte
|
var h [32]byte
|
||||||
_, err = io.ReadFull(r, h[:])
|
_, err = io.ReadFull(r, h[:])
|
||||||
@ -227,7 +253,7 @@ func readElement(r *bytes.Buffer, element interface{}) error {
|
|||||||
//Create TxIn
|
//Create TxIn
|
||||||
txins = append(txins, wire.NewTxIn(outPoint, nil))
|
txins = append(txins, wire.NewTxIn(outPoint, nil))
|
||||||
}
|
}
|
||||||
e = txins
|
*e = *&txins
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return fmt.Errorf("Unknown type in readElement: %T", e)
|
return fmt.Errorf("Unknown type in readElement: %T", e)
|
||||||
@ -236,12 +262,12 @@ func readElement(r *bytes.Buffer, element interface{}) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *CreateChannel) DeserializeFundingRequest(r *bytes.Buffer) error {
|
func (c *CreateChannel) DeserializeFundingRequest(r io.Reader) error {
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
//Message Type
|
//Message Type (0/1)
|
||||||
var messageType uint8
|
var messageType uint8
|
||||||
err = readElement(r, messageType)
|
err = readElement(r, &messageType)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -249,74 +275,68 @@ func (c *CreateChannel) DeserializeFundingRequest(r *bytes.Buffer) error {
|
|||||||
return fmt.Errorf("Message type does not match FundingRequest")
|
return fmt.Errorf("Message type does not match FundingRequest")
|
||||||
}
|
}
|
||||||
|
|
||||||
//Channel Type
|
//Channel Type (1/1)
|
||||||
err = readElement(r, c.ChannelType)
|
err = readElement(r, &c.ChannelType)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Funding Amount
|
// Funding Amount (2/8)
|
||||||
err = readElement(r, c.FundingAmount)
|
err = readElement(r, &c.FundingAmount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Channel Minimum Capacity
|
// Channel Minimum Capacity (10/8)
|
||||||
var theirMinimumFunding btcutil.Amount
|
err = readElement(r, &c.MinTotalFundingAmount)
|
||||||
err = readElement(r, theirMinimumFunding)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
//Replace with their minimum if it's greater than ours
|
|
||||||
//We still need to locally validate that info
|
|
||||||
if theirMinimumFunding > c.MinTotalFundingAmount {
|
|
||||||
c.MinTotalFundingAmount = theirMinimumFunding
|
|
||||||
}
|
|
||||||
|
|
||||||
// Revocation Hash
|
|
||||||
err = readElement(r, c.RevocationHash)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Commitment Pubkey
|
// Revocation Hash (18/20)
|
||||||
err = readElement(r, c.Pubkey)
|
err = readElement(r, &c.RevocationHash)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Reserve Amount
|
// Commitment Pubkey (38/32)
|
||||||
err = readElement(r, c.ReserveAmount)
|
err = readElement(r, &c.Pubkey)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
//Minimum Transaction Fee Per Kb
|
// Reserve Amount (70/8)
|
||||||
err = readElement(r, c.MinFeePerKb)
|
err = readElement(r, &c.ReserveAmount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
//LockTime
|
//Minimum Transaction Fee Per Kb (78/8)
|
||||||
err = readElement(r, c.LockTime)
|
err = readElement(r, &c.MinFeePerKb)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
//FeePayer
|
//LockTime (86/4)
|
||||||
err = readElement(r, c.FeePayer)
|
err = readElement(r, &c.LockTime)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
//FeePayer (90/1)
|
||||||
|
err = readElement(r, &c.FeePayer)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delivery PkScript
|
// Delivery PkScript
|
||||||
err = readElement(r, c.DeliveryPkScript)
|
err = readElement(r, &c.DeliveryPkScript)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
//Create the TxIns
|
//Create the TxIns
|
||||||
err = readElement(r, c.Inputs)
|
err = readElement(r, &c.Inputs)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -326,7 +346,7 @@ func (c *CreateChannel) DeserializeFundingRequest(r *bytes.Buffer) error {
|
|||||||
|
|
||||||
//Serializes the fundingRequest from the CreateChannel struct
|
//Serializes the fundingRequest from the CreateChannel struct
|
||||||
//Writes the data to w
|
//Writes the data to w
|
||||||
func (c *CreateChannel) SerializeFundingRequest(w *bytes.Buffer) error {
|
func (c *CreateChannel) SerializeFundingRequest(w io.Writer) error {
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
//Fund request byte
|
//Fund request byte
|
||||||
|
@ -7,11 +7,18 @@ import (
|
|||||||
"github.com/btcsuite/btcd/btcec"
|
"github.com/btcsuite/btcd/btcec"
|
||||||
"github.com/btcsuite/btcd/wire"
|
"github.com/btcsuite/btcd/wire"
|
||||||
"github.com/btcsuite/btcutil"
|
"github.com/btcsuite/btcutil"
|
||||||
// "io/ioutil"
|
// "io"
|
||||||
|
"io/ioutil"
|
||||||
|
"reflect"
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
//For debugging, writes to /dev/shm/
|
||||||
|
//Maybe in the future do it if you do "go test -v"
|
||||||
|
WRITE_FILE = false
|
||||||
|
FILENAME = "/dev/shm/fundingRequest.raw"
|
||||||
|
|
||||||
//echo -n | openssl sha256 | openssl ripemd160
|
//echo -n | openssl sha256 | openssl ripemd160
|
||||||
ourRevocationHashBytes, _ = hex.DecodeString("9a2cbd088763db88dd8ba79e5726daa6aba4aa7e")
|
ourRevocationHashBytes, _ = hex.DecodeString("9a2cbd088763db88dd8ba79e5726daa6aba4aa7e")
|
||||||
ourRevocationHash [20]byte
|
ourRevocationHash [20]byte
|
||||||
@ -27,13 +34,15 @@ var (
|
|||||||
ourDeliveryPkScript, _ = hex.DecodeString("76a914e8048c0fb75bdecc91ebfb99c174f4ece29ffbd488ac")
|
ourDeliveryPkScript, _ = hex.DecodeString("76a914e8048c0fb75bdecc91ebfb99c174f4ece29ffbd488ac")
|
||||||
|
|
||||||
//echo -n | openssl sha256
|
//echo -n | openssl sha256
|
||||||
|
//This stuff gets reversed!!!
|
||||||
shaHash1Bytes, _ = hex.DecodeString("e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855")
|
shaHash1Bytes, _ = hex.DecodeString("e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855")
|
||||||
shaHash1, _ = wire.NewShaHash(shaHash1Bytes)
|
shaHash1, _ = wire.NewShaHash(shaHash1Bytes)
|
||||||
outpoint1 = wire.NewOutPoint(shaHash1, 0)
|
outpoint1 = wire.NewOutPoint(shaHash1, 0)
|
||||||
//echo | openssl sha256
|
//echo | openssl sha256
|
||||||
|
//This stuff gets reversed!!!
|
||||||
shaHash2Bytes, _ = hex.DecodeString("01ba4719c80b6fe911b091a7c05124b64eeece964e09c058ef8f9805daca546b")
|
shaHash2Bytes, _ = hex.DecodeString("01ba4719c80b6fe911b091a7c05124b64eeece964e09c058ef8f9805daca546b")
|
||||||
shaHash2, _ = wire.NewShaHash(shaHash2Bytes)
|
shaHash2, _ = wire.NewShaHash(shaHash2Bytes)
|
||||||
outpoint2 = wire.NewOutPoint(shaHash2, 0)
|
outpoint2 = wire.NewOutPoint(shaHash2, 1)
|
||||||
//create inputs from outpoint1 and outpoint2
|
//create inputs from outpoint1 and outpoint2
|
||||||
ourInputs = []*wire.TxIn{wire.NewTxIn(outpoint1, nil), wire.NewTxIn(outpoint2, nil)}
|
ourInputs = []*wire.TxIn{wire.NewTxIn(outpoint1, nil), wire.NewTxIn(outpoint2, nil)}
|
||||||
|
|
||||||
@ -51,15 +60,46 @@ var (
|
|||||||
DeliveryPkScript: ourDeliveryPkScript,
|
DeliveryPkScript: ourDeliveryPkScript,
|
||||||
Inputs: ourInputs,
|
Inputs: ourInputs,
|
||||||
}
|
}
|
||||||
|
serializedString = "30000000000005f5e1000000000008f0d1809a2cbd088763db88dd8ba79e5726daa6aba4aa7e02f977808cb9577897582d7524b562691e180953dd0008eb44e09594c539d6daee00000000000200000000000000004e20000010e0001976a914e8048c0fb75bdecc91ebfb99c174f4ece29ffbd488ac02e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b8550000000001ba4719c80b6fe911b091a7c05124b64eeece964e09c058ef8f9805daca546b00000001"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestFundingRequestSerializeDeserialize(t *testing.T) {
|
func TestFundingRequestSerializeDeserialize(t *testing.T) {
|
||||||
b := new(bytes.Buffer)
|
b := new(bytes.Buffer)
|
||||||
err := createChannel.SerializeFundingRequest(b)
|
err := createChannel.SerializeFundingRequest(b)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println("ERR")
|
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
t.Error("Serialization error")
|
t.Error("Serialization error")
|
||||||
}
|
}
|
||||||
t.Logf("ASDF: %x\n", b.Bytes())
|
t.Logf("Serialized Funding Request: %x\n", b.Bytes())
|
||||||
|
|
||||||
|
//Check if we serialized correctly
|
||||||
|
if serializedString != hex.EncodeToString(b.Bytes()) {
|
||||||
|
t.Error("Serialization does not match expected")
|
||||||
|
}
|
||||||
|
|
||||||
|
//So I can do: hexdump -C /dev/shm/fundingRequest.raw
|
||||||
|
if WRITE_FILE {
|
||||||
|
err = ioutil.WriteFile(FILENAME, b.Bytes(), 0644)
|
||||||
|
if err != nil {
|
||||||
|
t.Error("File write error")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//Test deserialization
|
||||||
|
//Make a new buffer just to be clean
|
||||||
|
c := new(bytes.Buffer)
|
||||||
|
c.Write(b.Bytes())
|
||||||
|
|
||||||
|
var newChannel CreateChannel
|
||||||
|
err = newChannel.DeserializeFundingRequest(c)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
t.Error("Deserialiation Error")
|
||||||
|
}
|
||||||
|
|
||||||
|
if !reflect.DeepEqual(newChannel, createChannel) {
|
||||||
|
t.Error("Deserialization does not match!")
|
||||||
|
}
|
||||||
|
|
||||||
|
t.Log(newChannel.String())
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user