-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathraiden.py
142 lines (120 loc) · 5.64 KB
/
raiden.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
import subprocess
import asyncio
import aiohttp
import logging
from os import PathLike
from const import TOKEN_ADDRESS
log = logging.getLogger()
class RaidenNode:
def __init__(self, address: str, api_endpoint: str, config_file: PathLike):
self.address = address
self.config_file = config_file
self.api_endpoint = api_endpoint
self._raiden_process = None
def __repr__(self):
return "{}<address={}, api-endpoint={}>".format(self.__class__.__name__, self.address,
self.api_endpoint)
def start(self):
# start the subprocess
# FIXME better stripping of http:// in api-address
raiden = "nodejs" + \
" /home/train/demo-train/light-client/raiden-cli/build/index.js" + \
" --ethNode " + "http://parity.goerli.ethnodes.brainbot.com:8545" + \
" --store " + f"./store_{self.address}" + \
" --password " + "raiden" + \
" --port " + self.api_endpoint[-4:] + \
" --privateKey " + f"./receiver/key_storage/UTC--{self.address} &"
log.info("Starting {}".format(self))
with open(f'./raiden_{self.address[:10]}.log', 'w') as logfile:
self._raiden_process = subprocess.Popen(
raiden,
shell=True,
stdout=logfile,
stderr=logfile
)
def stop(self):
log.info("Stopping {}".format(self))
self._raiden_process.terminate()
async def query_for_started(self):
url = self.api_endpoint + "/api/v1/address"
async with aiohttp.ClientSession() as session:
try:
async with session.get(url) as response:
data = await response.json()
if response.status == 200:
if data["our_address"] == self.address:
log.info(f"Succesfully started node {self.address}")
return True
raise ValueError("Address doesn't match expected address")
else:
# no 200 OK means the Raiden Node is somehow not available
# TODO handle different connection errors
log.info("Node not available: {}".format(self))
return False
# If Raiden not online it will raise a ClientConnectorError
except aiohttp.ClientConnectorError:
return False
async def ensure_is_started(self, poll_interval=1):
# poll the raiden api and check if the process is started
# use a future, multiple futures can be gathered
# runs infinitely, when the task/future isn't scheduled with a timeout!
while True:
if await self.query_for_started() is True:
return True
await asyncio.sleep(poll_interval)
async def query_for_payment_received(self, sender_address, token_address, nonce):
url = self.api_endpoint + "/api/v1/payments/{}/{}".format(token_address, sender_address)
async with aiohttp.ClientSession() as session:
async with session.get(url) as response:
data = await response.json()
if response.status == 200:
for event in data:
if event["event"] == "EventPaymentReceivedSuccess" and \
int(event["amount"]) == 1 and \
int(event["identifier"]) == nonce:
return True
# Event not found in event list:
return False
else:
# no 200 OK means the Raiden Node is somehow not available
# TODO handle different connection errors
# This should probably raise an exception, since the node is unhealthy or
# something in the query is wrong
log.info("Node not available: {}".format(self))
return False
async def ensure_payment_received(self,
sender_address,
token_address,
nonce,
poll_interval=1):
while True:
received = await self.query_for_payment_received(sender_address, token_address, nonce)
if received is True:
return True
await asyncio.sleep(poll_interval)
class RaidenNodeMock(RaidenNode):
# TODO instead of mocking the node in-process, rather use the FakeRaiden server in the background
# to allow actual network requests and the logic of the RaidenNode
"""
this is mainly to provide the same interface as a raiden node without actually starting
a subprocess and querying the raiden api
"""
def __init__(self, address: str, api_endpoint: str, config_file: PathLike):
super(RaidenNodeMock, self).__init__(address, api_endpoint, config_file)
self._started = False
def start(self):
# don't start the raiden subprocess, but
# TODO rather start the FakeRaiden server
self._started = True
def stop(self):
# TODO stop the FakeRaiden server
self._started = False
async def query_for_started(self):
# TODO remove
await asyncio.sleep(0.1)
return self._started
async def query_for_payment_received(self, sender_address, token_address, nonce):
# TODO remove
await asyncio.sleep(0.1)
# always say the payment was received for now
return True