Python Blockchain - Panduan Cepat

Dalam tutorial tentang Blockchain, kami telah mempelajari secara detail tentang teori di balik blockchain. Blockchain adalah blok bangunan fundamental di balik mata uang digital paling populer di dunia Bitcoin. Tutorial secara mendalam membahas seluk-beluk Bitcoin yang menjelaskan sepenuhnya arsitektur blockchain. Langkah selanjutnya adalah membangun blockchain kami sendiri.

Satoshi Nakamoto menciptakan mata uang virtual pertama di dunia yang disebut Bitcoin. Melihat kesuksesan Bitcoin, banyak orang lainnya yang menciptakan mata uang virtual mereka sendiri. Untuk beberapa nama - Litecoin, Zcash, dan sebagainya.

Sekarang, Anda mungkin juga ingin meluncurkan mata uang Anda sendiri. Mari kita sebut ini sebagai TPCoin (TutorialsPoint Coin). Anda akan menulis blockchain untuk mencatat semua transaksi yang berhubungan dengan TPCoin. TPCoin dapat digunakan untuk membeli Pizza, Burger, Salad, dll. Mungkin ada penyedia layanan lain yang akan bergabung dengan jaringan Anda dan mulai menerima TPCoin sebagai mata uang untuk memberikan layanan mereka. Kemungkinannya tidak terbatas.

Dalam tutorial ini, mari kita pahami bagaimana membangun sistem seperti itu dan meluncurkan mata uang digital Anda sendiri di pasar.

Komponen yang Terlibat dalam Pengembangan Proyek Blockchain

Seluruh pengembangan proyek blockchain terdiri dari tiga komponen utama -

  • Client
  • Miners
  • Blockchain

Klien

Klien adalah orang yang akan membeli barang dari vendor lain. Klien itu sendiri dapat menjadi vendor dan akan menerima uang dari orang lain atas barang yang dia suplai. Di sini kami berasumsi bahwa klien dapat menjadi pemasok dan penerima TPCoins. Jadi, kami akan membuat kelas klien dalam kode kami yang memiliki kemampuan untuk mengirim dan menerima uang.

Buruh tambang

Penambang adalah orang yang mengambil transaksi dari kumpulan transaksi dan mengumpulkannya dalam satu blok. Penambang harus memberikan bukti kerja yang valid untuk mendapatkan hadiah penambangan. Semua uang yang dikumpulkan penambang sebagai bayaran akan dia simpan. Dia dapat membelanjakan uang itu untuk membeli barang atau jasa dari vendor terdaftar lainnya di jaringan, seperti yang dilakukan oleh Klien yang dijelaskan di atas.

Blockchain

Terakhir, Blockchain adalah struktur data yang menghubungkan semua blok yang ditambang dalam urutan kronologis. Rantai ini tidak dapat diubah dan karenanya tahan temper.

Anda dapat mengikuti tutorial ini dengan mengetikkan kode yang disajikan di setiap langkah di notebook Jupyter baru. Atau, Anda dapat mengunduh seluruh notebook Jupyter dari www.anaconda.com .

Di bab selanjutnya, kami akan mengembangkan klien yang menggunakan sistem blockchain kami.

Klien adalah seseorang yang memegang TPCoin dan mentransaksikannya untuk barang / jasa dari vendor lain di jaringan termasuk miliknya sendiri. Kita harus mendefinisikan fileClientkelas untuk tujuan ini. Untuk membuat identifikasi unik global untuk klien, kami menggunakan PKI (Public Key Infrastructure). Dalam bab ini, mari kita bahas hal ini secara detail.

