Python 블록 체인-빠른 가이드

블록 체인 튜토리얼에서 우리는 블록 체인 이론에 대해 자세히 배웠습니다. 블록 체인은 세계에서 가장 인기있는 디지털 통화 비트 코인의 기본 구성 요소입니다. 이 튜토리얼은 블록 체인 아키텍처를 완전히 설명하는 비트 코인의 복잡성을 깊이 다루었습니다. 다음 단계는 자체 블록 체인을 구축하는 것입니다.

Satoshi Nakamoto는 Bitcoin이라는 세계 최초의 가상 화폐를 만들었습니다. 비트 코인의 성공을보고 다른 많은 사람들이 자체 가상 화폐를 만들었습니다. 몇 가지 예를 들자면 Litecoin, Zcash 등이 있습니다.

이제 자신의 통화를 시작하고 싶을 수도 있습니다. 이것을 TPCoin (TutorialsPoint Coin)이라고합니다. TPCoin을 다루는 모든 거래를 기록하는 블록 체인을 작성합니다. TPCoin은 피자, 버거, 샐러드 등을 구매하는 데 사용할 수 있습니다. 네트워크에 가입하여 TPCoin을 서비스 제공을위한 통화로 받기 시작하는 다른 서비스 제공 업체가있을 수 있습니다. 가능성은 무한합니다.

이 튜토리얼에서는 이러한 시스템을 구축하고 시장에서 자신의 디지털 통화를 출시하는 방법을 이해해 보겠습니다.

블록 체인 프로젝트 개발에 관련된 구성 요소

전체 블록 체인 프로젝트 개발은 세 가지 주요 구성 요소로 구성됩니다.

  • Client
  • Miners
  • Blockchain

고객

고객은 다른 공급 업체로부터 상품을 구매하는 사람입니다. 고객 자신이 공급 업체가되어 그가 공급하는 상품에 대해 다른 사람으로부터 돈을받을 수 있습니다. 여기서는 클라이언트가 TPCoins의 공급자이자 수령자 일 수 있다고 가정합니다. 따라서 우리는 돈을 보내고받을 수있는 클라이언트 클래스를 코드에 만들 것입니다.

갱부

채굴자는 트랜잭션 풀에서 트랜잭션을 선택하여 블록으로 조립하는 사람입니다. 채굴자는 채굴 보상을 받기 위해 유효한 작업 증명을 제공해야합니다. 광부가 수수료로 모은 모든 돈은 그가 보관할 것입니다. 그는 위에서 설명한 클라이언트가하는 방식대로 네트워크에 등록 된 다른 공급 업체로부터 상품이나 서비스를 구매하는 데 그 돈을 사용할 수 있습니다.

블록 체인

마지막으로 블록 체인은 모든 채굴 된 블록을 시간순으로 연결하는 데이터 구조입니다. 이 사슬은 불변이므로 성질이 없습니다.

새 Jupyter 노트북의 각 단계에 표시된 코드를 입력하여이 자습서를 따를 수 있습니다. 양자 택일로, 당신은에서 전체 Jupyter 노트북 다운로드 할 수 있습니다 www.anaconda.com을 .

다음 장에서는 블록 체인 시스템을 사용하는 클라이언트를 개발할 것입니다.

클라이언트는 TPCoin을 보유하고 자신을 포함하여 네트워크상의 다른 공급 업체의 상품 / 서비스에 대해이를 거래하는 사람입니다. 우리는Client이 목적을 위해 클래스. 클라이언트에 대한 전역 고유 ID를 생성하기 위해 PKI (공개 키 인프라)를 사용합니다. 이 장에서는 이에 대해 자세히 설명하겠습니다.

클라이언트는 지갑에서 다른 알려진 사람에게 돈을 보낼 수 있어야합니다. 마찬가지로 클라이언트는 제 3 자로부터 돈을받을 수 있어야합니다. 돈을 지출하기 위해 클라이언트는 송금인의 이름과 지불 할 금액을 지정하는 트랜잭션을 생성합니다. 돈을 받기 위해 고객은 자신의 신원을 제 3 자 (기본적으로 송금인)에게 제공합니다. 우리는 고객이 지갑에 보유한 잔액을 저장하지 않습니다. 거래 중에 고객이 지불을 할 수있는 충분한 잔액이 있는지 확인하기 위해 실제 잔액을 계산합니다.

개발하려면 Client클래스와 프로젝트의 나머지 코드에 대해 많은 Python 라이브러리를 가져와야합니다. 이들은 아래에 나열되어 있습니다-

# import libraries
import hashlib
import random
import string
import json
import binascii
import numpy as np
import pandas as pd
import pylab as pl
import logging
import datetime
import collections

위의 표준 라이브러리 외에도 트랜잭션에 서명하고 객체의 해시를 만드는 등의 작업을 수행합니다.이를 위해 다음 라이브러리를 가져와야합니다.

# following imports are required by PKI
import Crypto
import Crypto.Random
from Crypto.Hash import SHA
from Crypto.PublicKey import RSA
from Crypto.Signature import PKCS1_v1_5

다음 장에서는 클라이언트 클래스에 대해 이야기하겠습니다.

그만큼 Client 클래스는 privatepublic 기본 제공 Python을 사용하여 키 RSA연산. 관심있는 독자는this tutorialRSA 구현을 위해. 객체 초기화 중에 개인 및 공개 키를 만들고 해당 값을 인스턴스 변수에 저장합니다.

self._private_key = RSA.generate(1024, random)
self._public_key = self._private_key.publickey()

개인 키를 분실해서는 안됩니다. 기록 보관을 위해 생성 된 개인 키를 안전한 외부 저장소에 복사하거나 단순히 종이에 ASCII 표현을 적어 둘 수 있습니다.

생성 public키는 클라이언트의 ID로 사용됩니다. 이를 위해 우리는identity 공개 키의 HEX 표현을 반환합니다.

@property
   def identity(self):
      return
binascii.hexlify(self._public_key.exportKey(format='DER'))
.decode('ascii')

그만큼 identity각 클라이언트에 고유하며 공개적으로 사용할 수 있습니다. 누구나 이것을 사용하여 가상 화폐를 보낼 수 있습니다identity 지갑에 추가됩니다.

에 대한 전체 코드 Client 클래스는 여기에 표시됩니다-

