-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathblockchain.py
104 lines (85 loc) · 2.5 KB
/
blockchain.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
import hashlib
import json
from time import time
from urllib.parse import urlparse
import requests
####### block generation & its principle
class Blockchain(object):
# initialize the blockchain info
def __init__(self):
self.chain = []
self.current_transaction = []
self.nodes = set()
# genesis block
self.new_block(previous_hash=1, proof=100)
def new_block(self,proof,previous_hash=None):
block = {
'index': len(self.chain)+1,
'timestamp': time(), # timestamp from 1970
'transactions': self.current_transaction,
'proof': proof,
'previous_hash': previous_hash or self.hash(self.chain[-1])
}
self.current_transaction = []
self.chain.append(block)
return block
def new_transaction(self,sender,recipient,amount):
self.current_transaction.append(
{
'sender' : sender,
'recipient' : recipient,
'amount' : amount
}
)
return self.last_block['index'] + 1
def register_node(self, address):
parsed_url = urlparse(address)
self.nodes.add(parsed_url.netloc) # netloc attribute! network lockation
def valid_chain(self,chain):
last_block = chain[0]
current_index = 1
while current_index < len(chain):
block = chain[current_index]
print('%s' % last_block)
print('%s' % block)
print("\n---------\n")
# check that the hash of the block is correct
if block['previous_hash'] != self.hash(last_block):
return False
last_block = block
current_index += 1
return True
def resolve_conflicts(self):
neighbours = self.nodes
new_chain = None
max_length = len(self.chain) # Our chain length
for node in neighbours:
tmp_url = 'http://' + str(node) + '/chain'
response = requests.get(tmp_url)
if response.status_code == 200:
length = response.json()['length']
chain = response.json()['chain']
if length > max_length and self.valid_chain(chain):
max_length = length
if new_chain:
self.chain = new_chain
return True
return False
# directly access from class, share! not individual instance use it
@staticmethod
def hash(block):
block_string = json.dumps(block, sort_keys=True).encode()
return hashlib.sha256(block_string).hexdigest()
@property
def last_block(self):
return self.chain[-1]
def pow(self, last_proof):
proof = 0
while self.valid_proof(last_proof, proof) is False:
proof += 1
return proof
@staticmethod
def valid_proof(last_proof, proof):
guess = str(last_proof + proof).encode()
guess_hash = hashlib.sha256(guess).hexdigest()
return guess_hash[:4] == "0000" # nonce