通过写一个区块链来学习区块链 101

区块链齐步走

想要理解区块链如何工作最好的方式就是制作一个啦。

Learn Blockchains by Building One
加密货币与区块链(三):什么是信任

Keywords

ruby, blockchain, consensus

A

作为一个匪菜大军中的一员,不了解区块链是不能称为一个合格的匪菜的。空气币的火热,让更多的匪菜充满了渴望,似乎我们也需要了解一些在这个背后的基础。了解这个其实并不是很简单,因为更多的匪菜喜欢看到的是绿色的涨幅,而不是背后的技术,你可以在得(不)道上面找到很多奇怪的Talk,但是你的确得不到。
我喜欢边做边学,看完下面的例子,我相信你能成为一颗不一样的匪菜。

B

区块链(Blockchain),顾名思义就是由块组成的链,每一个块就是一些数据,然后通过一种手法把这个块连接起来,就是用一个哈希函数 H,把block B[i]的哈希值 H(B[i]) 包含在下一个 block B[i+1] 里。H 具有单向性,也就是知道 B 就很容易算出 H(B),但是反过来如果只知道 H(B) 的值很难构造出一个满足条件的 B。当然啦,这个其实就是一个链表,POC。这样做的结果就意味着如果其中任何一块被修改了。而因为 H(B0) 是 B1 的一部分,所以导致 H(B1) 也要跟着变。如果有人要修改记录在这个链上的数据,就需要修改后面所有的块。这个就叫做Merkle List。如果你用过Gayhub,那么你应该也知道,Git存储的方式就是基于Merkle List。

C

在你开始之前,我和你们说这篇教程使用的是Ruby语言写的。这里用了一些很简单的库来帮助我们可以做一个简单的Web Application,cuba, faraday。这里就不多做解释了。

STEP 1

在开始前,你可以在这里看到源代码传送门

我们在这里创建一个Blockchain的Blueprint

class Blockchain
end

Emmmm, that was a joke.

class Blockchain

  class << self
    def hash blk
      
    end
  end
  def initialize
  end

  def new_block
  end

  def new_transacction
  end

  def last_block
  end
end

我们的Blockchain是用来对链初始化,然后添加一些常用的操作的,new_block, new_transaction, hash等。

那么一个Block应该是什么样子的呢?

block = {
    'index': 1,
    'timestamp': 1506057125,
    'transactions': [
        {
            'sender': "8527147fe1f5426f9dd545de4b27ee00",
            'recipient': "a77f5cdfa2934df3954a5c7c7da5df1f",
            'amount': 5,
        }
    ],
    'proof': 324984774000,
    'previous_hash': "2cf24dba5fb0a30e26e83b2ac5b9e29e1b161e5c1fa7425e73043362938b9824"
}

接下来我们需要创建新的块了,在我们的Blockchain初始化的时候,我们需要给他一个创世块,一个没有祖先的块,同时我们也需要给创世块增加一个 proof ,这是挖矿的结果,我稍后再说啦。

我们现在需要了解什么是 PoW (Proof of Work),顾名思义就是新的区块是如何产生或者如何被挖出来的,它存在的目的就是发现能够解决一个问题的数字,这个数字需要具备两个属性,难找和易验证。

我们举一个简单的例子

from hashlib import sha256
x = 5
y = 0  # We don't know what y should be yet...
while sha256(f'{x*y}'.encode()).hexdigest()[-1] != "0":
    y += 1
print(f'The solution is y = {y}')

那么结果就是21,在比特币中,PoW的算法叫做Hashcash,和上面的例子是差不多的,矿工们算出结果之后是会被奖赏的,矿工会在一个交易中收到一个币。

