2019-02-06 06:27:51 -05:00
|
|
|
from web3 import Web3, HTTPProvider
|
|
|
|
from web3.shh import Shh
|
|
|
|
import random
|
|
|
|
|
2019-02-06 07:27:31 -05:00
|
|
|
# Temp
|
|
|
|
import sync_pb2
|
2019-02-06 06:27:51 -05:00
|
|
|
|
|
|
|
# XXX: This assumes a node is actually running - shell out to boot geth?
|
|
|
|
class WhisperNodeHelper():
|
2019-02-23 02:48:46 -05:00
|
|
|
def __init__(self, keypair, host):
|
|
|
|
self.host = host
|
2019-02-06 07:27:31 -05:00
|
|
|
self.web3 = Web3(HTTPProvider(self.host))
|
|
|
|
Shh.attach(self.web3, "shh")
|
2019-02-06 06:27:51 -05:00
|
|
|
|
2019-02-18 23:12:58 -05:00
|
|
|
assert self.web3.shh.web3.isConnected(), "Node not connected, are you running it?"
|
|
|
|
|
2019-02-06 06:27:51 -05:00
|
|
|
self.topic="0xf8946aac" # discovery-topic
|
|
|
|
|
2019-02-06 07:27:31 -05:00
|
|
|
self.keyPair = keypair
|
2019-02-06 10:38:34 -05:00
|
|
|
# XXX: Doesn't belong here
|
|
|
|
self.kId = self.web3.shh.addPrivateKey(self.keyPair)
|
2019-02-06 07:27:31 -05:00
|
|
|
self.myFilter = self.poll_filter(self.topic, self.keyPair)
|
2019-02-06 06:27:51 -05:00
|
|
|
|
2019-02-23 02:57:52 -05:00
|
|
|
# XXX: Race, since before this is set it isn't ready to receive
|
|
|
|
# Later sync node - setup in Node init fn
|
|
|
|
self.sync_node = None
|
|
|
|
|
2019-02-06 06:27:51 -05:00
|
|
|
# XXX: Prune this
|
2019-02-23 02:57:52 -05:00
|
|
|
#self.nodes = []
|
2019-02-06 06:27:51 -05:00
|
|
|
self.time = 0
|
2019-02-06 07:27:31 -05:00
|
|
|
#self.queue = {}
|
|
|
|
#self.peers = {}
|
2019-02-06 06:27:51 -05:00
|
|
|
# Global network reliability
|
|
|
|
self.reliability = 1 # 0.95? Dunno.
|
|
|
|
|
|
|
|
def poll_filter(self, topic, keyPair):
|
|
|
|
# XXX: Doesn't belong here
|
2019-02-06 10:38:34 -05:00
|
|
|
#kId = self.web3.shh.addPrivateKey(keyPair)
|
|
|
|
pubKey = self.web3.shh.getPublicKey(self.kId)
|
2019-02-23 03:07:13 -05:00
|
|
|
#print("***KID", self.kId)
|
|
|
|
#print("***PUBKEY", pubKey)
|
2019-02-23 02:44:38 -05:00
|
|
|
myFilter = self.web3.shh.newMessageFilter({'topic': self.topic,
|
2019-02-06 10:38:34 -05:00
|
|
|
'privateKeyID': self.kId})
|
2019-02-06 06:27:51 -05:00
|
|
|
# Purpose of this if we do getMessages?
|
|
|
|
myFilter.poll_interval = 600;
|
2019-02-23 02:44:38 -05:00
|
|
|
# XXX: Does this actually do anything?
|
2019-02-06 06:27:51 -05:00
|
|
|
return myFilter
|
2019-02-23 02:44:38 -05:00
|
|
|
|
|
|
|
# XXX: BUG - this isn't ticking! I guess it should?!?!?!?!
|
|
|
|
# TODO HEREATM
|
2019-02-06 06:27:51 -05:00
|
|
|
def tick(self):
|
2019-02-06 07:27:31 -05:00
|
|
|
filterID = self.myFilter.filter_id
|
2019-02-23 02:44:38 -05:00
|
|
|
#print("*** tick whisper", filterID)
|
2019-02-06 07:27:31 -05:00
|
|
|
retreived_messages = self.web3.shh.getMessages(filterID)
|
2019-02-23 02:44:38 -05:00
|
|
|
#print("*** tick whisper retrieved", retreived_messages)
|
2019-02-06 06:27:51 -05:00
|
|
|
|
|
|
|
# TODO: Deal with these messages similar to simulation
|
|
|
|
# receiver.on_receive(sender, msg)
|
|
|
|
for i in range(0, len(retreived_messages)):
|
|
|
|
#print(retreived_messages[i]['payload'])
|
2019-02-06 07:27:31 -05:00
|
|
|
#print("\nRECV TYPE", type(retreived_messages[i]['payload']))
|
|
|
|
|
|
|
|
# XXX: This parsing should probably happen elsewhere
|
|
|
|
msg = sync_pb2.Record()
|
2019-02-06 10:38:34 -05:00
|
|
|
#full = retreived_messages[i]
|
|
|
|
sig = retreived_messages[i]['sig']
|
2019-02-23 03:07:13 -05:00
|
|
|
#print("***SIG", sig.hex())
|
2019-02-06 07:27:31 -05:00
|
|
|
payload = retreived_messages[i]['payload']
|
|
|
|
#print("\nRECV payload", payload)
|
|
|
|
msg.ParseFromString(payload)
|
2019-02-23 03:07:13 -05:00
|
|
|
#print("\nRECV parsed", msg)
|
2019-02-06 07:27:31 -05:00
|
|
|
# XXX correct way to refer to MESSAGE
|
2019-02-23 03:07:13 -05:00
|
|
|
# if msg.header.type == 1:
|
|
|
|
# print("\nRECV parse", msg.payload.message.body.decode())
|
2019-02-06 07:27:31 -05:00
|
|
|
|
2019-02-06 10:38:34 -05:00
|
|
|
# XXX Only one receiver, this is a node not network
|
2019-02-23 02:57:52 -05:00
|
|
|
# XXX: Not populating? Why do we need this anyway?
|
|
|
|
# Well this is the sync node, so how self? node
|
|
|
|
# IF I recv something it is myself!
|
|
|
|
#receiver = self.nodes[0]
|
|
|
|
receiver = self.sync_node
|
2019-02-06 10:38:34 -05:00
|
|
|
# HEREATM
|
|
|
|
# How sender?
|
|
|
|
# TODO: Figure out how we know sender, assumes signed message
|
|
|
|
# inside payload? but if it isn't your own message then how work
|
|
|
|
# How does this currently work? How do we know from who it is?
|
|
|
|
# chat-id seems to be pubkey and some stuff
|
|
|
|
# sohuld be in signature sig
|
|
|
|
|
|
|
|
sender = sig.hex()
|
|
|
|
receiver.on_receive(sender, msg)
|
2019-02-06 07:27:31 -05:00
|
|
|
|
2019-02-06 06:27:51 -05:00
|
|
|
#print ""
|
2019-02-23 02:44:38 -05:00
|
|
|
#print("tick", self.time + 1)
|
2019-02-06 06:27:51 -05:00
|
|
|
#print "-----------"
|
2019-02-06 07:27:31 -05:00
|
|
|
|
|
|
|
# XXX: This is ugly, why is this ticking nodes?
|
2019-02-23 02:57:52 -05:00
|
|
|
# Also then don't tick
|
|
|
|
self.sync_node.tick()
|
2019-02-06 06:27:51 -05:00
|
|
|
self.time += 1
|
|
|
|
|
|
|
|
# NetworkSim stub
|
|
|
|
# def tick(self):
|
|
|
|
# if self.time in self.queue:
|
|
|
|
# # XXX: Should sender be here?
|
|
|
|
# for sender, receiver, msg in self.queue[self.time]:
|
|
|
|
# if random.random() < self.reliability:
|
|
|
|
# #print "*** message ok", sender.name, "->", receiver.name
|
|
|
|
# receiver.on_receive(sender, msg)
|
|
|
|
# #else:
|
|
|
|
# #print "*** message dropped", sender.name, "->", receiver.name
|
|
|
|
# #print ""
|
|
|
|
# print "tick", self.time + 1
|
|
|
|
# #print "-----------"
|
|
|
|
# for n in self.nodes:
|
|
|
|
# n.tick()
|
|
|
|
# self.time += 1
|
|
|
|
|
|
|
|
# sender id / pubkey not needed for now
|
|
|
|
# topic assumed to be hardcoded
|
2019-02-23 02:44:38 -05:00
|
|
|
# HEREATM, not sure if it works or not
|
|
|
|
# ok it sends, but not being picked up
|
|
|
|
# static-nodes same?
|
2019-02-06 06:27:51 -05:00
|
|
|
def send_message(self, sender_id, address_to, msg):
|
2019-02-23 02:57:52 -05:00
|
|
|
#print("*** (whisper-network) send_message to", address_to)
|
2019-02-06 07:27:31 -05:00
|
|
|
# XXX: Is this what we want to do?
|
|
|
|
payload = msg.SerializeToString()
|
2019-02-23 02:57:52 -05:00
|
|
|
#print("*** (whisper-network) send_message payload", payload)
|
2019-02-06 07:27:31 -05:00
|
|
|
#print("*** (whisper-network) send_message hex", self.web3.toHex(payload))
|
2019-02-06 06:27:51 -05:00
|
|
|
topic = self.topic
|
2019-02-06 07:27:31 -05:00
|
|
|
self.web3.shh.post({
|
2019-02-06 06:27:51 -05:00
|
|
|
'pubKey': address_to,
|
|
|
|
'topic': topic,
|
|
|
|
'powTarget': 2.01,
|
|
|
|
'powTime': 2,
|
|
|
|
'ttl': 10,
|
2019-02-06 10:38:34 -05:00
|
|
|
'sig': self.kId,
|
2019-02-06 07:27:31 -05:00
|
|
|
'payload': self.web3.toHex(payload)
|
2019-02-06 06:27:51 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
# NetworkSim stub
|
|
|
|
# def send_message(self, sender_id, receiver_id, message):
|
|
|
|
# #print "*** (network) send_message", sender_id, receiver_id
|
|
|
|
# # XXX: Assuming sender exists
|
|
|
|
# sender = self.peers[sender_id]
|
|
|
|
# receiver = self.peers[receiver_id]
|
|
|
|
# recv_time = self.time + self.latency_uniform_random()
|
|
|
|
# if recv_time not in self.queue:
|
|
|
|
# self.queue[recv_time] = []
|
|
|
|
# self.queue[recv_time].append((sender, receiver, message))
|
|
|
|
|
|
|
|
# XXX: This should be normal distribution or Poisson
|
|
|
|
# NetworkSim stub; not needed for Whisper probably
|
|
|
|
# def latency_uniform_random(self):
|
|
|
|
# # XXX: Hardcode for now, easier analyze
|
|
|
|
# latency = 1
|
|
|
|
# #latency = random.randint(1,3)
|
|
|
|
# return latency
|
|
|
|
|
|
|
|
|
|
|
|
|