class Client:
   def __init__(self):
      random = Crypto.Random.new().read
      self._private_key = RSA.generate(1024, random)
      self._public_key = self._private_key.publickey()
      self._signer = PKCS1_v1_5.new(self._private_key)

   @property
   def identity(self):
      return
binascii.hexlify(self._public_key.exportKey(format='DER')).decode('ascii')

테스트 클라이언트

이제 사용 방법을 설명하는 코드를 작성합니다. Client 클래스-

Dinesh = Client()
print (Dinesh.identity)

위의 코드는 다음 인스턴스를 만듭니다. Client 변수에 할당합니다. Dinesh. 공개 키를 인쇄합니다.Dinesh 그것을 호출하여 identity방법. 출력은 다음과 같습니다.

30819f300d06092a864886f70d010101050003818d0030818902818100b547fafceeb131e07
0166a6b23fec473cce22c3f55c35ce535b31d4c74754fecd820aa94c1166643a49ea5f49f72
3181ff943eb3fdc5b2cb2db12d21c06c880ccf493e14dd3e93f3a9e175325790004954c34d3
c7bc2ccc9f0eb5332014937f9e49bca9b7856d351a553d9812367dc8f2ac734992a4e6a6ff6
6f347bd411d07f0203010001

이제 다음 장에서 트랜잭션을 생성하겠습니다.

이 장에서는 Transaction클라이언트가 누군가에게 돈을 보낼 수 있도록 클래스. 고객은 송금인 또는 수취인이 될 수 있습니다. 돈을 받고 싶을 때 다른 송금인이 거래를 생성하고public그것의 주소. 트랜잭션 클래스의 초기화를 다음과 같이 정의합니다.

def __init__(self, sender, recipient, value):
   self.sender = sender
   self.recipient = recipient
   self.value = value
   self.time = datetime.datetime.now()

그만큼 init 메소드는 세 개의 매개 변수를 취합니다-발신자의 public 키, 수신자의 public키 및 보낼 금액. 다른 메소드에서 사용할 수 있도록 인스턴스 변수에 저장됩니다. 또한 트랜잭션 시간을 저장하기위한 변수를 하나 더 만듭니다.

다음으로, 우리는 to_dict위에서 언급 한 네 가지 인스턴스 변수를 모두 사전 객체에 결합합니다. 이것은 단지 하나의 변수를 통해 접근 가능한 전체 거래 정보를 넣는 것입니다.

이전 튜토리얼에서 알다시피 블록 체인의 첫 번째 블록은 Genesis블록. Genesis 블록에는 블록 체인 작성자가 시작한 첫 번째 트랜잭션이 포함됩니다. 이 사람의 신원은 비트 코인의 경우처럼 비밀로 유지 될 수 있습니다. 따라서이 첫 번째 트랜잭션이 생성 될 때 생성자는 자신의 신원을 다음과 같이 보낼 수 있습니다.Genesis. 따라서 사전을 만드는 동안 보낸 사람이Genesis만약 그렇다면 우리는 단순히 식별 변수에 문자열 값을 할당합니다. 그렇지 않으면 보낸 사람의 신원을identity 변하기 쉬운.

if self.sender == "Genesis":
   identity = "Genesis"
else:
   identity = self.sender.identity

다음 코드 줄을 사용하여 사전을 구성합니다.

return collections.OrderedDict({
   'sender': identity,
   'recipient': self.recipient,
   'value': self.value,
   'time' : self.time})

에 대한 전체 코드 to_dict 방법은 다음과 같습니다-

def to_dict(self):
   if self.sender == "Genesis":
      identity = "Genesis"
   else:
      identity = self.sender.identity

   return collections.OrderedDict({
      'sender': identity,
      'recipient': self.recipient,
      'value': self.value,
      'time' : self.time})

마지막으로 보낸 사람의 개인 키를 사용하여이 사전 개체에 서명합니다. 이전과 마찬가지로 SHA 알고리즘과 함께 기본 제공 PKI를 사용합니다. 생성 된 서명은 인쇄 및 블록 체인에 저장하기위한 ASCII 표현을 얻기 위해 디코딩됩니다. 그만큼sign_transaction 메서드 코드는 여기에 나와 있습니다.

def sign_transaction(self):
   private_key = self.sender._private_key
   signer = PKCS1_v1_5.new(private_key)
   h = SHA.new(str(self.to_dict()).encode('utf8'))
   return binascii.hexlify(signer.sign(h)).decode('ascii')

이제 이것을 테스트하겠습니다. Transaction 수업.

트랜잭션 클래스 테스트

이를 위해 우리는 DineshRamesh. Dinesh는 5 개의 TPCoin을 Ramesh로 보냅니다. 먼저 Dinesh와 Ramesh라는 클라이언트를 만듭니다.

Dinesh = Client()
Ramesh = Client()

인스턴스화 할 때 Client 클래스, public and클라이언트에 고유 한 개인 키가 생성됩니다. Dinesh가 Ramesh에 지불을 보낼 때 클라이언트의 ID 속성을 사용하여 얻은 Ramesh의 공개 키가 필요합니다.

따라서 다음 코드를 사용하여 트랜잭션 인스턴스를 생성합니다.

t = Transaction(
   Dinesh,
   Ramesh.identity,
   5.0
)

첫 번째 매개 변수는 보낸 사람이고 두 번째 매개 변수는받는 사람의 공개 키이며 세 번째 매개 변수는 전송할 금액입니다. 그만큼sign_transaction 메서드는 트랜잭션을 노래하기 위해 첫 번째 매개 변수에서 보낸 사람의 개인 키를 검색합니다.

트랜잭션 개체가 생성 된 후에는 해당 개체를 호출하여 서명합니다. sign_transaction방법. 이 메서드는 생성 된 서명을 인쇄 가능한 형식으로 반환합니다. 다음 두 줄의 코드를 사용하여 서명을 생성하고 인쇄합니다.

signature = t.sign_transaction()
print (signature)

위의 코드를 실행하면 다음과 유사한 출력이 표시됩니다.

7c7e3c97629b218e9ec6e86b01f9abd8e361fd69e7d373c38420790b655b9abe3b575e343c7
13703ca1aee781acd7157a0624db3d57d7c2f1172730ee3f45af943338157f899965856f6b0
0e34db240b62673ad5a08c8e490f880b568efbc36035cae2e748f1d802d5e8e66298be826f5
c6363dc511222fb2416036ac04eb972

