-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathSocialChain.py
190 lines (139 loc) · 5.41 KB
/
SocialChain.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
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
import hashlib
import json
from time import time
from uuid import uuid4
from urllib.parse import urlparse
import requests
class Blockchain(object):
def __init__(self):
# Initiates the blockchain
self.chain = []
self.current_transactions = []
# Creating a genesis block
self.new_block(previous_hash=1, proof=100)
# Unique nodes
self.nodes = set()
def new_block(self, proof, previous_hash=None):
'''
Adds new block to the chain
:param proof: <int> The proof provided by Proof of Work algorithm
:param previous_hash: (Optional) <str> Hash of previous block
:return: <dict> New block
'''
block = {
'index': len(self.chain) + 1,
'timestamp': time(),
'transactions': self.current_transactions,
'proof': proof,
'previous_hash': previous_hash or self.hash(self.chain[-1]),
}
# Resetting the current list of transactions
self.current_transactions = []
self.chain.append(block)
return block
def new_transaction(self, sender, recipient, amount):
'''
Creates a new transaction to move to the newly mined block
:param sender: <str> Address of the sender
:param recipient: <str> Address of the recipient
:param amount: <int> Amount transfered
:return: <int> Index of the block that will hold this transaction (New block)
'''
self.current_transactions.append({
'sender': sender,
'recipient': recipient,
'amount': amount,
})
return self.last_block['index'] + 1
@staticmethod
def hash(block):
'''
Adds hashing to the newly added block (SHA-256 hashing alogrithm)
:param block: <dict> Newly created block to which hashing needs to be added
:return: <str>
'''
# Sorting keys to prevent inconsistent hashing
block_string = json.dumps(block, sort_keys=True).encode()
return hashlib.sha256(block_string).hexdigest()
@property
def last_block(self):
# Returns the last block in the chain
return self.chain[-1]
def proof_of_work(self, last_proof):
'''
Proof of Work Algorithm to check the validity of provided solution by miner to mine new block
- Find a number p' such that hash(pp') contains leading numbers as 2711, where p is the previous p'
- p is the previous proof and p' is the new proof
:param last_proof: <int>
:return: <int>
'''
proof = 0
while self.valid_proof(last_proof, proof) is False:
proof += 1
return proof
@staticmethod
def valid_proof(last_proof, proof):
'''
Validates the proof: Does hash(last_proof,proof) contains leading digits as 2711 or not?
:param last_proof: <int> Previous proof
:param proof: <int> Current Proof
:return: <bool> True if correct, Fasle if algorithm isn't satisfied
'''
guess = f'{last_proof}{proof}'.encode()
guess_hash = hashlib.sha256(guess).hexdigest()
return guess_hash[:4] == '2711'
def register_node(self, address):
"""
Add a new node to the list of nodes
:param address: <str> Address of node. Eg. 'http://192.168.0.5:5000'
:return: None
"""
parsed_url = urlparse(address)
self.nodes.add(parsed_url.netloc)
def valid_chain(self, chain):
"""
Determine if a given blockchain is valid
:param chain: <list> A blockchain
:return: <bool> True if valid, False if not
"""
last_block = chain[0]
current_index = 1
while current_index < len(chain):
block = chain[current_index]
print(f'{last_block}')
print(f'{block}')
print("\n-----------\n")
# Check that the hash of the block is correct
if block['previous_hash'] != self.hash(last_block):
return False
# Check that the Proof of Work is correct
if not self.valid_proof(last_block['proof'], block['proof']):
return False
last_block = block
current_index += 1
return True
def resolve_conflicts(self):
"""
This is the Consensus Algorithm, it resolves conflicts
by replacing the user's chain with the longest one in the network.
:return: <bool> True if our chain was replaced, False if not
"""
neighbours = self.nodes
new_chain = None
# We're only looking for chains longer than the main
max_length = len(self.chain)
# Grab and verify the chains from all the nodes in the network
for node in neighbours:
response = requests.get(f'http://{node}/chain')
if response.status_code == 200:
length = response.json()['length']
chain = response.json()['chain']
# Check if the length is longer and the chain is valid
if length > max_length and self.valid_chain(chain):
max_length = length
new_chain = chain
# Replace main chain if new valid longer chain is discovered
if new_chain:
self.chain = new_chain
return True
return False