Klien harus dapat mengirim uang dari dompetnya ke orang lain yang dikenal. Demikian pula, klien harus dapat menerima uang dari pihak ketiga. Untuk membelanjakan uang, klien akan membuat transaksi dengan menyebutkan nama pengirim dan jumlah yang harus dibayarkan. Untuk menerima uang, klien akan memberikan identitasnya kepada pihak ketiga - pada dasarnya pengirim uang. Kami tidak menyimpan jumlah saldo uang yang disimpan klien di dompetnya. Selama transaksi, kami akan menghitung saldo sebenarnya untuk memastikan bahwa klien memiliki saldo yang cukup untuk melakukan pembayaran.

Untuk mengembangkan Clientkelas dan kode lainnya dalam proyek ini, kita perlu mengimpor banyak pustaka Python. Ini tercantum di bawah -

# 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

Selain pustaka standar di atas, kami akan menandatangani transaksi kami, membuat hash objek, dll. Untuk ini, Anda perlu mengimpor pustaka berikut -

# 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

Pada bab berikutnya, mari kita bicara tentang kelas klien.

Itu Client kelas menghasilkan private dan public kunci dengan menggunakan Python bawaan RSAalgoritma. Pembaca yang tertarik bisa merujuk kethis tutorialuntuk implementasi RSA. Selama inisialisasi objek, kami membuat kunci privat dan publik dan menyimpan nilainya di variabel instance.

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

Perhatikan bahwa Anda tidak boleh kehilangan kunci pribadi Anda. Untuk pencatatan, kunci pribadi yang dihasilkan dapat disalin pada penyimpanan eksternal yang aman atau Anda dapat dengan mudah menuliskan representasi ASCII-nya di selembar kertas.

Yang dihasilkan publickey akan digunakan sebagai identitas klien. Untuk ini, kami mendefinisikan properti yang disebutidentity yang mengembalikan representasi HEX dari kunci publik.

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

Itu identityunik untuk setiap klien dan dapat dibuat tersedia untuk umum. Siapa pun dapat mengirim mata uang virtual kepada Anda menggunakan iniidentity dan itu akan ditambahkan ke dompet Anda.

Kode lengkap untuk Client kelas ditampilkan di sini -

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')

Menguji Klien

Sekarang, kita akan menulis kode yang akan mengilustrasikan cara menggunakan Client kelas -

Dinesh = Client()
print (Dinesh.identity)

Kode di atas membuat instance Client dan menugaskannya ke variabel Dinesh. Kami mencetak kunci publik dariDinesh dengan memanggilnya identitymetode. Outputnya ditampilkan di sini -

30819f300d06092a864886f70d010101050003818d0030818902818100b547fafceeb131e07
0166a6b23fec473cce22c3f55c35ce535b31d4c74754fecd820aa94c1166643a49ea5f49f72
3181ff943eb3fdc5b2cb2db12d21c06c880ccf493e14dd3e93f3a9e175325790004954c34d3
c7bc2ccc9f0eb5332014937f9e49bca9b7856d351a553d9812367dc8f2ac734992a4e6a6ff6
6f347bd411d07f0203010001

Sekarang, mari kita lanjutkan untuk membuat transaksi di bab berikutnya.

Dalam bab ini, mari kita buat file Transactionkelas sehingga klien dapat mengirim uang ke seseorang. Perhatikan bahwa klien bisa menjadi pengirim atau penerima uang. Saat Anda ingin menerima uang, beberapa pengirim lain akan membuat transaksi dan menentukanpublicalamat di dalamnya. Kami mendefinisikan inisialisasi kelas transaksi sebagai berikut -

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

Itu init Metode mengambil tiga parameter - pengirim public kunci, penerima publickunci, dan jumlah yang akan dikirim. Ini disimpan dalam variabel instan untuk digunakan oleh metode lain. Selain itu, kami membuat satu variabel lagi untuk menyimpan waktu transaksi.

Selanjutnya, kami menulis metode utilitas yang disebut to_dictyang menggabungkan keempat variabel contoh yang disebutkan di atas dalam objek kamus. Ini hanya untuk menempatkan seluruh informasi transaksi yang dapat diakses melalui satu variabel.

