- Published on
Building blockchain in Go. Part 6: Transactions 2
- Authors
- Name
- Eddie Ho
- @doublemay33
Table of Contents
Introduction
In the very first part of this series I said that blockchain is a distributed database. Back then, we decided to skip the “distributed” part and focus on the “database” part. So far, we’ve implemented almost all the things that make a blockchain database. In this post, we’ll cover some mechanisms that were skipped in the previous parts, and in the next part we’ll start working on the distributed nature of blockchain.
Previous parts:
This part introduces significant code changes, so it makes no sense explaining all of them here. Please refer to this page to see all the changes since the last article.
Reward
One tiny thing we skipped in a previous article is rewards for mining. And we already have everything to implement it.
The reward is just a coinbase transaction. When a mining node starts mining a new block, it takes transactions from the queue and prepends a coinbase transaction to them. The coinbase transaction’s only output contains miner’s public key hash.
Implementing rewards is as easy as updating the send
command:
// pkg/cli/cli_send.go
func (cli *CLI) send(from, to string, amount int) {
...
bc := blockchain.NewBlockchain()
UTXOSet := blockchain.UTXOSet{bc}
defer bc.db.Close()
tx := blockchain.NewUTXOTransaction(from, to, amount, &UTXOSet)
cbTx := blockchain.NewCoinbaseTX(from, "")
txs := []*blockchain.Transaction{cbTx, tx}
newBlock := bc.MineBlock(txs)
fmt.Println("Success!")
}
In our implementation, the one who creates a transaction mines the new block, and thus, receives a reward.
The UTXO Set
In Part 3: Persistence and CLI we studied the way Bitcoin Core stores blocks in a database. It was said that blocks are stored in blocks
database and transaction outputs are stored in chainstate
database. Let me remind you what the structure of chainstate
is:
- 'c' + 32-byte transaction hash -> unspent transaction output record for that transaction
- 'B' -> 32-byte block hash: the block hash up to which the database represents the unspent transaction outputs
Since that article, we’ve already implemented transactions, but we haven’t used the chainstate
to store their outputs. So, this is what we’re going to do now.
chainstate
doesn’t store transactions. Instead, it stores what is called the UTXO set, or the set of unspent transaction outputs. Besides this, it stores “the block hash up to which the database represents the unspent transaction outputs”, which we’ll omit for now because we’re not using block heights (but we’ll implement them in next articles).
So, why do we want to have the UTXO set?
Consider the Blockchain.FindUnspentTransactions
method we’ve implemented earlier:
// pkg/blockchain/blockchain.go
func (bc *Blockchain) FindUnspentTransactions(pubKeyHash []byte) []Transaction {
...
bci := bc.Iterator()
for {
block := bci.Next()
for _, tx := range block.Transactions {
...
}
if len(block.PrevBlockHash) == 0 {
break
}
}
...
}
The function finds transactions with unspent outputs. Since transactions are stored in blocks, it iterates over each block in the blockchain and checks every transaction in it. As of September 18, 2017, there’re 485,860 blocks in Bitcoin and the whole database takes 140+ Gb of disk space. This means that one has to run a full node to validate transactions. Moreover, validating transactions would require iterating over many blocks.
The solution to the problem is to have an index that stores only unspent outputs, and this is what the UTXO set does: this is a cache that is built from all blockchain transactions (by iterating over blocks, yes, but this is done only once), and is later used to calculate balance and validate new transactions. The UTXO set is about 2.7 Gb as of September 2017.
Alright, let’s think what we need to change to implement the UTXO set. Currently, the following methods are used to find transactions:
Blockchain.FindUnspentTransactions
– the main function that finds transactions with unspent outputs. It’s this function where the iteration of all blocks happens.Blockchain.FindSpendableOutputs
– this function is used when a new transaction is created. If finds the enough number of outputs holding required amount. UsesBlockchain.FindUnspentTransactions
.Blockchain.FindUTXO
– finds unspent outputs for a public key hash, used to get balance. UsesBlockchain.FindUnspentTransactions
.Blockchain.FindTransaction
– finds a transaction in the blockchain by its ID. It iterates over all blocks until finds it.
As you can see, all the methods iterate over blocks in the database. But we cannot improve all of them for now, because the UTXO set doesn’t store all transactions, but only those that have unspent outputs. Thus, it cannot be used in Blockchain.FindTransaction
.
So, we want the following methods:
Blockchain.FindUTXO
– finds all unspent outputs by iterating over blocks.UTXOSet.Reindex
— uses FindUTXO to find unspent outputs, and stores them in a database. This is where caching happens.UTXOSet.FindSpendableOutputs
– analog ofBlockchain.FindSpendableOutputs
, but uses the UTXO set.UTXOSet.FindUTXO
– analog ofBlockchain.FindUTXO
, but uses the UTXO set.Blockchain.FindTransaction
remains the same.
Thus, the two most frequently used functions will use the cache from now! Let’s start coding.
// pkg/blockchain/utxo_set.go
type UTXOSet struct {
Blockchain *Blockchain
}
We’ll use a single database, but we’ll store the UTXO set in a different bucket. Thus, UTXOSet
is coupled with Blockchain
.
// pkg/blockchain/utxo_set.go
// Reindex rebuilds the UTXO set
func (u UTXOSet) Reindex() {
db := u.Blockchain.db
bucketName := []byte(internal.UtxoBucket)
err := db.Update(func(tx *bolt.Tx) error {
tx.DeleteBucket(bucketName)
_, err := tx.CreateBucket(bucketName)
return err
})
utils.Check(err)
UTXO := u.Blockchain.FindUTXO()
err = db.Update(func(tx *bolt.Tx) error {
b := tx.Bucket(bucketName)
for txID, outs := range UTXO {
key, err := hex.DecodeString(txID)
utils.Check(err)
err = b.Put(key, outs.Serialize())
utils.Check(err)
}
return nil
})
utils.Check(err)
}
This method creates the UTXO set initially. First, it removes the bucket if it exists, then it gets all unspent outputs from blockchain, and finally it saves the outputs to the bucket.
Blockchain.FindUTXO
is almost identical to Blockchain.FindUnspentTransactions
, but now it returns a map of TransactionID → TransactionOutputs
pairs.
Now, the UTXO set can be used to send coins:
// pkg/blockchain/utxo_set.go
// FindSpendableOutputs finds and returns unspent outputs to reference in inputs
func (u *UTXOSet) FindSpendableOutputs(
pubKeyHash []byte,
amount int,
) (int, map[string][]int) {
unspentOutputs := make(map[string][]int)
accumlated := 0
db := u.Blockchain.db
err := db.View(func(tx *bolt.Tx) error {
b := tx.Bucket([]byte(internal.UtxoBucket))
c := b.Cursor()
for k, v := c.First(); k != nil; k, v = c.Next() {
txID := hex.EncodeToString(k)
outs := DeserializeOutputs(v)
for outIdx, out := range outs.Outputs {
if out.IsLockedWithKey(pubKeyHash) && accumlated < amount {
accumlated += out.Value
unspentOutputs[txID] = append(unspentOutputs[txID], outIdx)
}
}
}
return nil
})
utils.Check(err)
return accumlated, unspentOutputs
}
Or check balance:
// pkg/blockchain/utxo_set.go
// FindUTXO finds UTXO for a public key hash
func (u *UTXOSet) FindUTXO(pubKeyHash []byte) []TXOutput {
var UTXOs []TXOutput
db := u.Blockchain.db
err := db.View(func(tx *bolt.Tx) error {
b := tx.Bucket([]byte(internal.UtxoBucket))
c := b.Cursor()
for k, v := c.First(); k != nil; k, v = c.Next() {
outs := DeserializeOutputs(v)
for _, out := range outs.Outputs {
if out.IsLockedWithKey(pubKeyHash) {
UTXOs = append(UTXOs, out)
}
}
}
return nil
})
utils.Check(err)
return UTXOs
}
These are slightly modified versions of corresponding Blockchain methods. Those Blockchain methods are not needed anymore.
Having the UTXO set means that our data (transactions) are now split into to storages: actual transactions are stored in the blockchain, and unspent outputs are stored in the UTXO set. Such separation requires solid synchronization mechanism because we want the UTXO set to always be updated and store outputs of most recent transactions. But we don’t want to reindex every time a new block is mined because it’s these frequent blockchain scans that we want to avoid. Thus, we need a mechanism of updating the UTXO set:
// pkg/blockchain/utxo_set.go
// Update updates the UTXO set with transactions from the Block
// The Block is considered to be the tip of a blockchain
func (u *UTXOSet) Update(block *Block) {
db := u.Blockchain.db
err := db.Update(func(tx *bolt.Tx) error {
b := tx.Bucket([]byte(internal.UtxoBucket))
for _, tx := range block.Transactions {
if !tx.IsCoinbase() {
for _, vin := range tx.Vin {
updateOuts := TXOutputs{}
outsBytes := b.Get(vin.Txid)
outs := DeserializeOutputs(outsBytes)
// Put unspent outputs into updateOuts
for outIdx, out := range outs.Outputs {
if outIdx != vin.Vout {
updateOuts.Outputs = append(updateOuts.Outputs, out)
}
}
// Remove pair if all outputs are spent, otherwise save the
// updated one
if len(updateOuts.Outputs) == 0 {
err := b.Delete(vin.Txid)
utils.Check(err)
} else {
err := b.Put(vin.Txid, updateOuts.Serialize())
utils.Check(err)
}
}
}
// Insert outputs of newly mined transactions
newOutputs := TXOutputs{}
newOutputs.Outputs = append(newOutputs.Outputs, tx.Vout...)
err := b.Put(tx.ID, newOutputs.Serialize())
utils.Check(err)
}
return nil
})
utils.Check(err)
}
The method looks big, but what it does is quite straightforward. When a new block is mined, the UTXO set should be updated. Updating means removing spent outputs and adding unspent outputs from newly mined transactions. If a transaction which outputs were removed, contains no more outputs, it’s removed as well. Quite simple!
Let’s now use the UTXO set where it’s necessary:
// pkg/cli/cli_createblockchain.go
func (cli *CLI) createBlockchain(address string) {
...
bc := blockchain.CreateBlockchain(address)
defer bc.CloseDB()
utxoSet := blockchain.UTXOSet{Blockchain: bc}
utxoSet.Reindex()
...
}
Reindexing happens right after a new blockchain is created. For now, this is the only place where Reindex
is used, even though it looks excessive here because in the beginning of a blockchain there’s only one block with one transaction, and Update
could’ve been used instead. But we might need the reindexing mechanism in the future.
// pkg/cli/cli_send.go
func (cli *CLI) send(from, to string, amount int) {
...
newBlock := bc.MineBlock(txs)
UTXOSet.Update(newBlock)
}
And the UTXO set is updated after a new block is mined.
Let’s check that it works
$ ./blockchain-go createblockchain -address 178vmmuHgj54fzK1cgYvTdJkjWBtfFHFzw
0000103b30fac8ef92bf70823fc541435caa4fd0e9f2fb0ebd58075356c312b6
Done!
$ ./blockchain-go send -from 178vmmuHgj54fzK1cgYvTdJkjWBtfFHFzw -to 1KySbPGvNVE4rf57ehtMAKB2HFEgJyNvsE -amount 6
000013d1a6f62380c2e2c8c6a6a12b879fd207663c8a494a58875e496cba554a
Success!
$ ./blockchain-go send -from 178vmmuHgj54fzK1cgYvTdJkjWBtfFHFzw -to 1McmeAmzBcFjQvVjH6ZmMWSM8UuXKzTqck -amount 6
0000117c5fabadab330bd7a9c75db12b33dda7918838995ae0aecfc9286331c9
Success!
$ ./blockchain-go getbalance -address 178vmmuHgj54fzK1cgYvTdJkjWBtfFHFzw
Balance of '178vmmuHgj54fzK1cgYvTdJkjWBtfFHFzw': 8
$ ./blockchain-go getbalance -address 1KySbPGvNVE4rf57ehtMAKB2HFEgJyNvsE
Balance of '1KySbPGvNVE4rf57ehtMAKB2HFEgJyNvsE': 6
$ ./blockchain-go getbalance -address 1McmeAmzBcFjQvVjH6ZmMWSM8UuXKzTqck
Balance of '1McmeAmzBcFjQvVjH6ZmMWSM8UuXKzTqck': 6
Nice! The 1JnMDSqVoHi4TEFXNw5wJ8skPsPf4LHkQ1 address received reward 3 times:
- Once for mining the genesis blocks.
- Once for mining the block
0000001f75cb3a5033aeecbf6a8d378e15b25d026fb0a665c7721a5bb0faa21b
- And once for mining the block
000000cc51e665d53c78af5e65774a72fc7b864140a8224bf4e7709d8e0fa433
Merkle Tree
There’s one more optimization mechanism I’d like to discuss in this post.
As it was said above, the full Bitcoin database (i.e., blockchain) takes more than 140 Gb of disk space. Because of the decentralized nature of Bitcoin, every node in the network must be independent and self-sufficient, i.e. every node must store a full copy of the blockchain. With many people starting using Bitcoin, this rule becomes more difficult to follow: it’s not likely that everyone will run a full node. Also, since nodes are full-fledged participants of the network, they have responsibilities: they must verify transactions and blocks. Also, there’s certain internet traffic required to interact with other nodes and download new blocks.
In the original Bitcoin paper published by Satoshi Nakamoto, there was a solution for this problem: Simplified Payment Verification (SPV). SPV is a light Bitcoin node that doesn’t download the whole blockchain and doesn’t verify blocks and transactions. Instead, it finds transactions in blocks (to verify payments) and is linked to a full node to retrieve just necessary data. This mechanism allows having multiple light wallet nodes with running just one full node.
For SPV to be possible, there should be a way to check if a block contains certain transaction without downloading the whole block. And this is where Merkle tree comes into play.
Merkle trees are used by Bitcoin to obtain transactions hash, which is then saved in block headers and is considered by the proof-of-work system. Until now, we just concatenated hashes of each transaction in a block and applied SHA-256 to them. This is also a good way of getting a unique representation of block transactions, but it doesn’t have benefits of Merkle trees.
Let’s look at a Merkle tree:
A Merkle tree is built for each block, and it starts with leaves (the bottom of the tree), where a leaf is a transaction hash (Bitcoins uses double SHA256 hashing). The number of leaves must be even, but not every block contains an even number of transactions. In case there is an odd number of transactions, the last transaction is duplicated (in the Merkle tree, not in the block!).
Moving from the bottom up, leaves are grouped in pairs, their hashes are concatenated, and a new hash is obtained from the concatenated hashes. The new hashes form new tree nodes. This process is repeated until there’s just one node, which is called the root of the tree. The root hash is then used as the unique representation of the transactions, is saved in block headers, and is used in the proof-of-work system.
The benefit of Merkle trees is that a node can verify membership of certain transaction without downloading the whole block. Just a transaction hash, a Merkle tree root hash, and a Merkle path are required for this.
Finally, let’s write code:
// pkg/merkletree/merkle_node.go
type MerkleNode struct {
Left *MerkleNode
Right *MerkleNode
Data []byte
}
// pkg/merkletree/merkle_tree.go
type MerkleTree struct {
RootNode *MerkleNode
}
We start with structs. Every MerkleNode
keeps data and links to its branches. MerkleTree
is actually the root node linked to the next nodes, which are in their turn linked to further nodes, etc.
Let’s create a new node first:
// pkg/merkletree/merkle_node.go
func NewMerkleNode(left, right *MerkleNode, data []byte) *MerkleNode {
mNode := MerkleNode{}
if left == nil && right == nil {
hash := sha256.Sum256(data)
mNode.Data = hash[:]
} else {
prevHashes := append(left.Data, right.Data...)
hash := sha256.Sum256(prevHashes)
mNode.Data = hash[:]
}
mNode.Left = left
mNode.Right = right
return &mNode
}
Every node contains some data. When a node is a leaf, the data is passed from the outside (a serialized transaction in our case). When a node is linked to other nodes, it takes their data and concatenates and hashes it.
// pkg/merkletree/merkle_tree.go
func NewMerkleTree(data [][]byte) *MerkleTree {
var nodes []MerkleNode
if len(data)%2 != 0 {
data = append(data, data[len(data)-1])
}
for _, datum := range data {
node := NewMerkleNode(nil, nil, datum)
nodes = append(nodes, *node)
}
for i := 0; i < len(data)/2; i++ {
var newLevel []MerkleNode
for j := 0; j < len(nodes); j += 2 {
node := NewMerkleNode(&nodes[j], &nodes[j+1], nil)
newLevel = append(newLevel, *node)
}
nodes = newLevel
}
mTree := MerkleTree{&nodes[0]}
return &mTree
}
When a new tree is created, the first thing to ensure is that there is an even number of leaves. After that, data (which is an array of serialized transactions) is converted into tree leaves, and a tree is grown from these leaves.
Now, let’s modify Block.HashTransactions
, which is used in the proof-of-work system to obtain transactions hash:
// pkg/blockchain/block.go
func (b *Block) HashTransactions() []byte {
var transactions [][]byte
for _, tx := range b.Transactions {
transactions = append(transactions, tx.Serialize())
}
mTree := merkletree.NewMerkleTree(transactions)
return mTree.RootNode.Data
}
First, transactions are serialized (using encoding/gob
), and then they are used to build a Merkle tree. The root of the tree will serve as the unique identifier of block’s transactions.
Conclusion
And that’s it! We’ve implemented almost all key feature of a blockchain-based cryptocurrency. We have blockchain, addresses, mining, and transactions. But there’s one more thing that gives life to all these mechanisms and makes Bitcoin a global system: consensus. In the next article, we’ll start implementing the “decentralized” part of the blockchain. Stay tuned!
References: