-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBlockChain.go
127 lines (111 loc) · 3.27 KB
/
BlockChain.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
/**
@author:BOEN
@data:2022/8/21
@note:
**/
package main
import (
"github.com/boltdb/bolt"
"github.com/go-kit/kit/log"
"os"
)
//数据库名字
const dbFile = "blockchain.db"
//数据库表名
const blocksBucket = "blocks"
type Blockchain struct {
//是区块链里面的最后一个区块的Hash
Tip []byte
//Bolt持久化操作对象
DB *bolt.DB
}
/**
* @Title NewBlockChain
* @Description //创建一个带有创世区块的区块链(如果没有区块链:1.创建创世区块-->2.存储到数据库-->3.将创世区块哈希保存为最后一个块的Hash-->4.创建一个新的BlockChain实例,初始化时tip指向创世区块)
* @Author Cofeesy 11:13 2022/8/14
* @Param nil
* @Return *Blockchain
**/
func NewBlockChain() *Blockchain {
logger := log.NewLogfmtLogger(os.Stdout)
var tip []byte
//打开bolt文件
//如果该文件不存在,则创建以dbFile为名字的".db"文件
db, err := bolt.Open(dbFile, 0600, nil)
if err != nil {
logger.Log("open", err)
}
//fmt.Println("bolt数据库连接成功")
//BoltDB文件的两种类型事务操作之一 update-->读写事务
err = db.Update(func(tx *bolt.Tx) error {
//获取存储区块的bucket
//Bucket retrieves a bucket by name. Returns nil if the bucket does not exist
b := tx.Bucket([]byte(blocksBucket))
if b == nil {
genesisBlock := CreateGenesisBlock("Genesis Block")
b, err := tx.CreateBucket([]byte(blocksBucket))
if err != nil {
logger.Log("newCreateBucket", err)
} else {
//func (b *Bucket) Put(key []byte, value []byte) error
//这里以需要存入的区块的Hash为key,整个区块序列化后的字节数组作为value存入bucket中
err = b.Put(genesisBlock.Hash, genesisBlock.SerializeBlock())
if err != nil {
logger.Log("newPutBlockHash", err)
} else {
//然后将此存入的区块作为最新的区块.以l为key,该区块Hash为value存入bucket中
err = b.Put([]byte("l"), genesisBlock.Hash)
if err != nil {
logger.Log("newPutLHash", err)
} else {
//更新tip
tip = genesisBlock.Hash
}
}
}
} else {
tip = b.Get([]byte("l"))
}
return nil
})
bc := &Blockchain{tip, db}
return bc
}
/**
* @Title AddBlock
* @Description //在创建好的区块链中添加区块
* @Author Cofeesy 11:42 2022/8/14
* @Param height, int64, data string, prevHash []byte
* @Return
**/
func (bc *Blockchain) AddBlock(data string) {
logger := log.NewLogfmtLogger(os.Stdout)
//用来接收最后一个区块的Hash值
var lastHash []byte
//BoltDB文件的两种类型事务操作之一 View:只读业务
if err := bc.DB.View(func(tx *bolt.Tx) error {
//获取bucket
b := tx.Bucket([]byte(blocksBucket))
//获取bucket中的l键对应的值
lastHash = b.Get([]byte("l"))
return nil
}); err != nil {
logger.Log("view", err)
}
newBlock := NewBlock(lastHash, data)
if err := bc.DB.Update(func(tx *bolt.Tx) error {
b := tx.Bucket([]byte(blocksBucket))
if errPut := b.Put(newBlock.Hash, newBlock.SerializeBlock()); errPut != nil {
logger.Log("addPutBlockHash", errPut)
} else {
if errPut = b.Put([]byte("l"), newBlock.Hash); errPut != nil {
logger.Log("addPutLHash", errPut)
} else {
bc.Tip = newBlock.Hash
}
}
return nil
}); err != nil {
logger.Log("addUpdate", err)
}
}