Seperti yang Anda ketahui dari tutorial sebelumnya bahwa blok pertama dalam blockchain adalah a Genesisblok. Blok Genesis berisi transaksi pertama yang dimulai oleh pembuat blockchain. Identitas orang ini mungkin dirahasiakan seperti dalam kasus Bitcoin. Jadi saat transaksi pertama ini dibuat, pencipta boleh saja mengirimkan identitasnya sebagaiGenesis. Jadi, saat membuat kamus, kami memeriksa apakah pengirimnyaGenesisdan jika demikian, kami hanya menetapkan beberapa nilai string ke variabel identitas; jika tidak, kami menetapkan identitas pengirim keidentity variabel.

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

Kami membangun kamus menggunakan baris kode berikut

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

Seluruh kode untuk to_dict metode ditunjukkan di bawah ini -

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})

Terakhir, kami akan menandatangani objek kamus ini menggunakan kunci pribadi pengirim. Seperti sebelumnya, kami menggunakan PKI bawaan dengan algoritme SHA. Tanda tangan yang dihasilkan didekodekan untuk mendapatkan representasi ASCII untuk dicetak dan disimpan di blockchain kami. Itusign_transaction kode metode ditampilkan di sini -

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')

Kami sekarang akan menguji ini Transaction kelas.

Menguji Kelas Transaksi

Untuk tujuan ini, kami akan membuat dua pengguna, yang disebut Dinesh dan Ramesh. Dinesh akan mengirim 5 TPCoin ke Ramesh. Untuk ini pertama kami membuat klien bernama Dinesh dan Ramesh.

Dinesh = Client()
Ramesh = Client()

Ingatlah bahwa ketika Anda membuat Client kelas, itu public andkunci pribadi yang unik untuk klien akan dibuat. Saat Dinesh mengirim pembayaran ke Ramesh, dia akan membutuhkan kunci publik Ramesh yang diperoleh dengan menggunakan properti identitas klien.

Jadi, kami akan membuat contoh transaksi menggunakan kode berikut -

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

Perhatikan bahwa parameter pertama adalah pengirim, parameter kedua adalah kunci publik penerima dan parameter ketiga adalah jumlah yang akan ditransfer. Itusign_transaction metode mengambil kunci pribadi pengirim dari parameter pertama untuk menyanyikan transaksi.

Setelah objek transaksi dibuat, Anda akan menandatanganinya dengan memanggilnya sign_transactionmetode. Metode ini mengembalikan tanda tangan yang dihasilkan dalam format yang dapat dicetak. Kami menghasilkan dan mencetak tanda tangan menggunakan dua baris kode berikut -

signature = t.sign_transaction()
print (signature)

Ketika Anda menjalankan kode di atas, Anda akan melihat output yang mirip dengan ini -

7c7e3c97629b218e9ec6e86b01f9abd8e361fd69e7d373c38420790b655b9abe3b575e343c7
13703ca1aee781acd7157a0624db3d57d7c2f1172730ee3f45af943338157f899965856f6b0
0e34db240b62673ad5a08c8e490f880b568efbc36035cae2e748f1d802d5e8e66298be826f5
c6363dc511222fb2416036ac04eb972

Sekarang karena infrastruktur dasar kami untuk membuat klien dan transaksi sudah siap, kami sekarang akan memiliki banyak klien yang melakukan banyak transaksi seperti dalam situasi kehidupan nyata.

Transaksi yang dilakukan oleh berbagai klien dimasukkan ke dalam sistem; para penambang mengambil transaksi dari antrian ini dan menambahkannya ke blok. Mereka kemudian akan menambang blok tersebut dan penambang yang menang akan memiliki hak istimewa untuk menambahkan blok tersebut ke blockchain dan dengan demikian mendapatkan sejumlah uang untuk dirinya sendiri.

Kami akan menjelaskan proses penambangan ini nanti ketika kami membahas pembuatan blockchain. Sebelum kita menulis kode untuk beberapa transaksi, mari kita tambahkan fungsi utilitas kecil untuk mencetak konten dari transaksi tertentu.

Menampilkan Transaksi

Itu display_transactionfungsi menerima satu parameter dari jenis transaksi. Objek kamus dalam transaksi yang diterima disalin ke variabel sementara yang dipanggildict dan menggunakan kunci kamus, berbagai nilai dicetak di konsol.

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 ('-----')

Selanjutnya, kami menentukan antrian transaksi untuk menyimpan objek transaksi kami.

Antrian Transaksi

Untuk membuat antrian, kami mendeklarasikan global list variabel yang disebut transactions sebagai berikut -

transactions = []

Kami hanya akan menambahkan setiap transaksi yang baru dibuat ke antrian ini. Harap dicatat bahwa untuk singkatnya, kami tidak akan menerapkan logika manajemen antrian dalam tutorial ini.

Membuat Banyak Klien

Sekarang, kita akan mulai membuat transaksi. Pertama, kami akan membuat empat klien yang akan saling mengirim uang untuk mendapatkan berbagai layanan atau barang dari orang lain.

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

Saat ini, kami memiliki empat klien bernama Dinesh, Ramesh, Seema, dan Vijay. Saat ini kami berasumsi bahwa masing-masing klien ini memegang beberapa TPCoin di dompet mereka untuk bertransaksi. Identitas masing-masing klien ini akan ditentukan dengan menggunakan properti identitas objek ini.

Membuat Transaksi Pertama

Sekarang, kami memulai transaksi pertama kami sebagai berikut -

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

Dalam transaksi ini Dinesh mengirimkan 5 TPCoin ke Ramesh. Agar transaksi berhasil, kami harus memastikan Dinesh memiliki cukup uang di dompetnya untuk pembayaran ini. Perhatikan bahwa, kita memerlukan transaksi genesis untuk memulai sirkulasi TPCoin dalam sistem. Anda akan menulis kode transaksi untuk transaksi genesis ini segera setelah Anda membaca.

Kami akan menandatangani transaksi ini menggunakan kunci pribadi Dinesh dan menambahkannya ke antrian transaksi sebagai berikut -

t1.sign_transaction()
transactions.append(t1)

Setelah transaksi pertama dilakukan oleh Dinesh, kami akan membuat beberapa transaksi lagi antara klien berbeda yang kami buat di atas.

Menambahkan Lebih Banyak Transaksi

Kami sekarang akan membuat beberapa transaksi lagi, setiap transaksi diberikan beberapa TPCoin ke pihak lain. Ketika seseorang membelanjakan uang, dia tidak perlu memeriksa saldo yang cukup di dompet ini. Penambang dengan cara apapun akan memvalidasi setiap transaksi untuk saldo yang dimiliki pengirim saat memulai transaksi.

Jika saldo tidak mencukupi, penambang akan menandai transaksi ini sebagai tidak valid dan tidak akan menambahkannya ke blok ini.

Kode berikut membuat dan menambahkan sembilan transaksi lagi ke antrian kami.

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)

Saat Anda menjalankan kode di atas, Anda akan memiliki sepuluh transaksi dalam antrian untuk penambang untuk membuat blok mereka.

Transaksi Dumping

Sebagai manajer blockchain, Anda mungkin ingin meninjau konten antrian transaksi secara berkala. Untuk tujuan ini, Anda dapat menggunakan filedisplay_transactionfungsi yang kami kembangkan sebelumnya. Untuk membuang semua transaksi dalam antrian, cukup ulangi daftar transaksi dan untuk setiap transaksi yang direferensikan, panggildisplay_transaction berfungsi seperti yang ditunjukkan di sini -

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

Transaksi dipisahkan dengan garis putus-putus untuk membedakan. Jika Anda menjalankan kode di atas, Anda akan melihat daftar transaksi seperti yang ditunjukkan di bawah ini -

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
-----
--------------

Untuk singkatnya, saya hanya mencetak beberapa transaksi pertama dalam daftar. Pada kode di atas, kami mencetak semua transaksi yang dimulai dengan transaksi pertama kecuali transaksi genesis yang tidak pernah ditambahkan ke daftar ini. Karena transaksi ditambahkan ke blok secara berkala, Anda biasanya hanya tertarik untuk melihat daftar transaksi yang belum ditambang. Dalam hal ini, Anda perlu membuat filefor loop untuk mengulang melalui transaksi yang belum ditambang.

Sejauh ini, Anda telah mempelajari cara membuat klien, mengizinkan mereka untuk berada di antara mereka sendiri, dan menjaga antrian transaksi tertunda yang akan ditambang. Sekarang, sampai pada bagian terpenting dari tutorial ini dan itu adalah membuat blockchain itu sendiri. Anda akan mempelajarinya di pelajaran berikutnya.

Sebuah blok terdiri dari sejumlah transaksi yang bervariasi. Untuk kesederhanaan, dalam kasus kami, kami akan mengasumsikan bahwa blok terdiri dari sejumlah transaksi tetap, yaitu tiga dalam kasus ini. Karena blok perlu menyimpan daftar dari tiga transaksi ini, kami akan mendeklarasikan variabel instan yang dipanggilverified_transactions sebagai berikut -

self.verified_transactions = []

Kami menamai variabel ini sebagai verified_transactionsuntuk menunjukkan bahwa hanya transaksi valid yang diverifikasi yang akan ditambahkan ke blok. Setiap blok juga menyimpan nilai hash dari blok sebelumnya, sehingga rantai blok menjadi tidak dapat diubah.

Untuk menyimpan hash sebelumnya, kami mendeklarasikan variabel instance sebagai berikut -

self.previous_block_hash = ""

Akhirnya, kami mendeklarasikan satu variabel lagi yang disebut Nonce untuk menyimpan nonce yang dibuat oleh penambang selama proses penambangan.

self.Nonce = ""

Definisi lengkap dari Block kelas diberikan di bawah -

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

Karena setiap blok membutuhkan nilai hash blok sebelumnya, kami mendeklarasikan variabel global yang dipanggil last_block_hash sebagai berikut -

last_block_hash = ""

Sekarang mari kita buat blok pertama kita di blockchain.

Kami berasumsi bahwa pembuat TPCoins awalnya memberikan 500 TPCoin kepada klien yang dikenal Dinesh. Untuk ini, pertama-tama dia membuat instance Dinesh -

Dinesh = Client()

Kami kemudian membuat transaksi genesis dan mengirim 500 TPCoin ke alamat publik Dinesh.

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

Sekarang, kami membuat instance Block kelas dan menyebutnya block0.

block0 = Block()

Kami menginisialisasi previous_block_hash dan Nonce variabel instan ke None, karena ini adalah transaksi pertama yang disimpan di blockchain kami.

block0.previous_block_hash = None
Nonce = None

Selanjutnya, kami akan menambahkan transaksi t0 di atas ke verified_transactions daftar dipertahankan dalam blok -

block0.verified_transactions.append (t0)

Pada titik ini, blok tersebut benar-benar diinisialisasi dan siap untuk ditambahkan ke blockchain kami. Kami akan membuat blockchain untuk tujuan ini. Sebelum kami menambahkan blok ke blockchain, kami akan melakukan hash blok dan menyimpan nilainya dalam variabel global yang disebutlast_block_hashyang kami nyatakan sebelumnya. Nilai ini akan digunakan oleh penambang berikutnya di bloknya.

Kami menggunakan dua baris pengkodean berikut untuk melakukan hashing pada blok dan menyimpan nilai intisari.

digest = hash (block0)
last_block_hash = digest