이제 클라이언트와 트랜잭션을 생성하는 기본 인프라가 준비되었으므로 이제 실제 상황에서와 같이 여러 클라이언트가 여러 트랜잭션을 수행하게됩니다.

다양한 클라이언트가 만든 트랜잭션은 시스템에 대기합니다. 채굴자는이 대기열에서 트랜잭션을 선택하여 블록에 추가합니다. 그런 다음 블록을 채굴하고 승리 한 채굴자는 블록을 블록 체인에 추가하여 자신을 위해 돈을 벌 수있는 특권을 갖게됩니다.

이 마이닝 프로세스는 나중에 블록 체인 생성에 대해 논의 할 때 설명 할 것입니다. 여러 트랜잭션에 대한 코드를 작성하기 전에 주어진 트랜잭션의 내용을 인쇄하는 작은 유틸리티 함수를 추가해 보겠습니다.

거래 표시

그만큼 display_transaction함수는 트랜잭션 유형의 단일 매개 변수를 허용합니다. 수신 된 트랜잭션 내의 사전 객체는 다음과 같은 임시 변수에 복사됩니다.dict 사전 키를 사용하여 다양한 값이 콘솔에 인쇄됩니다.

def display_transaction(transaction):
   #for transaction in transactions:
   dict = transaction.to_dict()
   print ("sender: " + dict['sender'])
   print ('-----')
   print ("recipient: " + dict['recipient'])
   print ('-----')
   print ("value: " + str(dict['value']))
   print ('-----')
   print ("time: " + str(dict['time']))
   print ('-----')

다음으로 트랜잭션 객체를 저장하기위한 트랜잭션 큐를 정의합니다.

거래 대기열

대기열을 생성하기 위해 글로벌 list 변수 호출 transactions 다음과 같이-

transactions = []

새로 생성 된 각 트랜잭션을이 큐에 추가하기 만하면됩니다. 간결성을 위해이 자습서에서는 대기열 관리 논리를 구현하지 않습니다.

여러 클라이언트 생성

이제 트랜잭션 생성을 시작합니다. 첫째, 우리는 다른 사람으로부터 다양한 서비스 또는 상품을 얻기 위해 서로에게 돈을 보낼 네 명의 클라이언트를 만들 것입니다.

Dinesh = Client()
Ramesh = Client()
Seema = Client()
Vijay = Client()

이 시점에서 우리는 Dinesh, Ramesh, Seema 및 Vijay라는 4 개의 클라이언트가 있습니다. 우리는 현재 이러한 각 클라이언트가 거래를 위해 지갑에 일부 TPCoin을 보유하고 있다고 가정합니다. 이러한 각 클라이언트의 ID는 이러한 개체의 ID 속성을 사용하여 지정됩니다.

첫 번째 거래 생성

이제 다음과 같이 첫 번째 트랜잭션을 시작합니다.

t1 = Transaction(
   Dinesh,
   Ramesh.identity,
   15.0
)

이 거래에서 Dinesh는 5 개의 TPCoin을 Ramesh로 보냅니다. 거래가 성공적으로 이루어 지려면 Dinesh가이 결제를 위해 지갑에 충분한 돈을 가지고 있는지 확인해야합니다. 시스템에서 TPCoin 유통을 시작하려면 제네시스 트랜잭션이 필요합니다. 이 제네시스 트랜잭션에 대한 트랜잭션 코드를 읽으면서 곧 작성할 것입니다.

Dinesh의 개인 키를 사용하여이 트랜잭션에 서명하고 다음과 같이 트랜잭션 대기열에 추가합니다.

t1.sign_transaction()
transactions.append(t1)

Dinesh가 만든 첫 번째 트랜잭션 이후, 위에서 생성 한 서로 다른 클라이언트간에 여러 트랜잭션을 더 생성합니다.

더 많은 거래 추가

이제 여러 트랜잭션을 생성하고 각 트랜잭션은 다른 당사자에게 몇 개의 TPCoin을 제공합니다. 누군가 돈을 쓸 때이 지갑에 충분한 잔액이 있는지 확인할 필요가 없습니다. 어쨌든 채굴자는 거래를 시작하는 동안 발신자가 보유한 잔액에 대해 각 거래를 검증 할 것입니다.

잔액이 부족한 경우 채굴자는이 거래를 유효하지 않은 것으로 표시하고이 거래를이 블록에 추가하지 않습니다.

다음 코드는 9 개의 트랜잭션을 생성하고 대기열에 추가합니다.

t2 = Transaction(
   Dinesh,
   Seema.identity,
   6.0
)
t2.sign_transaction()
transactions.append(t2)
t3 = Transaction(
   Ramesh,
   Vijay.identity,
   2.0
)
t3.sign_transaction()
transactions.append(t3)
t4 = Transaction(
   Seema,
   Ramesh.identity,
   4.0
)
t4.sign_transaction()
transactions.append(t4)
t5 = Transaction(
   Vijay,
   Seema.identity,
   7.0
)
t5.sign_transaction()
transactions.append(t5)
t6 = Transaction(
   Ramesh,
   Seema.identity,
   3.0
)
t6.sign_transaction()
transactions.append(t6)
t7 = Transaction(
   Seema,
   Dinesh.identity,
   8.0
)
t7.sign_transaction()
transactions.append(t7)
t8 = Transaction(
   Seema,
   Ramesh.identity,
   1.0
)
t8.sign_transaction()
transactions.append(t8)
t9 = Transaction(
   Vijay,
   Dinesh.identity,
   5.0
)
t9.sign_transaction()
transactions.append(t9)
t10 = Transaction(
   Vijay,
   Ramesh.identity,
   3.0
)
t10.sign_transaction()
transactions.append(t10)

위의 코드를 실행하면 채굴자가 블록을 생성 할 수 있도록 대기열에 10 개의 트랜잭션이 있습니다.

거래 덤핑

블록 체인 관리자는 주기적으로 트랜잭션 큐의 내용을 검토하고 싶을 수 있습니다. 이를 위해 다음을 사용할 수 있습니다.display_transaction앞서 개발 한 기능입니다. 대기열의 모든 트랜잭션을 덤프하려면 트랜잭션 목록을 반복하고 참조 된 각 트랜잭션에 대해display_transaction 여기에 표시된 기능-

for transaction in transactions:
   display_transaction (transaction)
   print ('--------------')

거래는 구분을 위해 점선으로 구분됩니다. 위의 코드를 실행하면 아래와 같이 트랜잭션 목록이 표시됩니다.

