2019-07-17 15:12:06 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2022-01-23 12:00:28 +00:00
|
|
|
# Copyright (c) 2019-2022 tecnovert
|
2019-07-17 15:12:06 +00:00
|
|
|
# Distributed under the MIT software license, see the accompanying
|
2020-10-30 08:55:45 +00:00
|
|
|
# file LICENSE or http://www.opensource.org/licenses/mit-license.php.
|
2019-07-17 15:12:06 +00:00
|
|
|
|
|
|
|
import os
|
2021-11-24 16:28:15 +00:00
|
|
|
import json
|
2019-07-17 15:12:06 +00:00
|
|
|
import traceback
|
|
|
|
import threading
|
|
|
|
import http.client
|
2022-10-11 20:09:20 +00:00
|
|
|
from urllib import parse
|
2019-07-17 15:12:06 +00:00
|
|
|
from http.server import BaseHTTPRequestHandler, HTTPServer
|
2019-07-26 10:36:44 +00:00
|
|
|
from jinja2 import Environment, PackageLoader
|
2019-07-25 20:15:08 +00:00
|
|
|
|
2019-07-26 10:07:25 +00:00
|
|
|
from . import __version__
|
2019-07-17 15:12:06 +00:00
|
|
|
from .util import (
|
2019-08-05 18:31:02 +00:00
|
|
|
dumpj,
|
2022-01-23 12:00:28 +00:00
|
|
|
ensure,
|
2021-01-18 22:52:05 +00:00
|
|
|
format_timestamp,
|
2019-07-17 15:12:06 +00:00
|
|
|
)
|
|
|
|
from .chainparams import (
|
|
|
|
Coins,
|
2022-01-23 12:00:28 +00:00
|
|
|
chainparams,
|
2019-07-17 15:12:06 +00:00
|
|
|
)
|
2021-10-18 18:48:48 +00:00
|
|
|
from .basicswap_util import (
|
2019-07-27 18:51:50 +00:00
|
|
|
strBidState,
|
|
|
|
strTxState,
|
2021-10-20 17:47:49 +00:00
|
|
|
strAddressType,
|
2019-07-17 15:12:06 +00:00
|
|
|
)
|
2022-10-10 21:44:14 +00:00
|
|
|
|
2020-11-27 22:20:35 +00:00
|
|
|
from .js_server import (
|
|
|
|
js_error,
|
2022-09-28 19:42:38 +00:00
|
|
|
js_url_to_function,
|
2020-11-27 22:20:35 +00:00
|
|
|
)
|
2022-03-26 22:08:15 +00:00
|
|
|
from .ui.util import (
|
2022-04-10 20:08:05 +00:00
|
|
|
getCoinName,
|
2021-01-27 23:55:42 +00:00
|
|
|
get_data_entry,
|
|
|
|
have_data_entry,
|
2022-04-10 20:08:05 +00:00
|
|
|
listAvailableCoins,
|
2020-11-30 14:29:40 +00:00
|
|
|
)
|
2022-05-23 21:51:06 +00:00
|
|
|
from .ui.page_automation import (
|
|
|
|
page_automation_strategies,
|
|
|
|
page_automation_strategy,
|
2022-05-31 22:38:50 +00:00
|
|
|
page_automation_strategy_new,
|
2022-05-23 21:51:06 +00:00
|
|
|
)
|
2022-10-10 21:44:14 +00:00
|
|
|
|
2022-10-05 17:30:52 +00:00
|
|
|
from .ui.page_bids import page_bids, page_bid
|
|
|
|
from .ui.page_offers import page_offers, page_offer, page_newoffer
|
2022-10-08 12:37:22 +00:00
|
|
|
from .ui.page_tor import page_tor, get_tor_established_state
|
2022-10-05 17:30:52 +00:00
|
|
|
from .ui.page_wallet import page_wallets, page_wallet
|
2019-07-17 15:12:06 +00:00
|
|
|
|
2019-07-26 10:36:44 +00:00
|
|
|
|
|
|
|
env = Environment(loader=PackageLoader('basicswap', 'templates'))
|
2019-07-26 16:07:32 +00:00
|
|
|
env.filters['formatts'] = format_timestamp
|
2019-07-25 20:15:08 +00:00
|
|
|
|
2019-07-17 15:12:06 +00:00
|
|
|
|
2021-10-19 18:59:18 +00:00
|
|
|
def validateTextInput(text, name, messages, max_length=None):
|
|
|
|
if max_length is not None and len(text) > max_length:
|
|
|
|
messages.append(f'Error: {name} is too long')
|
|
|
|
return False
|
2021-10-20 18:56:30 +00:00
|
|
|
if len(text) > 0 and all(c.isalnum() or c.isspace() for c in text) is False:
|
2021-10-19 18:59:18 +00:00
|
|
|
messages.append(f'Error: {name} must consist of only letters and digits')
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
2019-08-05 18:31:02 +00:00
|
|
|
def extractDomain(url):
|
|
|
|
return url.split('://', 1)[1].split('/', 1)[0]
|
|
|
|
|
|
|
|
|
|
|
|
def listAvailableExplorers(swap_client):
|
|
|
|
explorers = []
|
|
|
|
for c in Coins:
|
2021-02-07 10:01:58 +00:00
|
|
|
if c not in chainparams:
|
2021-02-06 22:35:12 +00:00
|
|
|
continue
|
2019-08-05 18:31:02 +00:00
|
|
|
for i, e in enumerate(swap_client.coin_clients[c]['explorers']):
|
|
|
|
explorers.append(('{}_{}'.format(int(c), i), swap_client.coin_clients[c]['name'].capitalize() + ' - ' + extractDomain(e.base_url)))
|
|
|
|
return explorers
|
|
|
|
|
|
|
|
|
|
|
|
def listExplorerActions(swap_client):
|
|
|
|
actions = [('height', 'Chain Height'),
|
|
|
|
('block', 'Get Block'),
|
|
|
|
('tx', 'Get Transaction'),
|
|
|
|
('balance', 'Address Balance'),
|
|
|
|
('unspent', 'List Unspent')]
|
|
|
|
return actions
|
|
|
|
|
2022-10-11 05:55:35 +00:00
|
|
|
|
2019-07-17 15:12:06 +00:00
|
|
|
class HttpHandler(BaseHTTPRequestHandler):
|
|
|
|
|
2022-10-11 20:09:20 +00:00
|
|
|
def generate_form_id(self):
|
|
|
|
return os.urandom(8).hex()
|
|
|
|
|
2019-07-29 10:14:46 +00:00
|
|
|
def checkForm(self, post_string, name, messages):
|
|
|
|
if post_string == '':
|
|
|
|
return None
|
2022-10-11 20:09:20 +00:00
|
|
|
form_data = parse.parse_qs(post_string)
|
2019-07-29 10:14:46 +00:00
|
|
|
form_id = form_data[b'formid'][0].decode('utf-8')
|
|
|
|
if self.server.last_form_id.get(name, None) == form_id:
|
|
|
|
messages.append('Prevented double submit for form {}.'.format(form_id))
|
|
|
|
else:
|
|
|
|
self.server.last_form_id[name] = form_id
|
|
|
|
return form_data
|
|
|
|
|
2022-07-31 17:33:01 +00:00
|
|
|
def render_template(self, template, args_dict):
|
|
|
|
swap_client = self.server.swap_client
|
|
|
|
if swap_client.ws_server:
|
|
|
|
args_dict['ws_url'] = swap_client.ws_server.url
|
2022-09-15 19:54:15 +00:00
|
|
|
if swap_client.debug:
|
|
|
|
args_dict['debug_mode'] = True
|
|
|
|
if swap_client.debug_ui:
|
|
|
|
args_dict['debug_ui_mode'] = True
|
2022-10-08 12:37:22 +00:00
|
|
|
if swap_client.use_tor_proxy:
|
|
|
|
args_dict['use_tor_proxy'] = True
|
|
|
|
# TODO: Cache value?
|
|
|
|
try:
|
|
|
|
args_dict['tor_established'] = True if get_tor_established_state(swap_client) == '1' else False
|
|
|
|
except Exception:
|
|
|
|
if swap_client.debug:
|
|
|
|
swap_client.log.error(traceback.format_exc())
|
|
|
|
|
2022-10-11 20:09:20 +00:00
|
|
|
self.putHeaders(200, 'text/html')
|
2022-07-31 17:33:01 +00:00
|
|
|
return bytes(template.render(
|
|
|
|
title=self.server.title,
|
|
|
|
h2=self.server.title,
|
2022-10-11 20:09:20 +00:00
|
|
|
form_id=self.generate_form_id(),
|
2022-07-31 17:33:01 +00:00
|
|
|
**args_dict,
|
|
|
|
), 'UTF-8')
|
|
|
|
|
2022-09-15 19:54:15 +00:00
|
|
|
def render_simple_template(self, template, args_dict):
|
|
|
|
swap_client = self.server.swap_client
|
|
|
|
return bytes(template.render(
|
|
|
|
title=self.server.title,
|
|
|
|
**args_dict,
|
|
|
|
), 'UTF-8')
|
|
|
|
|
2022-10-11 06:22:13 +00:00
|
|
|
def page_info(self, info_str, post_string=None):
|
2022-09-15 19:54:15 +00:00
|
|
|
template = env.get_template('info.html')
|
2022-10-10 21:44:14 +00:00
|
|
|
swap_client = self.server.swap_client
|
|
|
|
summary = swap_client.getSummary()
|
|
|
|
return self.render_template(template, {
|
2022-09-15 19:54:15 +00:00
|
|
|
'title_str': 'BasicSwap Info',
|
2022-09-23 20:07:41 +00:00
|
|
|
'message_str': info_str,
|
2022-10-10 21:44:14 +00:00
|
|
|
'summary': summary,
|
2022-09-15 19:54:15 +00:00
|
|
|
})
|
|
|
|
|
2022-10-11 06:22:13 +00:00
|
|
|
def page_error(self, error_str, post_string=None):
|
2022-09-15 19:54:15 +00:00
|
|
|
template = env.get_template('error.html')
|
2022-10-10 21:44:14 +00:00
|
|
|
swap_client = self.server.swap_client
|
|
|
|
summary = swap_client.getSummary()
|
|
|
|
return self.render_template(template, {
|
2022-09-15 19:54:15 +00:00
|
|
|
'title_str': 'BasicSwap Error',
|
|
|
|
'message_str': error_str,
|
2022-10-10 21:44:14 +00:00
|
|
|
'summary': summary,
|
2022-09-15 19:54:15 +00:00
|
|
|
})
|
|
|
|
|
2019-08-05 18:31:02 +00:00
|
|
|
def page_explorers(self, url_split, post_string):
|
|
|
|
swap_client = self.server.swap_client
|
2022-10-10 21:44:14 +00:00
|
|
|
summary = swap_client.getSummary()
|
2019-08-05 18:31:02 +00:00
|
|
|
|
|
|
|
result = None
|
|
|
|
explorer = -1
|
|
|
|
action = -1
|
|
|
|
messages = []
|
|
|
|
form_data = self.checkForm(post_string, 'explorers', messages)
|
|
|
|
if form_data:
|
|
|
|
|
|
|
|
explorer = form_data[b'explorer'][0].decode('utf-8')
|
|
|
|
action = form_data[b'action'][0].decode('utf-8')
|
|
|
|
|
2019-08-05 22:04:40 +00:00
|
|
|
args = '' if b'args' not in form_data else form_data[b'args'][0].decode('utf-8')
|
2019-08-05 18:31:02 +00:00
|
|
|
try:
|
|
|
|
c, e = explorer.split('_')
|
|
|
|
exp = swap_client.coin_clients[Coins(int(c))]['explorers'][int(e)]
|
|
|
|
if action == 'height':
|
|
|
|
result = str(exp.getChainHeight())
|
|
|
|
elif action == 'block':
|
|
|
|
result = dumpj(exp.getBlock(args))
|
|
|
|
elif action == 'tx':
|
|
|
|
result = dumpj(exp.getTransaction(args))
|
|
|
|
elif action == 'balance':
|
|
|
|
result = dumpj(exp.getBalance(args))
|
|
|
|
elif action == 'unspent':
|
|
|
|
result = dumpj(exp.lookupUnspentByAddress(args))
|
|
|
|
else:
|
|
|
|
result = 'Unknown action'
|
|
|
|
except Exception as ex:
|
|
|
|
result = str(ex)
|
|
|
|
|
|
|
|
template = env.get_template('explorers.html')
|
2022-07-31 17:33:01 +00:00
|
|
|
return self.render_template(template, {
|
|
|
|
'explorers': listAvailableExplorers(swap_client),
|
|
|
|
'explorer': explorer,
|
|
|
|
'actions': listExplorerActions(swap_client),
|
|
|
|
'action': action,
|
2022-10-10 21:44:14 +00:00
|
|
|
'result': result,
|
|
|
|
'summary': summary,
|
2022-07-31 17:33:01 +00:00
|
|
|
})
|
2019-08-05 18:31:02 +00:00
|
|
|
|
2019-07-27 21:14:28 +00:00
|
|
|
def page_rpc(self, url_split, post_string):
|
|
|
|
swap_client = self.server.swap_client
|
2022-10-10 21:44:14 +00:00
|
|
|
summary = swap_client.getSummary()
|
2019-07-27 21:14:28 +00:00
|
|
|
|
|
|
|
result = None
|
2019-07-31 18:21:41 +00:00
|
|
|
coin_type = -1
|
2022-07-06 11:16:18 +00:00
|
|
|
coin_id = -1
|
2019-07-27 21:14:28 +00:00
|
|
|
messages = []
|
2019-07-29 10:14:46 +00:00
|
|
|
form_data = self.checkForm(post_string, 'rpc', messages)
|
|
|
|
if form_data:
|
|
|
|
try:
|
2022-07-06 11:16:18 +00:00
|
|
|
coin_id = int(form_data[b'coin_type'][0])
|
|
|
|
if coin_id in (-2, -3, -4):
|
|
|
|
coin_type = Coins(Coins.XMR)
|
|
|
|
else:
|
|
|
|
coin_type = Coins(coin_id)
|
2019-07-29 10:14:46 +00:00
|
|
|
except Exception:
|
|
|
|
raise ValueError('Unknown Coin Type')
|
2019-07-27 21:14:28 +00:00
|
|
|
|
2019-07-29 10:14:46 +00:00
|
|
|
cmd = form_data[b'cmd'][0].decode('utf-8')
|
2021-11-24 16:28:15 +00:00
|
|
|
|
2019-07-29 10:14:46 +00:00
|
|
|
try:
|
2021-11-24 16:28:15 +00:00
|
|
|
if coin_type == Coins.XMR:
|
|
|
|
ci = swap_client.ci(coin_type)
|
|
|
|
arr = cmd.split(None, 1)
|
|
|
|
method = arr[0]
|
|
|
|
params = json.loads(arr[1]) if len(arr) > 1 else []
|
2022-07-06 11:16:18 +00:00
|
|
|
if coin_id == -4:
|
|
|
|
rv = ci.rpc_wallet_cb(method, params)
|
|
|
|
elif coin_id == -3:
|
|
|
|
rv = ci.rpc_cb(method, params)
|
|
|
|
elif coin_id == -2:
|
|
|
|
if params == []:
|
|
|
|
params = None
|
|
|
|
rv = ci.rpc_cb2(method, params)
|
|
|
|
else:
|
|
|
|
raise ValueError('Unknown XMR RPC variant')
|
|
|
|
result = json.dumps(rv, indent=4)
|
2021-11-24 16:28:15 +00:00
|
|
|
else:
|
|
|
|
result = cmd + '\n' + swap_client.callcoincli(coin_type, cmd)
|
2019-07-29 10:14:46 +00:00
|
|
|
except Exception as ex:
|
|
|
|
result = str(ex)
|
2022-07-06 11:16:18 +00:00
|
|
|
if self.server.swap_client.debug is True:
|
|
|
|
self.server.swap_client.log.error(traceback.format_exc())
|
2019-07-27 21:14:28 +00:00
|
|
|
|
|
|
|
template = env.get_template('rpc.html')
|
2022-07-06 11:16:18 +00:00
|
|
|
|
|
|
|
coins = listAvailableCoins(swap_client, with_variants=False)
|
|
|
|
coins = [c for c in coins if c[0] != Coins.XMR]
|
|
|
|
coins.append((-2, 'Monero'))
|
|
|
|
coins.append((-3, 'Monero JSON'))
|
|
|
|
coins.append((-4, 'Monero Wallet'))
|
|
|
|
|
2022-07-31 17:33:01 +00:00
|
|
|
return self.render_template(template, {
|
|
|
|
'coins': coins,
|
|
|
|
'coin_type': coin_id,
|
|
|
|
'result': result,
|
|
|
|
'messages': messages,
|
2022-10-10 21:44:14 +00:00
|
|
|
'summary': summary,
|
2022-07-31 17:33:01 +00:00
|
|
|
})
|
2021-11-24 21:54:31 +00:00
|
|
|
|
|
|
|
def page_debug(self, url_split, post_string):
|
|
|
|
swap_client = self.server.swap_client
|
2022-10-10 21:44:14 +00:00
|
|
|
summary = swap_client.getSummary()
|
2021-11-24 21:54:31 +00:00
|
|
|
|
|
|
|
result = None
|
|
|
|
messages = []
|
|
|
|
form_data = self.checkForm(post_string, 'wallets', messages)
|
|
|
|
if form_data:
|
|
|
|
if have_data_entry(form_data, 'reinit_xmr'):
|
|
|
|
try:
|
|
|
|
swap_client.initialiseWallet(Coins.XMR)
|
|
|
|
messages.append('Done.')
|
|
|
|
except Exception as a:
|
|
|
|
messages.append('Failed.')
|
|
|
|
|
|
|
|
template = env.get_template('debug.html')
|
2022-07-31 17:33:01 +00:00
|
|
|
return self.render_template(template, {
|
|
|
|
'messages': messages,
|
|
|
|
'result': result,
|
2022-10-10 21:44:14 +00:00
|
|
|
'summary': summary,
|
2022-07-31 17:33:01 +00:00
|
|
|
})
|
2019-07-27 21:14:28 +00:00
|
|
|
|
2019-07-17 15:12:06 +00:00
|
|
|
def page_active(self, url_split, post_string):
|
|
|
|
swap_client = self.server.swap_client
|
|
|
|
active_swaps = swap_client.listSwapsInProgress()
|
2022-10-10 21:44:14 +00:00
|
|
|
summary = swap_client.getSummary()
|
2019-07-17 15:12:06 +00:00
|
|
|
|
2019-07-25 21:30:27 +00:00
|
|
|
template = env.get_template('active.html')
|
2022-07-31 17:33:01 +00:00
|
|
|
return self.render_template(template, {
|
|
|
|
'refresh': 30,
|
|
|
|
'active_swaps': [(s[0].hex(), s[1], strBidState(s[2]), strTxState(s[3]), strTxState(s[4])) for s in active_swaps],
|
2022-10-10 21:44:14 +00:00
|
|
|
'summary': summary,
|
2022-07-31 17:33:01 +00:00
|
|
|
})
|
2019-07-17 15:12:06 +00:00
|
|
|
|
2019-08-05 22:04:40 +00:00
|
|
|
def page_settings(self, url_split, post_string):
|
|
|
|
swap_client = self.server.swap_client
|
2022-10-10 21:44:14 +00:00
|
|
|
summary = swap_client.getSummary()
|
2019-08-05 22:04:40 +00:00
|
|
|
|
|
|
|
messages = []
|
|
|
|
form_data = self.checkForm(post_string, 'settings', messages)
|
|
|
|
if form_data:
|
|
|
|
for name, c in swap_client.settings['chainclients'].items():
|
2021-11-13 21:13:54 +00:00
|
|
|
if have_data_entry(form_data, 'apply_' + name):
|
|
|
|
data = {'lookups': get_data_entry(form_data, 'lookups_' + name)}
|
2020-12-22 11:21:25 +00:00
|
|
|
if name == 'monero':
|
2021-11-13 21:13:54 +00:00
|
|
|
data['fee_priority'] = int(get_data_entry(form_data, 'fee_priority_' + name))
|
|
|
|
data['manage_daemon'] = True if get_data_entry(form_data, 'managedaemon_' + name) == 'true' else False
|
|
|
|
data['rpchost'] = get_data_entry(form_data, 'rpchost_' + name)
|
|
|
|
data['rpcport'] = int(get_data_entry(form_data, 'rpcport_' + name))
|
|
|
|
data['remotedaemonurls'] = get_data_entry(form_data, 'remotedaemonurls_' + name)
|
|
|
|
data['automatically_select_daemon'] = True if get_data_entry(form_data, 'autosetdaemon_' + name) == 'true' else False
|
2021-01-18 22:52:05 +00:00
|
|
|
else:
|
2021-11-13 21:13:54 +00:00
|
|
|
data['conf_target'] = int(get_data_entry(form_data, 'conf_target_' + name))
|
2022-04-10 22:11:51 +00:00
|
|
|
if name == 'particl':
|
|
|
|
data['anon_tx_ring_size'] = int(get_data_entry(form_data, 'rct_ring_size_' + name))
|
2020-12-22 11:21:25 +00:00
|
|
|
|
2021-11-13 21:13:54 +00:00
|
|
|
settings_changed, suggest_reboot = swap_client.editSettings(name, data)
|
|
|
|
if settings_changed is True:
|
2021-01-18 22:52:05 +00:00
|
|
|
messages.append('Settings applied.')
|
2021-11-13 21:13:54 +00:00
|
|
|
if suggest_reboot is True:
|
|
|
|
messages.append('Please restart BasicSwap.')
|
|
|
|
elif have_data_entry(form_data, 'enable_' + name):
|
2021-02-14 10:12:41 +00:00
|
|
|
swap_client.enableCoin(name)
|
|
|
|
messages.append(name.capitalize() + ' enabled, shutting down.')
|
|
|
|
swap_client.stopRunning()
|
2021-11-13 21:13:54 +00:00
|
|
|
elif have_data_entry(form_data, 'disable_' + name):
|
2021-02-14 10:12:41 +00:00
|
|
|
swap_client.disableCoin(name)
|
|
|
|
messages.append(name.capitalize() + ' disabled, shutting down.')
|
|
|
|
swap_client.stopRunning()
|
2019-08-05 22:04:40 +00:00
|
|
|
chains_formatted = []
|
|
|
|
|
2021-11-13 21:13:54 +00:00
|
|
|
sorted_names = sorted(swap_client.settings['chainclients'].keys())
|
|
|
|
for name in sorted_names:
|
|
|
|
c = swap_client.settings['chainclients'][name]
|
2019-08-05 22:04:40 +00:00
|
|
|
chains_formatted.append({
|
|
|
|
'name': name,
|
2021-02-14 10:12:41 +00:00
|
|
|
'lookups': c.get('chain_lookups', 'local'),
|
|
|
|
'manage_daemon': c.get('manage_daemon', 'Unknown'),
|
|
|
|
'connection_type': c.get('connection_type', 'Unknown'),
|
2019-08-05 22:04:40 +00:00
|
|
|
})
|
2020-12-22 11:21:25 +00:00
|
|
|
if name == 'monero':
|
|
|
|
chains_formatted[-1]['fee_priority'] = c.get('fee_priority', 0)
|
2021-02-14 10:12:41 +00:00
|
|
|
chains_formatted[-1]['manage_wallet_daemon'] = c.get('manage_wallet_daemon', 'Unknown')
|
2021-11-13 21:13:54 +00:00
|
|
|
chains_formatted[-1]['rpchost'] = c.get('rpchost', 'localhost')
|
|
|
|
chains_formatted[-1]['rpcport'] = int(c.get('rpcport', 18081))
|
|
|
|
chains_formatted[-1]['remotedaemonurls'] = '\n'.join(c.get('remote_daemon_urls', []))
|
|
|
|
chains_formatted[-1]['autosetdaemon'] = c.get('automatically_select_daemon', False)
|
2021-01-18 22:52:05 +00:00
|
|
|
else:
|
|
|
|
chains_formatted[-1]['conf_target'] = c.get('conf_target', 2)
|
2022-04-10 22:11:51 +00:00
|
|
|
|
|
|
|
if name == 'particl':
|
|
|
|
chains_formatted[-1]['anon_tx_ring_size'] = c.get('anon_tx_ring_size', 12)
|
|
|
|
else:
|
2021-02-14 10:12:41 +00:00
|
|
|
if c.get('connection_type', 'Unknown') == 'none':
|
|
|
|
if 'connection_type_prev' in c:
|
|
|
|
chains_formatted[-1]['can_reenable'] = True
|
|
|
|
else:
|
|
|
|
chains_formatted[-1]['can_disable'] = True
|
2019-08-05 22:04:40 +00:00
|
|
|
|
|
|
|
template = env.get_template('settings.html')
|
2022-07-31 17:33:01 +00:00
|
|
|
return self.render_template(template, {
|
|
|
|
'messages': messages,
|
|
|
|
'chains': chains_formatted,
|
2022-10-10 21:44:14 +00:00
|
|
|
'summary': summary,
|
2022-07-31 17:33:01 +00:00
|
|
|
})
|
2019-08-05 22:04:40 +00:00
|
|
|
|
2019-07-17 15:12:06 +00:00
|
|
|
def page_watched(self, url_split, post_string):
|
|
|
|
swap_client = self.server.swap_client
|
|
|
|
watched_outputs, last_scanned = swap_client.listWatchedOutputs()
|
2022-10-10 21:44:14 +00:00
|
|
|
summary = swap_client.getSummary()
|
2019-07-25 21:30:27 +00:00
|
|
|
|
2019-07-25 21:15:35 +00:00
|
|
|
template = env.get_template('watched.html')
|
2022-07-31 17:33:01 +00:00
|
|
|
return self.render_template(template, {
|
|
|
|
'refresh': 30,
|
|
|
|
'last_scanned': [(getCoinName(ls[0]), ls[1]) for ls in last_scanned],
|
|
|
|
'watched_outputs': [(wo[1].hex(), getCoinName(wo[0]), wo[2], wo[3], int(wo[4])) for wo in watched_outputs],
|
2022-10-10 21:44:14 +00:00
|
|
|
'summary': summary,
|
2022-07-31 17:33:01 +00:00
|
|
|
})
|
2019-07-17 15:12:06 +00:00
|
|
|
|
2021-10-19 18:59:18 +00:00
|
|
|
def page_smsgaddresses(self, url_split, post_string):
|
|
|
|
swap_client = self.server.swap_client
|
2022-10-10 21:44:14 +00:00
|
|
|
summary = swap_client.getSummary()
|
2021-10-19 18:59:18 +00:00
|
|
|
|
|
|
|
page_data = {}
|
|
|
|
messages = []
|
|
|
|
smsgaddresses = []
|
|
|
|
|
|
|
|
listaddresses = True
|
|
|
|
form_data = self.checkForm(post_string, 'smsgaddresses', messages)
|
|
|
|
if form_data:
|
|
|
|
edit_address_id = None
|
|
|
|
for key in form_data:
|
|
|
|
if key.startswith(b'editaddr_'):
|
|
|
|
edit_address_id = int(key.split(b'_')[1])
|
|
|
|
break
|
|
|
|
if edit_address_id is not None:
|
|
|
|
listaddresses = False
|
|
|
|
page_data['edit_address'] = edit_address_id
|
|
|
|
page_data['addr_data'] = swap_client.listAllSMSGAddresses(addr_id=edit_address_id)[0]
|
|
|
|
elif b'saveaddr' in form_data:
|
|
|
|
edit_address_id = int(form_data[b'edit_address_id'][0].decode('utf-8'))
|
|
|
|
edit_addr = form_data[b'edit_address'][0].decode('utf-8')
|
|
|
|
active_ind = int(form_data[b'active_ind'][0].decode('utf-8'))
|
2022-01-23 12:00:28 +00:00
|
|
|
ensure(active_ind in (0, 1), 'Invalid sort by')
|
2021-10-19 18:59:18 +00:00
|
|
|
addressnote = '' if b'addressnote' not in form_data else form_data[b'addressnote'][0].decode('utf-8')
|
|
|
|
if not validateTextInput(addressnote, 'Address note', messages, max_length=30):
|
|
|
|
listaddresses = False
|
|
|
|
page_data['edit_address'] = edit_address_id
|
|
|
|
else:
|
|
|
|
swap_client.editSMSGAddress(edit_addr, active_ind=active_ind, addressnote=addressnote)
|
|
|
|
messages.append(f'Edited address {edit_addr}')
|
|
|
|
elif b'shownewaddr' in form_data:
|
|
|
|
listaddresses = False
|
|
|
|
page_data['new_address'] = True
|
2021-10-20 17:47:49 +00:00
|
|
|
elif b'showaddaddr' in form_data:
|
|
|
|
listaddresses = False
|
|
|
|
page_data['new_send_address'] = True
|
2021-10-19 18:59:18 +00:00
|
|
|
elif b'createnewaddr' in form_data:
|
|
|
|
addressnote = '' if b'addressnote' not in form_data else form_data[b'addressnote'][0].decode('utf-8')
|
|
|
|
if not validateTextInput(addressnote, 'Address note', messages, max_length=30):
|
|
|
|
listaddresses = False
|
|
|
|
page_data['new_address'] = True
|
|
|
|
else:
|
2021-10-20 17:47:49 +00:00
|
|
|
new_addr, pubkey = swap_client.newSMSGAddress(addressnote=addressnote)
|
|
|
|
messages.append(f'Created address {new_addr}, pubkey {pubkey}')
|
|
|
|
elif b'createnewsendaddr' in form_data:
|
|
|
|
pubkey_hex = form_data[b'addresspubkey'][0].decode('utf-8')
|
|
|
|
addressnote = '' if b'addressnote' not in form_data else form_data[b'addressnote'][0].decode('utf-8')
|
|
|
|
if not validateTextInput(addressnote, 'Address note', messages, max_length=30) or \
|
|
|
|
not validateTextInput(pubkey_hex, 'Pubkey', messages, max_length=66):
|
|
|
|
listaddresses = False
|
|
|
|
page_data['new_send_address'] = True
|
|
|
|
else:
|
|
|
|
new_addr = swap_client.addSMSGAddress(pubkey_hex, addressnote=addressnote)
|
|
|
|
messages.append(f'Added address {new_addr}')
|
2021-10-19 18:59:18 +00:00
|
|
|
|
|
|
|
if listaddresses is True:
|
|
|
|
smsgaddresses = swap_client.listAllSMSGAddresses()
|
|
|
|
network_addr = swap_client.network_addr
|
|
|
|
|
|
|
|
for addr in smsgaddresses:
|
2021-10-20 17:47:49 +00:00
|
|
|
addr['type'] = strAddressType(addr['type'])
|
2021-10-19 18:59:18 +00:00
|
|
|
|
|
|
|
template = env.get_template('smsgaddresses.html')
|
2022-07-31 17:33:01 +00:00
|
|
|
return self.render_template(template, {
|
|
|
|
'messages': messages,
|
|
|
|
'data': page_data,
|
|
|
|
'smsgaddresses': smsgaddresses,
|
|
|
|
'network_addr': network_addr,
|
2022-10-10 21:44:14 +00:00
|
|
|
'summary': summary,
|
2022-07-31 17:33:01 +00:00
|
|
|
})
|
2021-10-19 18:59:18 +00:00
|
|
|
|
2021-11-14 23:26:43 +00:00
|
|
|
def page_identity(self, url_split, post_string):
|
2022-01-23 12:00:28 +00:00
|
|
|
ensure(len(url_split) > 2, 'Address not specified')
|
2021-11-14 23:26:43 +00:00
|
|
|
identity_address = url_split[2]
|
|
|
|
swap_client = self.server.swap_client
|
2022-10-10 21:44:14 +00:00
|
|
|
summary = swap_client.getSummary()
|
2021-11-14 23:26:43 +00:00
|
|
|
|
|
|
|
page_data = {'identity_address': identity_address}
|
|
|
|
messages = []
|
2021-12-05 23:06:34 +00:00
|
|
|
form_data = self.checkForm(post_string, 'identity', messages)
|
|
|
|
if form_data:
|
|
|
|
if have_data_entry(form_data, 'edit'):
|
|
|
|
page_data['show_edit_form'] = True
|
|
|
|
if have_data_entry(form_data, 'apply'):
|
|
|
|
new_label = get_data_entry(form_data, 'label')
|
|
|
|
|
|
|
|
try:
|
|
|
|
swap_client.updateIdentity(identity_address, new_label)
|
|
|
|
messages.append('Updated')
|
|
|
|
except Exception as e:
|
2021-12-19 17:30:55 +00:00
|
|
|
messages.append('Error')
|
2021-11-14 23:26:43 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
identity = swap_client.getIdentity(identity_address)
|
|
|
|
if identity is None:
|
|
|
|
raise ValueError('Unknown address')
|
2021-12-05 23:06:34 +00:00
|
|
|
page_data['label'] = identity.label
|
2021-11-14 23:26:43 +00:00
|
|
|
page_data['num_sent_bids_successful'] = identity.num_sent_bids_successful
|
|
|
|
page_data['num_recv_bids_successful'] = identity.num_recv_bids_successful
|
|
|
|
page_data['num_sent_bids_rejected'] = identity.num_sent_bids_rejected
|
|
|
|
page_data['num_recv_bids_rejected'] = identity.num_recv_bids_rejected
|
|
|
|
page_data['num_sent_bids_failed'] = identity.num_sent_bids_failed
|
|
|
|
page_data['num_recv_bids_failed'] = identity.num_recv_bids_failed
|
|
|
|
except Exception as e:
|
|
|
|
messages.append(e)
|
|
|
|
|
|
|
|
template = env.get_template('identity.html')
|
2022-07-31 17:33:01 +00:00
|
|
|
return self.render_template(template, {
|
|
|
|
'messages': messages,
|
|
|
|
'data': page_data,
|
2022-10-10 21:44:14 +00:00
|
|
|
'summary': summary,
|
2022-07-31 17:33:01 +00:00
|
|
|
})
|
2021-11-14 23:26:43 +00:00
|
|
|
|
2019-08-05 18:31:02 +00:00
|
|
|
def page_shutdown(self, url_split, post_string):
|
|
|
|
swap_client = self.server.swap_client
|
2021-11-22 20:24:48 +00:00
|
|
|
|
|
|
|
if len(url_split) > 2:
|
|
|
|
token = url_split[2]
|
|
|
|
expect_token = self.server.session_tokens.get('shutdown', None)
|
|
|
|
if token != expect_token:
|
|
|
|
return self.page_info('Unexpected token, still running.')
|
|
|
|
|
2019-08-05 18:31:02 +00:00
|
|
|
swap_client.stopRunning()
|
|
|
|
|
|
|
|
return self.page_info('Shutting down')
|
|
|
|
|
2019-07-17 15:12:06 +00:00
|
|
|
def page_index(self, url_split):
|
|
|
|
swap_client = self.server.swap_client
|
|
|
|
summary = swap_client.getSummary()
|
|
|
|
|
2021-11-22 20:24:48 +00:00
|
|
|
shutdown_token = os.urandom(8).hex()
|
|
|
|
self.server.session_tokens['shutdown'] = shutdown_token
|
|
|
|
|
2019-07-25 20:15:08 +00:00
|
|
|
template = env.get_template('index.html')
|
2022-07-31 17:33:01 +00:00
|
|
|
return self.render_template(template, {
|
|
|
|
'refresh': 30,
|
|
|
|
'version': __version__,
|
|
|
|
'summary': summary,
|
|
|
|
'use_tor_proxy': swap_client.use_tor_proxy,
|
|
|
|
'shutdown_token': shutdown_token
|
|
|
|
})
|
2019-07-17 15:12:06 +00:00
|
|
|
|
2021-01-02 16:16:44 +00:00
|
|
|
def page_404(self, url_split):
|
2022-10-11 09:38:20 +00:00
|
|
|
swap_client = self.server.swap_client
|
|
|
|
summary = swap_client.getSummary()
|
2021-01-02 16:16:44 +00:00
|
|
|
template = env.get_template('404.html')
|
2022-10-11 09:38:20 +00:00
|
|
|
return self.render_template(template, {
|
|
|
|
'summary': summary,
|
|
|
|
})
|
2021-01-02 16:16:44 +00:00
|
|
|
|
2019-07-17 15:12:06 +00:00
|
|
|
def putHeaders(self, status_code, content_type):
|
|
|
|
self.send_response(status_code)
|
|
|
|
if self.server.allow_cors:
|
|
|
|
self.send_header('Access-Control-Allow-Origin', '*')
|
2021-01-27 23:55:42 +00:00
|
|
|
self.send_header('Content-Type', content_type)
|
2019-07-17 15:12:06 +00:00
|
|
|
self.end_headers()
|
|
|
|
|
2021-01-27 23:55:42 +00:00
|
|
|
def handle_http(self, status_code, path, post_string='', is_json=False):
|
2022-10-11 20:09:20 +00:00
|
|
|
parsed = parse.urlparse(self.path)
|
2022-08-03 21:59:57 +00:00
|
|
|
url_split = parsed.path.split('/')
|
|
|
|
if post_string == '' and len(parsed.query) > 0:
|
|
|
|
post_string = parsed.query
|
2019-07-17 15:12:06 +00:00
|
|
|
if len(url_split) > 1 and url_split[1] == 'json':
|
|
|
|
try:
|
|
|
|
self.putHeaders(status_code, 'text/plain')
|
2022-09-28 19:42:38 +00:00
|
|
|
func = js_url_to_function(url_split)
|
2021-01-27 23:55:42 +00:00
|
|
|
return func(self, url_split, post_string, is_json)
|
2019-07-27 17:26:06 +00:00
|
|
|
except Exception as ex:
|
2019-08-15 19:14:28 +00:00
|
|
|
if self.server.swap_client.debug is True:
|
2021-12-16 08:44:10 +00:00
|
|
|
self.server.swap_client.log.error(traceback.format_exc())
|
2020-11-27 22:20:35 +00:00
|
|
|
return js_error(self, str(ex))
|
2021-01-02 16:16:44 +00:00
|
|
|
|
|
|
|
if len(url_split) > 1 and url_split[1] == 'static':
|
|
|
|
try:
|
|
|
|
static_path = os.path.join(os.path.dirname(__file__), 'static')
|
|
|
|
|
2022-07-30 17:00:53 +00:00
|
|
|
if len(url_split) > 3 and url_split[2] == 'sequence_diagrams':
|
|
|
|
with open(os.path.join(static_path, 'sequence_diagrams', url_split[3]), 'rb') as fp:
|
|
|
|
self.putHeaders(status_code, 'image/svg+xml')
|
2021-01-02 16:16:44 +00:00
|
|
|
return fp.read()
|
2022-07-30 17:00:53 +00:00
|
|
|
elif len(url_split) > 3 and url_split[2] == 'images':
|
|
|
|
filename = os.path.join(*url_split[3:])
|
|
|
|
_, extension = os.path.splitext(filename)
|
2022-07-31 17:33:01 +00:00
|
|
|
mime_type = {
|
|
|
|
'.svg': 'image/svg+xml',
|
|
|
|
'.png': 'image/png',
|
|
|
|
'.jpg': 'image/jpeg',
|
2022-10-10 21:44:14 +00:00
|
|
|
'.gif': 'image/gif',
|
|
|
|
'.ico': 'image/x-icon',
|
2022-07-31 17:33:01 +00:00
|
|
|
}.get(extension, '')
|
2022-07-30 17:00:53 +00:00
|
|
|
if mime_type == '':
|
|
|
|
raise ValueError('Unknown file type ' + filename)
|
|
|
|
with open(os.path.join(static_path, 'images', filename), 'rb') as fp:
|
|
|
|
self.putHeaders(status_code, mime_type)
|
2021-01-02 16:16:44 +00:00
|
|
|
return fp.read()
|
2022-07-30 17:00:53 +00:00
|
|
|
elif len(url_split) > 3 and url_split[2] == 'css':
|
|
|
|
filename = os.path.join(*url_split[3:])
|
|
|
|
with open(os.path.join(static_path, 'css', filename), 'rb') as fp:
|
|
|
|
self.putHeaders(status_code, 'text/css; charset=utf-8')
|
2022-06-28 23:45:06 +00:00
|
|
|
return fp.read()
|
2022-07-31 21:40:58 +00:00
|
|
|
elif len(url_split) > 3 and url_split[2] == 'js':
|
|
|
|
filename = os.path.join(*url_split[3:])
|
|
|
|
with open(os.path.join(static_path, 'js', filename), 'rb') as fp:
|
|
|
|
self.putHeaders(status_code, 'application/javascript')
|
|
|
|
return fp.read()
|
2021-01-02 16:16:44 +00:00
|
|
|
else:
|
|
|
|
self.putHeaders(status_code, 'text/html')
|
|
|
|
return self.page_404(url_split)
|
2022-07-30 17:00:53 +00:00
|
|
|
except FileNotFoundError:
|
2021-01-02 16:16:44 +00:00
|
|
|
self.putHeaders(status_code, 'text/html')
|
2022-07-30 17:00:53 +00:00
|
|
|
return self.page_404(url_split)
|
|
|
|
except Exception as ex:
|
2021-01-02 16:16:44 +00:00
|
|
|
if self.server.swap_client.debug is True:
|
2021-12-16 08:44:10 +00:00
|
|
|
self.server.swap_client.log.error(traceback.format_exc())
|
2022-07-30 17:00:53 +00:00
|
|
|
self.putHeaders(status_code, 'text/html')
|
2021-01-02 16:16:44 +00:00
|
|
|
return self.page_error(str(ex))
|
|
|
|
|
2019-07-17 15:12:06 +00:00
|
|
|
try:
|
|
|
|
if len(url_split) > 1:
|
2022-07-31 17:33:01 +00:00
|
|
|
page = url_split[1]
|
|
|
|
if page == 'active':
|
2019-07-17 15:12:06 +00:00
|
|
|
return self.page_active(url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'wallets':
|
2022-07-05 22:46:37 +00:00
|
|
|
return page_wallets(self, url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'wallet':
|
2022-07-05 22:46:37 +00:00
|
|
|
return page_wallet(self, url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'settings':
|
2019-08-05 22:04:40 +00:00
|
|
|
return self.page_settings(url_split, post_string)
|
2022-10-10 21:44:14 +00:00
|
|
|
if page == 'error':
|
|
|
|
return self.page_error(url_split, post_string)
|
|
|
|
if page == 'info':
|
|
|
|
return self.page_info(url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'rpc':
|
2019-07-27 21:14:28 +00:00
|
|
|
return self.page_rpc(url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'debug':
|
2021-11-24 21:54:31 +00:00
|
|
|
return self.page_debug(url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'explorers':
|
2019-08-05 18:31:02 +00:00
|
|
|
return self.page_explorers(url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'offer':
|
2022-05-31 22:38:50 +00:00
|
|
|
return page_offer(self, url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'offers':
|
2022-04-10 20:08:05 +00:00
|
|
|
return page_offers(self, url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'newoffer':
|
2022-05-31 22:38:50 +00:00
|
|
|
return page_newoffer(self, url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'sentoffers':
|
2022-04-10 20:08:05 +00:00
|
|
|
return page_offers(self, url_split, post_string, sent=True)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'bid':
|
2022-10-05 17:30:52 +00:00
|
|
|
return page_bid(self, url_split, post_string)
|
2022-10-10 21:44:14 +00:00
|
|
|
if page == 'receivedbids':
|
|
|
|
return page_bids(self, url_split, post_string, received=True)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'sentbids':
|
2022-10-05 17:30:52 +00:00
|
|
|
return page_bids(self, url_split, post_string, sent=True)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'availablebids':
|
2022-10-05 17:30:52 +00:00
|
|
|
return page_bids(self, url_split, post_string, available=True)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'watched':
|
2019-07-17 15:12:06 +00:00
|
|
|
return self.page_watched(url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'smsgaddresses':
|
2021-10-19 18:59:18 +00:00
|
|
|
return self.page_smsgaddresses(url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'identity':
|
2021-11-14 23:26:43 +00:00
|
|
|
return self.page_identity(url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'tor':
|
2022-03-26 22:08:15 +00:00
|
|
|
return page_tor(self, url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'automation':
|
2022-05-23 21:51:06 +00:00
|
|
|
return page_automation_strategies(self, url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'automationstrategy':
|
2022-05-23 21:51:06 +00:00
|
|
|
return page_automation_strategy(self, url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'newautomationstrategy':
|
2022-05-23 21:51:06 +00:00
|
|
|
return page_automation_strategy_new(self, url_split, post_string)
|
2022-07-31 17:33:01 +00:00
|
|
|
if page == 'shutdown':
|
2019-08-05 18:31:02 +00:00
|
|
|
return self.page_shutdown(url_split, post_string)
|
2022-09-23 20:07:41 +00:00
|
|
|
if page != '':
|
|
|
|
return self.page_404(url_split)
|
2019-07-17 15:12:06 +00:00
|
|
|
return self.page_index(url_split)
|
2019-07-27 17:26:06 +00:00
|
|
|
except Exception as ex:
|
2019-08-15 19:14:28 +00:00
|
|
|
if self.server.swap_client.debug is True:
|
2021-12-16 08:44:10 +00:00
|
|
|
self.server.swap_client.log.error(traceback.format_exc())
|
2019-07-27 17:26:06 +00:00
|
|
|
return self.page_error(str(ex))
|
2019-07-17 15:12:06 +00:00
|
|
|
|
|
|
|
def do_GET(self):
|
|
|
|
response = self.handle_http(200, self.path)
|
|
|
|
self.wfile.write(response)
|
|
|
|
|
|
|
|
def do_POST(self):
|
2021-01-27 23:55:42 +00:00
|
|
|
post_string = self.rfile.read(int(self.headers.get('Content-Length')))
|
|
|
|
|
|
|
|
is_json = True if 'json' in self.headers.get('Content-Type', '') else False
|
|
|
|
response = self.handle_http(200, self.path, post_string, is_json)
|
2019-07-17 15:12:06 +00:00
|
|
|
self.wfile.write(response)
|
|
|
|
|
|
|
|
def do_HEAD(self):
|
|
|
|
self.putHeaders(200, 'text/html')
|
|
|
|
|
|
|
|
def do_OPTIONS(self):
|
|
|
|
self.send_response(200, 'ok')
|
|
|
|
if self.server.allow_cors:
|
|
|
|
self.send_header('Access-Control-Allow-Origin', '*')
|
|
|
|
self.send_header('Access-Control-Allow-Headers', '*')
|
|
|
|
self.end_headers()
|
|
|
|
|
|
|
|
|
|
|
|
class HttpThread(threading.Thread, HTTPServer):
|
|
|
|
def __init__(self, fp, host_name, port_no, allow_cors, swap_client):
|
|
|
|
threading.Thread.__init__(self)
|
|
|
|
|
|
|
|
self.stop_event = threading.Event()
|
|
|
|
self.fp = fp
|
|
|
|
self.host_name = host_name
|
|
|
|
self.port_no = port_no
|
|
|
|
self.allow_cors = allow_cors
|
|
|
|
self.swap_client = swap_client
|
2022-10-10 21:44:14 +00:00
|
|
|
self.title = 'BasicSwap - ' + __version__
|
2019-07-17 15:12:06 +00:00
|
|
|
self.last_form_id = dict()
|
2021-11-22 20:24:48 +00:00
|
|
|
self.session_tokens = dict()
|
2022-03-26 22:08:15 +00:00
|
|
|
self.env = env
|
2019-07-17 15:12:06 +00:00
|
|
|
|
|
|
|
self.timeout = 60
|
|
|
|
HTTPServer.__init__(self, (self.host_name, self.port_no), HttpHandler)
|
|
|
|
|
|
|
|
def stop(self):
|
|
|
|
self.stop_event.set()
|
|
|
|
|
|
|
|
# Send fake request
|
|
|
|
conn = http.client.HTTPConnection(self.host_name, self.port_no)
|
|
|
|
conn.connect()
|
|
|
|
conn.request('GET', '/none')
|
|
|
|
response = conn.getresponse()
|
|
|
|
data = response.read()
|
|
|
|
conn.close()
|
|
|
|
|
|
|
|
def stopped(self):
|
|
|
|
return self.stop_event.is_set()
|
|
|
|
|
|
|
|
def serve_forever(self):
|
|
|
|
while not self.stopped():
|
|
|
|
self.handle_request()
|
|
|
|
self.socket.close()
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
self.serve_forever()
|