Posted in Python onJuly 03, 2019
接着上一篇交易记录整合交易类,这里描述区块的开发。
首先我们要明白一个区块,需要的内容,包括交易记录集合,时间戳,哈希,上一个区块的哈希。明白了这个,下面就容易代码开发了。
import datetime import hashlib from Message import DaDaMessage, InvalidMessage from Transaction import Transaction class Block: #不定参数,*args,区块集合 def __init__(self, *args): self.messagelist = [] # 存储多个交易记录 self.timestamp = None # 存储多个记录最终锁定时间 self.hash = None self.preHash = None if args: for arg in args: self.add_message(arg) def add_message(self, message): # 添加 交易信息 # 区分第一条与后面多条,是否需要链接 if len(self.messagelist) > 0: message.link(self.messagelist[-1]) # 链接最后一个 message.seal() # 密封 message.validate() # 校验 self.messagelist.append(message) def link(self, block): # 区块链接 self.preHash = block.hash def seal(self): # 密封当前hash self.timestamp = datetime.datetime.now() self.hash = self.hash_block() def hash_block(self): # 密封上一块哈希,时间戳,交易记录的最后一个 return hashlib.sha512( (str(self.timestamp) + str(self.preHash) + str(self.messagelist[-1].hash)).encode("utf-8")).hexdigest() def validate(self): # 校验 for i, message in enumerate(self.messagelist): # 每个交易记录校验一下 message.validate() #每一条交易记录校验 if i > 0 and message.prev_hash != self.messagelist[i - 1].hash: raise InvalidBlock("无效的block,交易记录被修改为在第{}条记录".format(i)+str(self)) return str(self) + "Ok" def __repr__(self): return "money block= hash:{},prehash:{},len:{},time:{}".format(self.hash, self.preHash, len(self.messagelist), self.timestamp)
自定义异常:
class InvalidBlock(Exception): def __init__(self,*args,**kwargs): super(Exception,self).__init__(*args,**kwargs)
编写测试模块:
if __name__=="__main__": t1 = Transaction("yicheng", "ddd1", 100) t2 = Transaction("yicheng", "ddd2", 200) t3 = Transaction("yicheng", "ddd3", 300) t4 = Transaction("yicheng", "ddd4", 400) m1 = DaDaMessage(t1) m2 = DaDaMessage(t2) m3 = DaDaMessage(t3) m4 = DaDaMessage(t4) try: block1 = Block(m1, m2, m3) block1.seal() #测试篡改数据 #m3.data = "你妹的直播" #block1.messagelist[0] = m3 print(block1.validate()) except InvalidMessage as e: #交易记录被修改 print(e) except InvalidBlock as e: #区块被修改 print(e)
测试结果如下,为了打印需要,我改成了md5格式下的结果:
篡改区块信息的结果,可能结果不一样,因为修改的内容不一样,报的错误也不一样:
至此,已经完成了:交易记录,区块的开发,现在进行区块链的开发就比较容易了。实现代码如下:
from Block import InvalidBlock, Block from Message import InvalidMessage, DaDaMessage from Transaction import Transaction # 区块链 class Dada_BlockCoin: def __init__(self): self.blocklist = [] # 装载所有区块 def validate(self):#校验所有区块 for i, block in enumerate(self.blocklist): try: block.validate() except InvalidBlockCoin as e: raise InvalidBlockCoin("区块校验错误,区块索引{}".format(i)) def add_block(self, block): # 增加区块 if len(self.blocklist) > 0: block.link(self.blocklist[-1]) #连接区块 block.seal()#密封 block.validate()#校验 self.blocklist.append(block)#添加到区块链中 def __repr__(self): return "Dada_BlockCoin:{}".format(len(self.blocklist))
自定义异常:
class InvalidBlockCoin(Exception): def __init__(self, *args, **kwargs): super(Exception, self).__init__(*args, **kwargs)
编写测试模块:
if __name__ == "__main__": t1 = Transaction("yicheng", "ddd1", 100) t2 = Transaction("yicheng", "ddd2", 200) t3 = Transaction("yicheng", "ddd3", 300) t4 = Transaction("yicheng", "ddd4", 400) t5 = Transaction("yicheng", "ddd5", 500) t6 = Transaction("yicheng", "ddd6", 600) m1 = DaDaMessage(t1) m2 = DaDaMessage(t2) m3 = DaDaMessage(t3) m4 = DaDaMessage(t4) m5 = DaDaMessage(t5) m6 = DaDaMessage(t6) try: yin1 = Block(m1, m2) yin1.seal() yin2 = Block(m3, m4) yin2.seal() yin3 = Block(m5, m6) yin3.seal() # 篡改区块 #yin3.messagelist.append(m1) coin = Dada_BlockCoin() # 区块链 coin.add_block(yin1) coin.add_block(yin2) coin.add_block(yin3) coin.validate() print(coin) except InvalidMessage as e: print(e) except InvalidBlock as e: print(e) except InvalidBlockCoin as e: print(e)
测试结果如下:
篡改区块链,测试模块区块链的内容,可以任意篡改,测试结果如下:
这里已经完成了数据层的部分开发,其余部分后续会完善。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。
python区块及区块链的开发详解
- Author -
charles_lun声明:登载此文出于传递更多信息之目的,并不意味着赞同其观点或证实其描述。
Reply on: @reply_date@
@reply_contents@