sender:
30819f300d06092a864886f70d010101050003818d0030818902818100bb064c99c49214
4a9f463480273aba93ac1db1f0da3cb9f3c1f9d058cf499fd8e54d244da0a8dd6ddd329e
c86794b04d773eb4841c9f935ea4d9ccc2821c7a1082d23b6c928d59863407f52fa05d8b
47e5157f8fe56c2ce3279c657f9c6a80500073b0be8093f748aef667c03e64f04f84d311
c4d866c12d79d3fc3034563dfb0203010001
-----
recipient:
30819f300d06092a864886f70d010101050003818d0030818902818100be93b516b28c6e
674abe7abdb11ce0fdf5bb728b75216b73f37a6432e4b402b3ad8139b8c0ba541a72c8ad
d126b6e1a1308fb98b727beb63c6060356bb177bb7d54b54dbe87aee7353d0a6baa93977
04de625d1836d3f42c7ee5683f6703259592cc24b09699376807f28fe0e00ff882974484
d805f874260dfc2d1627473b910203010001
-----
value: 15.0
-----
time: 2019-01-14 16:18:01.859915
-----
--------------
sender:
30819f300d06092a864886f70d010101050003818d0030818902818100bb064c99c49214
4a9f463480273aba93ac1db1f0da3cb9f3c1f9d058cf499fd8e54d244da0a8dd6ddd329e
c86794b04d773eb4841c9f935ea4d9ccc2821c7a1082d23b6c928d59863407f52fa05d8b
47e5157f8fe56c2ce3279c657f9c6a80500073b0be8093f748aef667c03e64f04f84d311
c4d866c12d79d3fc3034563dfb0203010001
-----
recipient:
30819f300d06092a864886f70d010101050003818d0030818902818100a070c82b34ae14
3cbe59b3a2afde7186e9d5bc274955d8112d87a00256a35369acc4d0edfe65e8f9dc93fb
d9ee74b9e7ea12334da38c8c9900e6ced1c4ce93f86e06611e656521a1eab561892b7db0
961b4f212d1fd5b5e49ae09cf8c603a068f9b723aa8a651032ff6f24e5de00387e4d0623
75799742a359b8f22c5362e5650203010001
-----
value: 6.0
-----
time: 2019-01-14 16:18:01.860966
-----
--------------
sender:
30819f300d06092a864886f70d010101050003818d0030818902818100be93b516b28c6e
674abe7abdb11ce0fdf5bb728b75216b73f37a6432e4b402b3ad8139b8c0ba541a72c8ad
d126b6e1a1308fb98b727beb63c6060356bb177bb7d54b54dbe87aee7353d0a6baa93977
04de625d1836d3f42c7ee5683f6703259592cc24b09699376807f28fe0e00ff882974484
d805f874260dfc2d1627473b910203010001
-----
recipient:
30819f300d06092a864886f70d010101050003818d0030818902818100cba097c0854876
f41338c62598c658f545182cfa4acebce147aedf328181f9c4930f14498fd03c0af6b0cc
e25be99452a81df4fa30a53eddbb7bb7b203adf8764a0ccd9db6913a576d68d642d8fd47
452590137869c25d9ff83d68ebe6d616056a8425b85b52e69715b8b85ae807b84638d8f0
0e321b65e4c33acaf6469e18e30203010001
-----
value: 2.0
-----
time: 2019-01-14 16:18:01.861958
-----
--------------

간결하게하기 위해 목록에서 처음 몇 개의 트랜잭션 만 인쇄했습니다. 위의 코드에서는이 목록에 추가되지 않은 제네시스 트랜잭션을 제외하고 첫 번째 트랜잭션부터 시작하는 모든 트랜잭션을 인쇄합니다. 트랜잭션이 주기적으로 블록에 추가되므로 일반적으로 아직 채굴되지 않은 트랜잭션 목록 만 보는 데 관심이 있습니다. 이 경우 적절한for 아직 채굴되지 않은 트랜잭션을 반복합니다.

지금까지 클라이언트를 생성하고, 클라이언트를 허용하고, 채굴 할 보류중인 트랜잭션의 대기열을 유지하는 방법을 배웠습니다. 이제이 튜토리얼에서 가장 중요한 부분은 블록 체인 자체를 만드는 것입니다. 다음 강의에서이를 배웁니다.

블록은 다양한 수의 트랜잭션으로 구성됩니다. 간단하게하기 위해 우리의 경우 블록이 고정 된 수의 트랜잭션으로 구성되어 있다고 가정합니다.이 경우에는 3 개입니다. 블록이이 세 가지 트랜잭션 목록을 저장해야하므로 다음과 같은 인스턴스 변수를 선언합니다.verified_transactions 다음과 같이-

self.verified_transactions = []

이 변수의 이름을 다음과 같이 지정했습니다. verified_transactions확인 된 유효한 트랜잭션 만 블록에 추가됨을 나타냅니다. 각 블록은 또한 이전 블록의 해시 값을 보유하므로 블록 체인이 변경되지 않습니다.

이전 해시를 저장하기 위해 다음과 같이 인스턴스 변수를 선언합니다.

self.previous_block_hash = ""

마지막으로, 우리는 Nonce 채굴 과정에서 채굴자가 생성 한 임시 값을 저장합니다.

self.Nonce = ""

의 전체 정의 Block 클래스는 다음과 같습니다-

class Block:
   def __init__(self):
      self.verified_transactions = []
      self.previous_block_hash = ""
      self.Nonce = ""

각 블록에는 이전 블록의 해시 값이 필요하므로 다음과 같은 전역 변수를 선언합니다. last_block_hash 다음과 같이-

last_block_hash = ""

이제 블록 체인에 첫 번째 블록을 생성하겠습니다.

우리는 TPCoins의 창시자가 처음에 알려진 클라이언트에게 500 개의 TPCoin을 제공한다고 가정합니다. Dinesh. 이를 위해 그는 먼저 Dinesh 인스턴스를 만듭니다.

Dinesh = Client()

그런 다음 제네시스 트랜잭션을 생성하고 500 개의 TPCoin을 Dinesh의 공개 주소로 보냅니다.

t0 = Transaction (
   "Genesis",
   Dinesh.identity,
   500.0
)

이제 우리는 Block 수업하고 불러 block0.

block0 = Block()