Akhirnya, kami membuat blockchain seperti yang kita lihat di bab selanjutnya.

Sebuah blockchain berisi daftar blok yang dirantai satu sama lain. Untuk menyimpan seluruh daftar, kita akan membuat variabel daftar yang disebut TPCoins -

TPCoins = []

Kami juga akan menulis metode utilitas yang disebut dump_blockchainuntuk membuang konten dari seluruh blockchain. Kami pertama kali mencetak panjang blockchain sehingga kami tahu berapa banyak blok yang saat ini ada di blockchain.

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

Perhatikan bahwa seiring berjalannya waktu, jumlah blok di blockchain akan sangat tinggi untuk dicetak. Jadi, saat Anda mencetak konten blockchain, Anda mungkin harus memutuskan kisaran yang ingin Anda periksa. Pada kode di bawah ini, kami telah mencetak seluruh blockchain karena kami tidak akan menambahkan terlalu banyak blok di demo saat ini.

Untuk melakukan iterasi melalui rantai, kami menyiapkan a for loop sebagai berikut -

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

Setiap blok yang direferensikan disalin ke variabel sementara yang dipanggil block_temp.

Kami mencetak nomor blok sebagai judul untuk setiap blok. Perhatikan bahwa angka akan dimulai dengan nol, blok pertama adalah blok genesis yang diberi nomor nol.

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

Di dalam setiap blok, kami telah menyimpan daftar tiga transaksi (kecuali untuk blok genesis) dalam variabel yang disebut verified_transactions. Kami mengulang daftar ini difor loop dan untuk setiap item yang diambil, kami memanggil display_transaction berfungsi untuk menampilkan detail transaksi.

for transaction in block_temp.verified_transactions:
   display_transaction (transaction)

Seluruh definisi fungsi ditampilkan di bawah -

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 ('=====================================')

Perhatikan bahwa di sini kami telah memasukkan pemisah pada titik-titik yang sesuai dalam kode untuk membatasi blok dan transaksi di dalamnya.

Karena kita sekarang telah membuat blockchain untuk menyimpan blok, tugas kita selanjutnya adalah membuat blok dan mulai menambahkannya ke blockchain. Untuk tujuan ini, kami akan menambahkan blok genesis yang telah Anda buat di langkah sebelumnya.

Menambahkan blok ke blockchain melibatkan penambahan blok yang dibuat ke kami TPCoins daftar.

TPCoins.append (block0)

Perhatikan bahwa tidak seperti blok lainnya dalam sistem, blok genesis hanya berisi satu transaksi yang diprakarsai oleh pembuat sistem TPCoins. Sekarang, Anda akan membuang konten blockchain dengan memanggil fungsi global kamidump_blockchain -

dump_blockchain(TPCoins)

Saat Anda menjalankan fungsi ini, Anda akan melihat output berikut -

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
-----
--------------
=====================================

Pada titik ini sistem blockchain siap digunakan. Kami sekarang akan memungkinkan klien yang tertarik untuk menjadi penambang dengan memberi mereka fungsionalitas penambangan.

Untuk memungkinkan penambangan, kita perlu mengembangkan fungsi penambangan. Fungsionalitas penambangan perlu menghasilkan intisari pada string pesan tertentu dan memberikan bukti kerja. Mari kita bahas ini di bab ini.

Fungsi Intisari Pesan

Kami akan menulis fungsi utilitas yang disebut sha256 untuk membuat intisari pada pesan tertentu -

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

Itu sha256 fungsi membutuhkan message sebagai parameter, mengkodekannya ke ASCII, menghasilkan intisari heksadesimal dan mengembalikan nilai ke pemanggil.

Fungsi Penambangan

Kami sekarang mengembangkan minefungsi yang menerapkan strategi penambangan kami sendiri. Strategi kami dalam hal ini adalah menghasilkan hash pada pesan tertentu yang diawali dengan angka 1 tertentu. Jumlah 1 yang ditentukan ditetapkan sebagai parameter kemine fungsi yang ditentukan sebagai tingkat kesulitan.

