#!/usr/bin/env python # encoding: utf-8 ''' 我们要创建一个 Blockchain 类 ,他的构造函数创建了一个初始化的空列表(要存储我们的区块链),并且另一个存储交易。 我们的 Blockchain 类负责管理链式数据,它会存储交易并且还有添加新的区块到链式数据的Method。 ''' import hashlib import json import time from textwrap import dedent from uuid import uuid4 from flask import Flask,jsonify,request class Blockchain(object): def __init__(self): self.chain=[] self.current_transations=[] ## 创建创世区块,源区块,第一个区块前面没了。 self.new_block(previous_hash=1,proof=100) def new_block(self,previous_hash,proof): ''' 创建一个新的区块到区块链 :param previous_hash:前一个区块的hash值 :param proof: 由工作量证明(PoW)算法生成的证明 :return: 返回新的区块 ''' block={ 'index':len(self.chain)+1, 'timestamp':time.time(), 'transactions':self.current_transations, 'proof':proof, 'previous_hash':previous_hash or self.hash(self.chain[-1]) } #重置当前交易记录 self.current_transations=[] self.chain.append(block) return block def new_transation(self,sender,recipient,amount): # Adds a new transaction to the list of transactions ''' 创建一笔新的交易到下一个被挖掘的区块中 :param sender: 发送者 :param recipient: 接收方 :param amount: 发送数量 :return: 返回将被添加到的区块的索引 ''' self.current_transations.append({ 'sender':sender, 'recipient':recipient, 'amount':amount, }) return self.last_block['index']+1 @staticmethod def hash(block): ''' 给一个区块生成SHA-256值 :param block: Block :return: 返回hash值 ''' #我们相比下确保这个字典(区块)是经过排序的,否则我们将会得到不一致的散列 block_string=json.dumps(block,sort_keys=True).encode() return hashlib.sha256(block_string).hexdigest() #@property装饰器就是负责把一个方法变成属性调用的 @property def last_block(self): # Returns the last Block in the chain return self.chain[-1] #每个块都有一个 索引,一个 时间戳(Unix时间戳),一个事务列表, 一个 校验(稍后详述) 和 前一个块的散列 。 """ 实现工作量证明 让我们来实现一个相似 PoW 算法。规则类似上面的例子: 找到一个数字 P ,使得它与前一个区块的 proof 拼接成的字符串的 Hash 值以 4 个零开头。 衡量算法复杂度的办法是修改零开头的个数。使用 4 个来用于演示,你会发现多一个零都会大大增加计算出结果所需的时间。 """ def proof_of_work(self,last_proof): ''' :param last_proof: :return: ''' proof=0 while self.valid_proof(last_proof,proof) is False: proof+=1 return proof @staticmethod def valid_proof(last_proof,proof): ''' :param last_proof: 上一个pow算法 :param proof: 现在的pow算法 :return: ''' guess=f'{last_proof}{proof}'.encode() guess_hash=hashlib.sha256(guess).hexdigest() return guess_hash[:4]=="0000" app=Flask(__name__) node_identity=str(uuid4()).replace('-','') blockchain=Blockchain() @app.route('/') @app.route('/mine',methods=['GET']) def mine(): # 挖矿 # 挖矿正是神奇所在,它很简单,做了一下三件事: # # 计算工作量证明 PoW # 通过新增一个交易授予矿工(自己)一个币 # 构造新区块并将其添加到链中 # We run the proof of work algorithm to get the next proof... last_block=blockchain.last_block last_proof=last_block['proof'] proof=blockchain.proof_of_work(last_proof) # We must receive a reward for finding the proof. # The sender is "0" to signify that this node has mined a new coin. blockchain.new_transation( sender="0", recipient=node_identity, amount=1, ) # Forge the new Block by adding it to the chain previous_hash=blockchain.hash(last_block) block=blockchain.new_block(proof,previous_hash) response={ 'message': "New Block Forged", 'index': block['index'], 'transactions': block['transactions'], 'proof': block['proof'], 'previous_hash': block['previous_hash'], } return jsonify(response),200 @app.route('/transactions/new',methods=['POST']) def new_transaction(): ''' 我们将添加一个新的事务处理 :return: ''' values=request.get_json() required=['sender','recipient','amount'] if not all(k in values for k in required): return "Missing values", 400 #创建一个新的事务 index=blockchain.new_transation(values['sender'],values['recipient'],values['amount']) response={'message':f'在区块上添加事务{index}'} return jsonify(response),201 @app.route('/chain',methods=['GET']) def full_chain(): a_chain=blockchain.chain response={ 'chain':a_chain, 'length':len(a_chain), } return jsonify(response),200 if __name__=="__main__": app.run(debug=True) # 注意交易的接收者是我们自己的服务器节点,我们做的大部分工作都只是围绕 Blockchain 类方法进行交互。 # 到此,我们的区块链就算完成了,我们来实际运行下.