우리는 previous_block_hashNonce 인스턴스 변수 None, 이것이 우리 블록 체인에 저장되는 최초의 트랜잭션이기 때문입니다.

block0.previous_block_hash = None
Nonce = None

다음으로 위의 t0 트랜잭션을 verified_transactions 블록 내에서 유지되는 목록-

block0.verified_transactions.append (t0)

이 시점에서 블록은 완전히 초기화되었으며 블록 체인에 추가 할 준비가되었습니다. 이를 위해 블록 체인을 만들 것입니다. 블록을 블록 체인에 추가하기 전에 블록을 해시하고 그 값을 다음과 같은 전역 변수에 저장합니다.last_block_hash이전에 선언했습니다. 이 값은 블록의 다음 채굴자가 사용할 것입니다.

블록을 해싱하고 다이제스트 값을 저장하기 위해 다음 두 줄의 코딩을 사용합니다.

digest = hash (block0)
last_block_hash = digest

마지막으로 다음 장에서 볼 수 있듯이 블록 체인을 만듭니다.

블록 체인은 서로 연결된 블록 목록을 포함합니다. 전체 목록을 저장하기 위해 TPCoins라는 목록 변수를 생성합니다.

TPCoins = []

또한 다음과 같은 유틸리티 메소드를 작성합니다. dump_blockchain전체 블록 체인의 내용을 덤핑하기 위해. 먼저 블록 체인에 현재 얼마나 많은 블록이 있는지 알 수 있도록 블록 체인의 길이를 인쇄합니다.

def dump_blockchain (self):
   print ("Number of blocks in the chain: " + str(len (self)))

시간이 지남에 따라 블록 체인의 블록 수가 인쇄 할 때 엄청나게 높아질 것입니다. 따라서 블록 체인의 내용을 인쇄 할 때 조사 할 범위를 결정해야 할 수 있습니다. 아래 코드에서는 현재 데모에서 너무 많은 블록을 추가하지 않을 것이므로 전체 블록 체인을 인쇄했습니다.

체인을 반복하기 위해 우리는 for 다음과 같이 루프-

for x in range (len(TPCoins)):
   block_temp = TPCoins[x]

참조 된 각 블록은 다음과 같은 임시 변수에 복사됩니다. block_temp.

각 블록의 제목으로 블록 번호를 인쇄합니다. 숫자는 0으로 시작하고 첫 번째 블록은 0으로 번호가 매겨진 생성 블록입니다.

print ("block # " + str(x))

각 블록 내에서 우리는 다음과 같은 변수에 세 가지 트랜잭션 (제네시스 블록 제외) 목록을 저장했습니다. verified_transactions. 우리는이 목록을for 루프하고 검색된 각 항목에 대해 display_transaction 거래 내역을 표시하는 기능.

for transaction in block_temp.verified_transactions:
   display_transaction (transaction)

전체 함수 정의는 다음과 같습니다.

def dump_blockchain (self):
   print ("Number of blocks in the chain: " + str(len (self)))
   for x in range (len(TPCoins)):
      block_temp = TPCoins[x]
      print ("block # " + str(x))
      for transaction in block_temp.verified_transactions:
         display_transaction (transaction)
         print ('--------------')
      print ('=====================================')

여기에서는 코드의 적절한 지점에 구분 기호를 삽입하여 블록과 트랜잭션을 구분했습니다.

이제 블록을 저장하기위한 블록 체인을 만들었으므로 다음 작업은 블록을 만들어 블록 체인에 추가하는 것입니다. 이를 위해 이전 단계에서 이미 생성 한 제네시스 블록을 추가합니다.

블록 체인에 블록을 추가하려면 생성 된 블록을 TPCoins 명부.

TPCoins.append (block0)

시스템의 나머지 블록과 달리 제네시스 블록에는 TPCoins 시스템의 발신자가 시작한 트랜잭션이 하나만 포함되어 있습니다. 이제 글로벌 함수를 호출하여 블록 체인의 내용을 덤프합니다.dump_blockchain

dump_blockchain(TPCoins)

이 함수를 실행하면 다음과 같은 출력이 표시됩니다.

Number of blocks in the chain: 1
block # 0
sender: Genesis
-----
recipient:
30819f300d06092a864886f70d010101050003818d0030818902818100ed272b52ccb539
e2cd779c6cc10ed1dfadf5d97c6ab6de90ed0372b2655626fb79f62d0e01081c163b0864
cc68d426bbe9438e8566303bb77414d4bfcaa3468ab7febac099294de10273a816f7047d
4087b4bafa11f141544d48e2f10b842cab91faf33153900c7bf6c08c9e47a7df8aa7e60d
c9e0798fb2ba3484bbdad2e4430203010001
-----
value: 500.0
-----
time: 2019-01-14 16:18:02.042739
-----
--------------
=====================================

이 시점에서 블록 체인 시스템을 사용할 준비가되었습니다. 이제 관심있는 고객에게 채굴 기능을 제공하여 채굴자가 될 수 있도록 할 것입니다.

채굴을 가능하게하기 위해서는 채굴 기능을 개발해야합니다. 마이닝 기능은 주어진 메시지 문자열에 대한 다이제스트를 생성하고 작업 증명을 제공해야합니다. 이 장에서 이에 대해 논의하겠습니다.

메시지 다이제스트 기능

다음과 같은 유틸리티 함수를 작성합니다. sha256 주어진 메시지에 대한 다이제스트 생성-

def sha256(message):
return hashlib.sha256(message.encode('ascii')).hexdigest()

그만큼 sha256 함수는 message 매개 변수로 ASCII로 인코딩하고 16 진 다이제스트를 생성하고 값을 호출자에게 리턴합니다.

마이닝 기능

이제 우리는 mine자체 채굴 전략을 구현하는 기능. 이 경우 우리의 전략은 주어진 메시지에 주어진 숫자의 접두사가 붙은 해시를 생성하는 것입니다. 주어진 1의 수는 매개 변수로 지정됩니다.mine 난이도로 지정된 기능.

예를 들어, 난이도를 2로 지정하면 주어진 메시지에 생성 된 해시는 11xxxxxxxx와 같이 2 개의 1로 시작해야합니다. 난이도가 3이면 생성 된 해시는 111xxxxxxxx와 같이 3 개의 1로 시작해야합니다. 이러한 요구 사항을 감안할 때 아래 단계와 같이 마이닝 기능을 개발할 것입니다.

1 단계