Misalnya, jika Anda menentukan tingkat kesulitan 2, hash yang dihasilkan pada pesan tertentu harus dimulai dengan dua angka 1 - seperti 11xxxxxxxx. Jika tingkat kesulitannya 3, hash yang dihasilkan harus dimulai dengan tiga angka 1 - seperti 111xxxxxxxx. Dengan persyaratan ini, kami sekarang akan mengembangkan fungsi penambangan seperti yang ditunjukkan pada langkah-langkah yang diberikan di bawah ini.

Langkah 1

Fungsi penambangan mengambil dua parameter - pesan dan tingkat kesulitan.

def mine(message, difficulty=1):

Langkah 2

Tingkat kesulitan harus lebih besar atau sama dengan 1, kami memastikan ini dengan pernyataan assert berikut -

assert difficulty >= 1

LANGKAH 3

Kami membuat prefix variabel menggunakan tingkat kesulitan yang ditetapkan.

prefix = '1' * difficulty

Perhatikan jika tingkat kesulitannya adalah 2, awalannya adalah "11" dan jika tingkat kesulitannya adalah 3, awalannya adalah "111", dan seterusnya. Kami akan memeriksa apakah prefiks ini ada di intisari pesan yang dihasilkan. Untuk mencerna pesan itu sendiri, kami menggunakan dua baris kode berikut -

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

Kami terus menambahkan nomor baru ike hash pesan di setiap iterasi dan menghasilkan intisari baru pada pesan gabungan. Sebagai masukan untuksha256 fungsi berubah di setiap iterasi, file digestnilai juga akan berubah. Kami memeriksa apakah inidigest nilai telah ditetapkan di atas prefix.

if digest.startswith(prefix):

Jika kondisi terpenuhi, kami akan menghentikan for ulangi dan kembalikan digest nilai bagi pemanggil.

Seluruhnya mine kode ditampilkan di sini -

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

Untuk pengertian Anda, kami telah menambahkan print pernyataan yang mencetak nilai intisari dan jumlah iterasi yang diperlukan untuk memenuhi kondisi sebelum kembali dari fungsi.

Menguji Fungsi Penambangan

Untuk menguji fungsi penambangan kami, cukup jalankan pernyataan berikut -

mine ("test message", 2)

Ketika Anda menjalankan kode di atas, Anda akan melihat output yang mirip dengan yang di bawah ini -

after 138 iterations found nonce:
11008a740eb2fa6bf8d55baecda42a41993ca65ce66b2d3889477e6bfad1484c

Perhatikan bahwa intisari yang dihasilkan dimulai dengan "11". Jika Anda mengubah tingkat kesulitan ke 3, intisari yang dihasilkan akan dimulai dengan "111", dan tentu saja, ini mungkin memerlukan lebih banyak jumlah iterasi. Seperti yang Anda lihat, penambang dengan lebih banyak kekuatan pemrosesan akan dapat menambang pesan yang diberikan sebelumnya. Begitulah cara para penambang bersaing satu sama lain untuk mendapatkan pendapatan mereka.

Sekarang, kami siap menambahkan lebih banyak blok ke blockchain kami. Mari kita pelajari ini di bab selanjutnya.

Setiap penambang akan mengambil transaksi dari kumpulan transaksi yang dibuat sebelumnya. Untuk melacak jumlah pesan yang sudah ditambang, kita harus membuat variabel global -

last_transaction_index = 0

Kami sekarang akan memiliki penambang pertama kami menambahkan blok ke blockchain.

Menambahkan Blok Pertama

Untuk menambahkan blok baru, pertama-tama kita membuat sebuah instance dari Block kelas.

block = Block()

Kami mengambil 3 transaksi teratas dari antrian -

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