PoW算法是很简单的,那么我们现在的题目就是:
找到一个数字p,使得hash(pp')的结果包含是由4个0开头的。这里p代表之前的proof,p'是新的proof

...

def PoW(last_proof)
  # proof of work algorithm (PoW)
  proof = 0
  while valid_proof?(last_proof, proof) == false
    proof += 1
  end
  proof
end
...

private
def valid_proof?(last_proof, proof)
  Digest::SHA256.hexdigest("#{last_proof}#{proof}")[0..3] == "0000"
end

如果你需要修改算法的难度,那么你只需要修改以0开头的个数就可以了。

STEP 2

我们这里用 cuba 做一个很小的 web 服务,它主要包含了三个功能

  • POST /transactions/new 生成一笔新的交易
  • GET /mine 告诉服务器产生一个新的块
  • GET /chain 把当前链返回

node_identifier = SecureRandom.hex(20)
blockchain = Blockchain.new


Cuba.define do
  on get do
    on 'mine' do
      last_block = blockchain.last_block
      last_proof = last_block[:proof]
      proof = blockchain.PoW(last_proof)

      blockchain.new_transaction("0", node_identifier.to_s, 1)

      previous_hash = Blockchain.hash(last_block)
      blk = blockchain.new_block(proof, previous_hash)

      data = {
        message: 'new block forged',
        index: blk[:index],
        transactions: blk[:transactions],
        proof: blk[:proof],
        previous_hash: blk[:previous_hash]
      }
      as_json {{ data: data }}
    end

    on 'chain' do
      as_json do
        {
          data: {
            chain: blockchain.chain,
            length: blockchain.chain.count
          }
        }
      end
    end

  end

  on post do
    on 'transactions/new' do
      on param('sender'), param('recipient'), param('amount') do |sender, recipient, amount|
        index = blockchain.new_transaction(sender,recipient, amount.to_f)

        as_json {{ data: "transaction will be added to block #{index}"}}
      end

      on true do
        as_json 400 do
          { error: 'missing params'}
        end
      end
    end

  end
end

接下来就可以跑一跑我们的简单的服务器啦

thin start -p 3000

你可以使用 Postman 或者 curl 来调用我们的服务。

STEP 3

共识,这个很重要,在分布式系统中,你需要保证数据的一致性,所以你需要知道我们需要通过一种什么样的算法来保证我们始终指向一条链。

  • POST /nodes/register 我们把当前网络的节点都存到一个地方
  • GET /nodes/resolve 这个地方用来解决冲突
def valid_chain?(chain)
  last_block = chain[0]
  current_index = 1

  while current_index < chain.size
    block = chain[current_index]
    puts "count 1"
    # Check that the hash of the block is correct
    if block[:previous_hash] != Blockchain.hash(last_block)
      return false
    end

    # Check that the Proof of Work is correct
    if !valid_proof?(last_block[:proof], block[:proof])
      return false
    end

    last_block = block
    current_index += 1
  end
  return true

end
def resolve_conflicts
  new_chain = nil
  # Only looking for chains longer than this one
  max_length = @chain.count
  aval = @nodes.delete @current_node
  aval.each do |node|
    conn = Faraday.new(url: "http://#{node}/chain")

    res = conn.get do |conn_get|
      conn_get.options.open_timeout = 15
      conn_get.options.timeout = 15
    end
    if res.status == 200
      content = JSON.parse(res.body, symbolize_names: true)
      length = content[:data][:length]
      chain = content[:data][:chain]
      ap "node #{node} len #{length > max_length} valid_chain #{valid_chain?(chain)}"
      if length > max_length && valid_chain?(chain)
        max_length = length
        new_chain = chain
      end
    end
  end

  if new_chain
    puts "found new chain here"
    @chain = new_chain
    return true
  end
  return false
end

# in server.rb

on 'nodes/resolve' do

  blockchain.current_node = "#{env["SERVER_NAME"]}:#{env["SERVER_PORT"]}"
  resolved = blockchain.resolve_conflicts
  if resolved
    data = {
      message: "our chain was replaced",
      new_chain: blockchain.chain
    }
  else
    data = {
      message: "our chain was authorized",
      new_chain: blockchain.chain
    }
  end

  as_json {{ data: data }}
end

valid_chain? 用来判断这个链是否正确的,可以看到 resolve_conflicts 里面主要用到的判断就是链长和是否是一条合格的链,如果更长并且合法,那么当前的链就替换为该链。不过这里值得注意的是,你需要把nodes中的自己的节点移除掉,这样其实是可以提高速度,并可以解决一个问题,单线程的机器中,你不能call自己。(会有一个OpenTimeOUt的错误,总是这里你把自己本身的节点去掉就可以了。)

到这里,你就可以用另一台机器或者使用不同的端口来模拟多个节点,我这边 Rakefile 里面默认是通过单机不同端口来模拟多个节点。

完整的代码请参考传送门

你可以开始和小伙伴们玩耍这个蠢蠢的链了

希望这个东西能够给你一些启发。

当然啦,最后送上一句话。

如果你认为技术能解决安全问题,那么你既不懂安全也不懂技术。 :)

有很多关于区块链的文章都说「区块链解决的核心问题是信任问题」,但是我没有看到有人回答了关键的问题:到底什么是信任?什么是所谓「信任问题」,它存不存在?什么算是「解决了信任问题」?事实上如果在 Google 上搜一下这句话,会找到大量的复制粘贴和人云亦云。Brice Schneier 书里那句话改一改也是适用的,如果有人认为技术能解决信任问题,那么他恐怕既不懂信任也不懂技术。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容