마이닝 기능은 메시지와 난이도의 두 가지 매개 변수를 사용합니다.

def mine(message, difficulty=1):

2 단계

난이도는 1 이상이어야합니다. 다음 assert 문으로이를 보장합니다.

assert difficulty >= 1

3 단계

우리는 prefix 설정된 난이도를 사용하여 변수.

prefix = '1' * difficulty

난이도가 2이면 접두사는“11”이되고 난이도가 3이면 접두사는“111”이되는 식입니다. 이 접두사가 생성 된 메시지 다이제스트에 존재하는지 확인합니다. 메시지 자체를 요약하기 위해 다음 두 줄의 코드를 사용합니다.

for i in range(1000):
   digest = sha256(str(hash(message)) + str(i))

계속해서 새 번호를 추가합니다 i각 반복의 메시지 해시에 추가하고 결합 된 메시지에 대해 새 다이제스트를 생성합니다. 입력으로sha256 함수는 모든 반복에서 변경됩니다. digest가치도 변할 것입니다. 우리는 이것이digest 값이 이상 설정 됨 prefix.

if digest.startswith(prefix):

조건이 충족되면 당사는 for 루프 및 반환 digest 발신자에게 가치.

전체 mine 여기에 코드가 나와 있습니다.

def mine(message, difficulty=1):
   assert difficulty >= 1
   prefix = '1' * difficulty
   for i in range(1000):
      digest = sha256(str(hash(message)) + str(i))
      if digest.startswith(prefix):
         print ("after " + str(i) + " iterations found nonce: "+ digest)
      return digest

이해를 돕기 위해 print 함수에서 반환하기 전에 조건을 충족하는 데 걸린 다이제스트 값과 반복 횟수를 인쇄하는 문입니다.

마이닝 기능 테스트

마이닝 기능을 테스트하려면 다음 문을 실행하십시오.

mine ("test message", 2)

위의 코드를 실행하면 아래와 비슷한 출력을 볼 수 있습니다.

after 138 iterations found nonce:
11008a740eb2fa6bf8d55baecda42a41993ca65ce66b2d3889477e6bfad1484c

생성 된 다이제스트는 "11"로 시작합니다. 난이도를 3으로 변경하면 생성 된 다이제스트가 "111"로 시작하며 물론 더 많은 반복이 필요할 것입니다. 보시다시피, 더 많은 처리 능력을 가진 채굴자는 주어진 메시지를 더 일찍 채굴 할 수 있습니다. 그것이 광부가 수익을 얻기 위해 서로 경쟁하는 방법입니다.

이제 블록 체인에 더 많은 블록을 추가 할 준비가되었습니다. 다음 장에서 이것을 배우도록합시다.

각 채굴자는 이전에 생성 된 트랜잭션 풀에서 트랜잭션을 선택합니다. 이미 채굴 된 메시지 수를 추적하려면 전역 변수를 만들어야합니다.

last_transaction_index = 0

이제 첫 번째 채굴자가 블록 체인에 블록을 추가하게됩니다.

첫 번째 블록 추가

새 블록을 추가하려면 먼저 Block 수업.

block = Block()

대기열에서 상위 3 개의 트랜잭션을 선택합니다.

for i in range(3):
   temp_transaction = transactions[last_transaction_index]
   # validate transaction

거래를 블록에 추가하기 전에 채굴자는 거래의 유효성을 확인합니다. 트랜잭션 유효성은 보낸 사람의 공개 키를 사용하여 채굴자가 생성 한 해시와 보낸 사람이 제공 한 해시가 동일한 지 테스트하여 확인됩니다. 또한 광부는 발신자가 현재 거래에 대해 지불하기에 충분한 잔액이 있는지 확인합니다.

간결성을 위해이 기능은 튜토리얼에 포함되지 않았습니다. 거래가 확인 된 후verified_transactions 목록에 block 예.

block.verified_transactions.append (temp_transaction)

다음 채굴자가 대기열에서 후속 트랜잭션을 선택할 수 있도록 마지막 트랜잭션 인덱스를 증가시킵니다.

last_transaction_index += 1

정확히 3 개의 트랜잭션을 블록에 추가합니다. 이 작업이 완료되면 나머지 인스턴스 변수를 초기화합니다.Block수업. 먼저 마지막 블록의 해시를 추가합니다.

block.previous_block_hash = last_block_hash

다음으로 난이도가 2 인 블록을 채굴합니다.

block.Nonce = mine (block, 2)

첫 번째 매개 변수는 mine함수는 이진 객체입니다. 이제 전체 블록을 해시하고 여기에 다이제스트를 만듭니다.

digest = hash (block)

마지막으로 생성 된 블록을 블록 체인에 추가하고 전역 변수를 다시 초기화합니다. last_block_hash 다음 블록에서 사용하기 위해.

블록을 추가하는 전체 코드는 다음과 같습니다.

block = Block()
for i in range(3):
   temp_transaction = transactions[last_transaction_index]
   # validate transaction
   # if valid
   block.verified_transactions.append (temp_transaction)
   last_transaction_index += 1

block.previous_block_hash = last_block_hash
block.Nonce = mine (block, 2)
digest = hash (block)
TPCoins.append (block)
last_block_hash = digest

더 많은 블록 추가

이제 블록 체인에 두 개의 블록을 더 추가 할 것입니다. 다음 두 블록을 추가하는 코드는 다음과 같습니다.

# Miner 2 adds a block
block = Block()

for i in range(3):
   temp_transaction = transactions[last_transaction_index]
   # validate transaction
   # if valid
   block.verified_transactions.append (temp_transaction)
   last_transaction_index += 1
block.previous_block_hash = last_block_hash
block.Nonce = mine (block, 2)digest = hash (block)
TPCoins.append (block)last_block_hash = digest
# Miner 3 adds a block
block = Block()

for i in range(3):
   temp_transaction = transactions[last_transaction_index]
   #display_transaction (temp_transaction)
   # validate transaction
   # if valid
   block.verified_transactions.append (temp_transaction)
   last_transaction_index += 1

block.previous_block_hash = last_block_hash
block.Nonce = mine (block, 2)
digest = hash (block)

TPCoins.append (block)
last_block_hash = digest

이 두 블록을 추가하면 Nonce를 찾는 데 걸린 반복 횟수도 볼 수 있습니다. 이 시점에서 우리의 블록 체인은 제네시스 블록을 포함하여 총 4 개의 블록으로 구성됩니다.