Sebelum menambahkan transaksi ke blok, penambang akan memverifikasi validitas transaksi. Validitas transaksi diverifikasi dengan menguji kesamaan hash yang diberikan oleh pengirim terhadap hash yang dihasilkan oleh penambang menggunakan kunci publik pengirim. Selain itu, penambang akan memverifikasi bahwa pengirim memiliki saldo yang cukup untuk membayar transaksi saat ini.

Untuk singkatnya, kami belum memasukkan fungsi ini dalam tutorial. Setelah transaksi divalidasi, kami menambahkannya keverified_transactions daftar di block contoh.

block.verified_transactions.append (temp_transaction)

Kami menaikkan indeks transaksi terakhir sehingga penambang berikutnya akan mengambil transaksi berikutnya dalam antrian.

last_transaction_index += 1

Kami menambahkan tepat tiga transaksi ke blok tersebut. Setelah ini selesai, kami akan menginisialisasi variabel instan lainnya dariBlockkelas. Kami pertama kali menambahkan hash dari blok terakhir.

block.previous_block_hash = last_block_hash

Selanjutnya, kami menambang blok dengan tingkat kesulitan 2.

block.Nonce = mine (block, 2)

Perhatikan bahwa parameter pertama ke minefungsi adalah objek biner. Kami sekarang hash seluruh blok dan membuat intisari di atasnya.

digest = hash (block)

Terakhir, kami menambahkan blok yang dibuat ke blockchain dan menginisialisasi ulang variabel global last_block_hash untuk digunakan di blok berikutnya.

Seluruh kode untuk menambahkan blok ditampilkan di bawah ini -

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

Menambahkan Lebih Banyak Blok

Kami sekarang akan menambahkan dua blok lagi ke blockchain kami. Kode untuk menambahkan dua blok berikutnya diberikan di bawah ini -

# 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

Saat Anda menambahkan dua blok ini, Anda juga akan melihat jumlah iterasi yang diperlukan untuk menemukan Nonce. Pada titik ini, blockchain kami terdiri dari total 4 blok termasuk blok genesis.

Membuang Seluruh Blockchain

Anda dapat memverifikasi konten dari seluruh blockchain menggunakan pernyataan berikut -

dump_blockchain(TPCoins)

Anda akan melihat output yang mirip dengan yang ditunjukkan di bawah ini -

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
-----
--------------
=====================================

Dalam tutorial ini, kami telah mempelajari cara membuat proyek blockchain dengan Python. Ada banyak area di mana Anda perlu menambahkan fungsionalitas lebih lanjut ke proyek ini.

Misalnya, Anda perlu menulis fungsi untuk mengelola antrian transaksi. Setelah transaksi ditambang dan blok yang ditambang diterima oleh sistem, mereka tidak perlu disimpan lagi.

Selain itu, para penambang pasti akan lebih memilih untuk mengambil transaksi dengan bayaran tertinggi. Pada saat yang sama, Anda harus memastikan bahwa transaksi dengan biaya rendah atau tanpa biaya tidak akan kelaparan selamanya.

Anda perlu mengembangkan algoritme untuk mengelola antrean. Selain itu, tutorial saat ini tidak menyertakan kode antarmuka klien. Anda perlu mengembangkan ini untuk klien normal dan penambang. Proyek blockchain yang lengkap akan mengalami beberapa baris kode lagi dan berada di luar cakupan tutorial ini. Pembaca yang tertarik dapat mengunduh sumber bitcoin untuk studi lebih lanjut.

Kesimpulan

Tutorial tajam ini akan membantu Anda mulai membuat proyek blockchain Anda sendiri.

Untuk pengembangan proyek blockchain lengkap, Anda dapat mempelajari lebih lanjut dari sumber bitcoin .

Untuk proyek komersial atau non-komersial yang lebih besar, Anda dapat mempertimbangkan untuk menggunakan Ethereum - platform aplikasi blockchain yang siap digunakan.