2014-01-09 06:46:05 +01:00
|
|
|
// Copyright (c) 2013-2014 Conformal Systems LLC.
|
2013-05-29 00:32:35 +02:00
|
|
|
// Use of this source code is governed by an ISC
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package btcutil
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"github.com/conformal/btcwire"
|
|
|
|
)
|
|
|
|
|
|
|
|
// OutOfRangeError describes an error due to accessing an element that is out
|
|
|
|
// of range.
|
|
|
|
type OutOfRangeError string
|
|
|
|
|
|
|
|
// BlockHeightUnknown is the value returned for a block height that is unknown.
|
|
|
|
// This is typically because the block has not been inserted into the main chain
|
|
|
|
// yet.
|
|
|
|
const BlockHeightUnknown = int64(-1)
|
|
|
|
|
|
|
|
// Error satisfies the error interface and prints human-readable errors.
|
|
|
|
func (e OutOfRangeError) Error() string {
|
|
|
|
return string(e)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Block defines a bitcoin block that provides easier and more efficient
|
2013-08-05 19:41:31 +02:00
|
|
|
// manipulation of raw blocks. It also memoizes hashes for the block and its
|
|
|
|
// transactions on their first access so subsequent accesses don't have to
|
|
|
|
// repeat the relatively expensive hashing operations.
|
2013-05-29 00:32:35 +02:00
|
|
|
type Block struct {
|
2014-03-24 19:21:44 +01:00
|
|
|
msgBlock *btcwire.MsgBlock // Underlying MsgBlock
|
|
|
|
serializedBlock []byte // Serialized bytes for the block
|
|
|
|
blockSha *btcwire.ShaHash // Cached block hash
|
|
|
|
blockHeight int64 // Height in the main block chain
|
|
|
|
transactions []*Tx // Transactions
|
|
|
|
txnsGenerated bool // ALL wrapped transactions generated
|
2013-05-29 00:32:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// MsgBlock returns the underlying btcwire.MsgBlock for the Block.
|
|
|
|
func (b *Block) MsgBlock() *btcwire.MsgBlock {
|
|
|
|
// Return the cached block.
|
|
|
|
return b.msgBlock
|
|
|
|
}
|
|
|
|
|
2013-08-05 19:41:31 +02:00
|
|
|
// Bytes returns the serialized bytes for the Block. This is equivalent to
|
|
|
|
// calling Serialize on the underlying btcwire.MsgBlock, however it caches the
|
|
|
|
// result so subsequent calls are more efficient.
|
|
|
|
func (b *Block) Bytes() ([]byte, error) {
|
|
|
|
// Return the cached serialized bytes if it has already been generated.
|
|
|
|
if len(b.serializedBlock) != 0 {
|
|
|
|
return b.serializedBlock, nil
|
2013-05-29 00:32:35 +02:00
|
|
|
}
|
|
|
|
|
2013-08-05 19:41:31 +02:00
|
|
|
// Serialize the MsgBlock.
|
2013-05-29 00:32:35 +02:00
|
|
|
var w bytes.Buffer
|
2013-08-05 19:41:31 +02:00
|
|
|
err := b.msgBlock.Serialize(&w)
|
2013-05-29 00:32:35 +02:00
|
|
|
if err != nil {
|
2013-08-05 19:41:31 +02:00
|
|
|
return nil, err
|
2013-05-29 00:32:35 +02:00
|
|
|
}
|
2013-08-05 19:41:31 +02:00
|
|
|
serializedBlock := w.Bytes()
|
2013-05-29 00:32:35 +02:00
|
|
|
|
2013-08-05 19:41:31 +02:00
|
|
|
// Cache the serialized bytes and return them.
|
|
|
|
b.serializedBlock = serializedBlock
|
|
|
|
return serializedBlock, nil
|
2013-05-29 00:32:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Sha returns the block identifier hash for the Block. This is equivalent to
|
|
|
|
// calling BlockSha on the underlying btcwire.MsgBlock, however it caches the
|
|
|
|
// result so subsequent calls are more efficient.
|
|
|
|
func (b *Block) Sha() (*btcwire.ShaHash, error) {
|
|
|
|
// Return the cached block hash if it has already been generated.
|
|
|
|
if b.blockSha != nil {
|
|
|
|
return b.blockSha, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate the block hash. Ignore the error since BlockSha can't
|
|
|
|
// currently fail.
|
2013-08-05 19:41:31 +02:00
|
|
|
sha, _ := b.msgBlock.BlockSha()
|
2013-05-29 00:32:35 +02:00
|
|
|
|
|
|
|
// Cache the block hash and return it.
|
|
|
|
b.blockSha = &sha
|
|
|
|
return &sha, nil
|
|
|
|
}
|
|
|
|
|
2013-10-28 16:32:41 +01:00
|
|
|
// Tx returns a wrapped transaction (btcutil.Tx) for the transaction at the
|
|
|
|
// specified index in the Block. The supplied index is 0 based. That is to
|
|
|
|
// say, the first transaction in the block is txNum 0. This is nearly
|
|
|
|
// equivalent to accessing the raw transaction (btcwire.MsgTx) from the
|
|
|
|
// underlying btcwire.MsgBlock, however the wrapped transaction has some helpful
|
|
|
|
// properties such as caching the hash so subsequent calls are more efficient.
|
|
|
|
func (b *Block) Tx(txNum int) (*Tx, error) {
|
2013-05-29 00:32:35 +02:00
|
|
|
// Ensure the requested transaction is in range.
|
2014-01-19 03:59:07 +01:00
|
|
|
numTx := uint64(len(b.msgBlock.Transactions))
|
2013-05-29 00:32:35 +02:00
|
|
|
if txNum < 0 || uint64(txNum) > numTx {
|
|
|
|
str := fmt.Sprintf("transaction index %d is out of range - max %d",
|
|
|
|
txNum, numTx-1)
|
|
|
|
return nil, OutOfRangeError(str)
|
|
|
|
}
|
|
|
|
|
2013-10-28 16:32:41 +01:00
|
|
|
// Generate slice to hold all of the wrapped transactions if needed.
|
|
|
|
if len(b.transactions) == 0 {
|
|
|
|
b.transactions = make([]*Tx, numTx)
|
2013-05-29 00:32:35 +02:00
|
|
|
}
|
|
|
|
|
2013-10-28 16:32:41 +01:00
|
|
|
// Return the wrapped transaction if it has already been generated.
|
|
|
|
if b.transactions[txNum] != nil {
|
|
|
|
return b.transactions[txNum], nil
|
2013-05-29 00:32:35 +02:00
|
|
|
}
|
|
|
|
|
2013-10-28 16:32:41 +01:00
|
|
|
// Generate and cache the wrapped transaction and return it.
|
|
|
|
newTx := NewTx(b.msgBlock.Transactions[txNum])
|
|
|
|
newTx.SetIndex(txNum)
|
|
|
|
b.transactions[txNum] = newTx
|
|
|
|
return newTx, nil
|
|
|
|
}
|
2013-05-29 00:32:35 +02:00
|
|
|
|
2013-10-28 16:32:41 +01:00
|
|
|
// Transactions returns a slice of wrapped transactions (btcutil.Tx) for all
|
|
|
|
// transactions in the Block. This is nearly equivalent to accessing the raw
|
|
|
|
// transactions (btcwire.MsgTx) in the underlying btcwire.MsgBlock, however it
|
|
|
|
// instead provides easy access to wrapped versions (btcutil.Tx) of them.
|
|
|
|
func (b *Block) Transactions() []*Tx {
|
|
|
|
// Return transactions if they have ALL already been generated. This
|
|
|
|
// flag is necessary because the wrapped transactions are lazily
|
|
|
|
// generated in a sparse fashion.
|
|
|
|
if b.txnsGenerated {
|
|
|
|
return b.transactions
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate slice to hold all of the wrapped transactions if needed.
|
|
|
|
if len(b.transactions) == 0 {
|
2014-01-19 03:59:07 +01:00
|
|
|
b.transactions = make([]*Tx, len(b.msgBlock.Transactions))
|
2013-10-28 16:32:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Generate and cache the wrapped transactions for all that haven't
|
|
|
|
// already been done.
|
|
|
|
for i, tx := range b.transactions {
|
|
|
|
if tx == nil {
|
|
|
|
newTx := NewTx(b.msgBlock.Transactions[i])
|
|
|
|
newTx.SetIndex(i)
|
|
|
|
b.transactions[i] = newTx
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
b.txnsGenerated = true
|
|
|
|
return b.transactions
|
|
|
|
}
|
|
|
|
|
|
|
|
// TxSha returns the hash for the requested transaction number in the Block.
|
|
|
|
// The supplied index is 0 based. That is to say, the first transaction in the
|
|
|
|
// block is txNum 0. This is equivalent to calling TxSha on the underlying
|
|
|
|
// btcwire.MsgTx, however it caches the result so subsequent calls are more
|
|
|
|
// efficient.
|
|
|
|
func (b *Block) TxSha(txNum int) (*btcwire.ShaHash, error) {
|
|
|
|
// Attempt to get a wrapped transaction for the specified index. It
|
|
|
|
// will be created lazily if needed or simply return the cached version
|
|
|
|
// if it has already been generated.
|
|
|
|
tx, err := b.Tx(txNum)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Defer to the wrapped transaction which will return the cached hash if
|
|
|
|
// it has already been generated.
|
|
|
|
return tx.Sha(), nil
|
2013-05-29 00:32:35 +02:00
|
|
|
}
|
|
|
|
|
2013-05-30 18:39:43 +02:00
|
|
|
// TxLoc returns the offsets and lengths of each transaction in a raw block.
|
2013-05-29 01:43:13 +02:00
|
|
|
// It is used to allow fast indexing into transactions within the raw byte
|
|
|
|
// stream.
|
2013-05-30 18:31:22 +02:00
|
|
|
func (b *Block) TxLoc() ([]btcwire.TxLoc, error) {
|
2013-08-05 19:41:31 +02:00
|
|
|
rawMsg, err := b.Bytes()
|
2013-05-29 00:32:35 +02:00
|
|
|
if err != nil {
|
2013-05-30 18:31:22 +02:00
|
|
|
return nil, err
|
2013-05-29 00:32:35 +02:00
|
|
|
}
|
|
|
|
rbuf := bytes.NewBuffer(rawMsg)
|
|
|
|
|
|
|
|
var mblock btcwire.MsgBlock
|
2013-08-05 19:41:31 +02:00
|
|
|
txLocs, err := mblock.DeserializeTxLoc(rbuf)
|
2013-05-29 00:32:35 +02:00
|
|
|
if err != nil {
|
2013-05-30 18:31:22 +02:00
|
|
|
return nil, err
|
2013-05-29 00:32:35 +02:00
|
|
|
}
|
2013-05-30 18:13:08 +02:00
|
|
|
return txLocs, err
|
2013-05-29 00:32:35 +02:00
|
|
|
}
|
|
|
|
|
2013-08-05 19:41:31 +02:00
|
|
|
// Height returns the saved height of the block in the block chain. This value
|
2013-05-29 00:32:35 +02:00
|
|
|
// will be BlockHeightUnknown if it hasn't already explicitly been set.
|
|
|
|
func (b *Block) Height() int64 {
|
|
|
|
return b.blockHeight
|
|
|
|
}
|
|
|
|
|
2013-08-05 19:41:31 +02:00
|
|
|
// SetHeight sets the height of the block in the block chain.
|
2013-05-29 00:32:35 +02:00
|
|
|
func (b *Block) SetHeight(height int64) {
|
|
|
|
b.blockHeight = height
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewBlock returns a new instance of a bitcoin block given an underlying
|
2013-08-05 19:41:31 +02:00
|
|
|
// btcwire.MsgBlock. See Block.
|
|
|
|
func NewBlock(msgBlock *btcwire.MsgBlock) *Block {
|
2013-05-29 00:32:35 +02:00
|
|
|
return &Block{
|
2013-08-05 19:41:31 +02:00
|
|
|
msgBlock: msgBlock,
|
|
|
|
blockHeight: BlockHeightUnknown,
|
2013-05-29 00:32:35 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewBlockFromBytes returns a new instance of a bitcoin block given the
|
2013-08-05 19:41:31 +02:00
|
|
|
// serialized bytes. See Block.
|
|
|
|
func NewBlockFromBytes(serializedBlock []byte) (*Block, error) {
|
|
|
|
// Deserialize the bytes into a MsgBlock.
|
2013-05-29 00:32:35 +02:00
|
|
|
var msgBlock btcwire.MsgBlock
|
2013-08-05 19:41:31 +02:00
|
|
|
br := bytes.NewBuffer(serializedBlock)
|
|
|
|
err := msgBlock.Deserialize(br)
|
2013-05-29 00:32:35 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
b := Block{
|
|
|
|
msgBlock: &msgBlock,
|
2013-08-05 19:41:31 +02:00
|
|
|
serializedBlock: serializedBlock,
|
2013-05-29 00:32:35 +02:00
|
|
|
blockHeight: BlockHeightUnknown,
|
|
|
|
}
|
|
|
|
return &b, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewBlockFromBlockAndBytes returns a new instance of a bitcoin block given
|
2013-08-05 19:41:31 +02:00
|
|
|
// an underlying btcwire.MsgBlock and the serialized bytes for it. See Block.
|
|
|
|
func NewBlockFromBlockAndBytes(msgBlock *btcwire.MsgBlock, serializedBlock []byte) *Block {
|
2013-05-29 00:32:35 +02:00
|
|
|
return &Block{
|
|
|
|
msgBlock: msgBlock,
|
2013-08-05 19:41:31 +02:00
|
|
|
serializedBlock: serializedBlock,
|
2013-05-29 00:32:35 +02:00
|
|
|
blockHeight: BlockHeightUnknown,
|
|
|
|
}
|
|
|
|
}
|