전체 블록 체인 덤핑

다음 문장을 사용하여 전체 블록 체인의 내용을 확인할 수 있습니다.

dump_blockchain(TPCoins)

아래 표시된 것과 유사한 출력이 표시됩니다.

Number of blocks in the chain: 4
block # 0
sender: Genesis
-----
recipient:
30819f300d06092a864886f70d010101050003818d0030818902818100ed272b52ccb539e2cd779
c6cc10ed1dfadf5d97c6ab6de90ed0372b2655626fb79f62d0e01081c163b0864cc68d426bbe943
8e8566303bb77414d4bfcaa3468ab7febac099294de10273a816f7047d4087b4bafa11f141544d4
8e2f10b842cab91faf33153900c7bf6c08c9e47a7df8aa7e60dc9e0798fb2ba3484bbdad2e44302
03010001
-----
value: 500.0
-----
time: 2019-01-14 16:18:02.042739
-----
--------------
=====================================
block # 1
sender:
30819f300d06092a864886f70d010101050003818d0030818902818100bb064c99c492144a9f463
480273aba93ac1db1f0da3cb9f3c1f9d058cf499fd8e54d244da0a8dd6ddd329ec86794b04d773e
b4841c9f935ea4d9ccc2821c7a1082d23b6c928d59863407f52fa05d8b47e5157f8fe56c2ce3279
c657f9c6a80500073b0be8093f748aef667c03e64f04f84d311c4d866c12d79d3fc3034563dfb02
03010001
-----
recipient:
30819f300d06092a864886f70d010101050003818d0030818902818100be93b516b28c6e674abe7
abdb11ce0fdf5bb728b75216b73f37a6432e4b402b3ad8139b8c0ba541a72c8add126b6e1a1308f
b98b727beb63c6060356bb177bb7d54b54dbe87aee7353d0a6baa9397704de625d1836d3f42c7ee
5683f6703259592cc24b09699376807f28fe0e00ff882974484d805f874260dfc2d1627473b9102
03010001
-----
value: 15.0
-----
time: 2019-01-14 16:18:01.859915
-----
--------------
sender:
30819f300d06092a864886f70d010101050003818d0030818902818100bb064c99c492144a9f463
480273aba93ac1db1f0da3cb9f3c1f9d058cf499fd8e54d244da0a8dd6ddd329ec86794b04d773e
b4841c9f935ea4d9ccc2821c7a1082d23b6c928d59863407f52fa05d8b47e5157f8fe56c2ce3279
c657f9c6a80500073b0be8093f748aef667c03e64f04f84d311c4d866c12d79d3fc3034563dfb02
03010001
-----
recipient:
30819f300d06092a864886f70d010101050003818d0030818902818100a070c82b34ae143cbe59b
3a2afde7186e9d5bc274955d8112d87a00256a35369acc4d0edfe65e8f9dc93fbd9ee74b9e7ea12
334da38c8c9900e6ced1c4ce93f86e06611e656521a1eab561892b7db0961b4f212d1fd5b5e49ae
09cf8c603a068f9b723aa8a651032ff6f24e5de00387e4d062375799742a359b8f22c5362e56502
03010001
-----
value: 6.0
-----
time: 2019-01-14 16:18:01.860966
-----
--------------
sender:
30819f300d06092a864886f70d010101050003818d0030818902818100be93b516b28c6e674abe7
abdb11ce0fdf5bb728b75216b73f37a6432e4b402b3ad8139b8c0ba541a72c8add126b6e1a1308f
b98b727beb63c6060356bb177bb7d54b54dbe87aee7353d0a6baa9397704de625d1836d3f42c7ee
5683f6703259592cc24b09699376807f28fe0e00ff882974484d805f874260dfc2d1627473b9102
03010001
-----
recipient:
30819f300d06092a864886f70d010101050003818d0030818902818100cba097c0854876f41338c
62598c658f545182cfa4acebce147aedf328181f9c4930f14498fd03c0af6b0cce25be99452a81d
f4fa30a53eddbb7bb7b203adf8764a0ccd9db6913a576d68d642d8fd47452590137869c25d9ff83
d68ebe6d616056a8425b85b52e69715b8b85ae807b84638d8f00e321b65e4c33acaf6469e18e302
03010001
-----
value: 2.0
-----
time: 2019-01-14 16:18:01.861958
-----
--------------
=====================================
block # 2
sender:
30819f300d06092a864886f70d010101050003818d0030818902818100a070c82b34ae143cbe59b
3a2afde7186e9d5bc274955d8112d87a00256a35369acc4d0edfe65e8f9dc93fbd9ee74b9e7ea12
334da38c8c9900e6ced1c4ce93f86e06611e656521a1eab561892b7db0961b4f212d1fd5b5e49ae
09cf8c603a068f9b723aa8a651032ff6f24e5de00387e4d062375799742a359b8f22c5362e56502
03010001
-----
recipient:
30819f300d06092a864886f70d010101050003818d0030818902818100be93b516b28c6e674abe7
abdb11ce0fdf5bb728b75216b73f37a6432e4b402b3ad8139b8c0ba541a72c8add126b6e1a1308f
b98b727beb63c6060356bb177bb7d54b54dbe87aee7353d0a6baa9397704de625d1836d3f42c7ee
5683f6703259592cc24b09699376807f28fe0e00ff882974484d805f874260dfc2d1627473b9102
03010001
-----
value: 4.0
-----
time: 2019-01-14 16:18:01.862946
-----
--------------
sender:
30819f300d06092a864886f70d010101050003818d0030818902818100cba097c0854876f41338c
62598c658f545182cfa4acebce147aedf328181f9c4930f14498fd03c0af6b0cce25be99452a81d
f4fa30a53eddbb7bb7b203adf8764a0ccd9db6913a576d68d642d8fd47452590137869c25d9ff83
d68ebe6d616056a8425b85b52e69715b8b85ae807b84638d8f00e321b65e4c33acaf6469e18e302
03010001
-----
recipient:
30819f300d06092a864886f70d010101050003818d0030818902818100a070c82b34ae143cbe59b
3a2afde7186e9d5bc274955d8112d87a00256a35369acc4d0edfe65e8f9dc93fbd9ee74b9e7ea12
334da38c8c9900e6ced1c4ce93f86e06611e656521a1eab561892b7db0961b4f212d1fd5b5e49ae
09cf8c603a068f9b723aa8a651032ff6f24e5de00387e4d062375799742a359b8f22c5362e56502
03010001
-----
value: 7.0
-----
time: 2019-01-14 16:18:01.863932
-----
--------------
sender:
30819f300d06092a864886f70d010101050003818d0030818902818100be93b516b28c6e674abe7
abdb11ce0fdf5bb728b75216b73f37a6432e4b402b3ad8139b8c0ba541a72c8add126b6e1a1308f
b98b727beb63c6060356bb177bb7d54b54dbe87aee7353d0a6baa9397704de625d1836d3f42c7ee
5683f6703259592cc24b09699376807f28fe0e00ff882974484d805f874260dfc2d1627473b9102
03010001
-----
recipient:
30819f300d06092a864886f70d010101050003818d0030818902818100a070c82b34ae143cbe59b
3a2afde7186e9d5bc274955d8112d87a00256a35369acc4d0edfe65e8f9dc93fbd9ee74b9e7ea12
334da38c8c9900e6ced1c4ce93f86e06611e656521a1eab561892b7db0961b4f212d1fd5b5e49ae
09cf8c603a068f9b723aa8a651032ff6f24e5de00387e4d062375799742a359b8f22c5362e56502
03010001
-----
value: 3.0
-----
time: 2019-01-14 16:18:01.865099
-----
--------------
=====================================
block # 3
sender:
30819f300d06092a864886f70d010101050003818d0030818902818100a070c82b34ae143cbe59b
3a2afde7186e9d5bc274955d8112d87a00256a35369acc4d0edfe65e8f9dc93fbd9ee74b9e7ea12
334da38c8c9900e6ced1c4ce93f86e06611e656521a1eab561892b7db0961b4f212d1fd5b5e49ae
09cf8c603a068f9b723aa8a651032ff6f24e5de00387e4d062375799742a359b8f22c5362e56502
03010001
-----
recipient:
30819f300d06092a864886f70d010101050003818d0030818902818100bb064c99c492144a9f463
480273aba93ac1db1f0da3cb9f3c1f9d058cf499fd8e54d244da0a8dd6ddd329ec86794b04d773e
b4841c9f935ea4d9ccc2821c7a1082d23b6c928d59863407f52fa05d8b47e5157f8fe56c2ce3279
c657f9c6a80500073b0be8093f748aef667c03e64f04f84d311c4d866c12d79d3fc3034563dfb02
03010001
-----
value: 8.0
-----
time: 2019-01-14 16:18:01.866219
-----
--------------
sender:
30819f300d06092a864886f70d010101050003818d0030818902818100a070c82b34ae143cbe59b
3a2afde7186e9d5bc274955d8112d87a00256a35369acc4d0edfe65e8f9dc93fbd9ee74b9e7ea12
334da38c8c9900e6ced1c4ce93f86e06611e656521a1eab561892b7db0961b4f212d1fd5b5e49ae
09cf8c603a068f9b723aa8a651032ff6f24e5de00387e4d062375799742a359b8f22c5362e56502
03010001
-----
recipient:
30819f300d06092a864886f70d010101050003818d0030818902818100be93b516b28c6e674abe7
abdb11ce0fdf5bb728b75216b73f37a6432e4b402b3ad8139b8c0ba541a72c8add126b6e1a1308f
b98b727beb63c6060356bb177bb7d54b54dbe87aee7353d0a6baa9397704de625d1836d3f42c7ee
5683f6703259592cc24b09699376807f28fe0e00ff882974484d805f874260dfc2d1627473b9102
03010001
-----
value: 1.0
-----
time: 2019-01-14 16:18:01.867223
-----
--------------
sender:
30819f300d06092a864886f70d010101050003818d0030818902818100cba097c0854876f41338c
62598c658f545182cfa4acebce147aedf328181f9c4930f14498fd03c0af6b0cce25be99452a81d
f4fa30a53eddbb7bb7b203adf8764a0ccd9db6913a576d68d642d8fd47452590137869c25d9ff83
d68ebe6d616056a8425b85b52e69715b8b85ae807b84638d8f00e321b65e4c33acaf6469e18e302
03010001
-----
recipient: 
30819f300d06092a864886f70d010101050003818d0030818902818100bb064c99c492144a9f463
480273aba93ac1db1f0da3cb9f3c1f9d058cf499fd8e54d244da0a8dd6ddd329ec86794b04d773e
b4841c9f935ea4d9ccc2821c7a1082d23b6c928d59863407f52fa05d8b47e5157f8fe56c2ce3279
c657f9c6a80500073b0be8093f748aef667c03e64f04f84d311c4d866c12d79d3fc3034563dfb02
03010001
-----
value: 5.0
-----
time: 2019-01-14 16:18:01.868241
-----
--------------
=====================================

이 튜토리얼에서는 Python으로 블록 체인 프로젝트를 구성하는 방법을 배웠습니다. 이 프로젝트에 추가 기능을 추가해야하는 영역이 많이 있습니다.

예를 들어 트랜잭션 큐를 관리하기위한 함수를 작성해야합니다. 트랜잭션이 채굴되고 채굴 된 블록이 시스템에 의해 승인되면 더 이상 저장할 필요가 없습니다.

또한 채굴 자들은 확실히 가장 높은 수수료로 거래를 선택하는 것을 선호 할 것입니다. 동시에 수수료가 적거나없는 거래가 영원히 굶주 리지 않도록해야합니다.

대기열 관리를위한 알고리즘을 개발해야합니다. 또한 현재 자습서에는 클라이언트 인터페이스 코드가 포함되어 있지 않습니다. 일반 클라이언트와 광부 모두를 위해 이것을 개발해야합니다. 본격적인 블록 체인 프로젝트는 여러 줄의 코드로 실행되며이 튜토리얼의 범위를 벗어납니다. 관심있는 독자는 추가 연구를 위해 비트 코인 소스 를 다운로드 할 수 있습니다 .

결론

이 선명한 튜토리얼을 통해 자신 만의 블록 체인 프로젝트를 만들 수 있습니다.

본격적인 블록 체인 프로젝트 개발을 위해 비트 코인 소스 에서 더 많은 것을 배울 수 있습니다 .

더 큰 상업용 또는 비상업적 프로젝트의 경우 블록 체인 앱 플랫폼을 바로 사용할 수있는 Ethereum 사용을 고려할 수 있습니다.