merge jsonrpc gui and daemon
This commit is contained in:
151
electrum
151
electrum
@@ -81,6 +81,7 @@ from electrum import SimpleConfig, Network, Wallet, WalletStorage
|
|||||||
from electrum.util import print_msg, print_error, print_stderr, json_encode, json_decode, set_verbosity, InvalidPassword
|
from electrum.util import print_msg, print_error, print_stderr, json_encode, json_decode, set_verbosity, InvalidPassword
|
||||||
from electrum.plugins import Plugins, run_hook, always_hook
|
from electrum.plugins import Plugins, run_hook, always_hook
|
||||||
from electrum.commands import get_parser, known_commands, Commands, config_variables
|
from electrum.commands import get_parser, known_commands, Commands, config_variables
|
||||||
|
from electrum.daemon import Daemon, get_daemon
|
||||||
|
|
||||||
|
|
||||||
# get password routine
|
# get password routine
|
||||||
@@ -233,138 +234,20 @@ def init_cmdline(config):
|
|||||||
return cmd, password
|
return cmd, password
|
||||||
|
|
||||||
|
|
||||||
def run_command(config, cmd, network, wallet, password):
|
def run_offline_command(config, cmd, wallet, password):
|
||||||
if wallet and network:
|
|
||||||
wallet.wait_until_synchronized()
|
|
||||||
# arguments passed to function
|
# arguments passed to function
|
||||||
args = map(lambda x: config.get(x), cmd.params)
|
args = map(lambda x: config.get(x), cmd.params)
|
||||||
# decode json arguments
|
# decode json arguments
|
||||||
args = map(json_decode, args)
|
args = map(json_decode, args)
|
||||||
# options
|
# options
|
||||||
args += map(lambda x: config.get(x), cmd.options)
|
args += map(lambda x: config.get(x), cmd.options)
|
||||||
cmd_runner = Commands(config, wallet, network)
|
cmd_runner = Commands(config, wallet, None)
|
||||||
cmd_runner.password = password
|
cmd_runner.password = password
|
||||||
func = getattr(cmd_runner, cmd.name)
|
func = getattr(cmd_runner, cmd.name)
|
||||||
result = func(*args)
|
result = func(*args)
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
|
||||||
class ClientThread(util.DaemonThread):
|
|
||||||
|
|
||||||
def __init__(self, server, s):
|
|
||||||
util.DaemonThread.__init__(self)
|
|
||||||
self.server = server
|
|
||||||
self.client_pipe = util.SocketPipe(s)
|
|
||||||
self.network = self.server.network
|
|
||||||
|
|
||||||
def run(self):
|
|
||||||
config_options = self.client_pipe.get()
|
|
||||||
password = config_options.get('password')
|
|
||||||
config = SimpleConfig(config_options)
|
|
||||||
cmd = config.get('cmd')
|
|
||||||
if cmd == 'gui':
|
|
||||||
if self.server.gui:
|
|
||||||
if hasattr(server.gui, 'new_window'):
|
|
||||||
path = config.get_wallet_path()
|
|
||||||
self.server.gui.new_window(path, config.get('url'))
|
|
||||||
response = "ok"
|
|
||||||
else:
|
|
||||||
response = "error: current GUI does not support multiple windows"
|
|
||||||
else:
|
|
||||||
response = "Error: Electrum is running in daemon mode. Please stop the daemon first."
|
|
||||||
elif cmd == 'daemon':
|
|
||||||
sub = config.get('subcommand')
|
|
||||||
assert sub in ['start', 'stop', 'status']
|
|
||||||
if sub == 'start':
|
|
||||||
response = "Daemon already running"
|
|
||||||
elif sub == 'status':
|
|
||||||
p = self.network.get_parameters()
|
|
||||||
response = {
|
|
||||||
'path': self.network.config.path,
|
|
||||||
'server': p[0],
|
|
||||||
'blockchain_height': self.network.get_local_height(),
|
|
||||||
'server_height': self.network.get_server_height(),
|
|
||||||
'nodes': self.network.get_interfaces(),
|
|
||||||
'connected': self.network.is_connected(),
|
|
||||||
'auto_connect': p[4],
|
|
||||||
'wallets': self.server.wallets.keys(),
|
|
||||||
}
|
|
||||||
elif sub == 'stop':
|
|
||||||
self.server.stop()
|
|
||||||
response = "Daemon stopped"
|
|
||||||
else:
|
|
||||||
c = known_commands[cmd]
|
|
||||||
wallet = self.server.load_wallet(config) if c.requires_wallet else None
|
|
||||||
try:
|
|
||||||
response = run_command(config, c, self.network, wallet, password)
|
|
||||||
except BaseException as e:
|
|
||||||
err = traceback.format_exc()
|
|
||||||
response = {'error':err}
|
|
||||||
# send response and exit
|
|
||||||
self.client_pipe.send(response)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class NetworkServer(util.DaemonThread):
|
|
||||||
|
|
||||||
def __init__(self, config, network):
|
|
||||||
util.DaemonThread.__init__(self)
|
|
||||||
self.debug = False
|
|
||||||
self.config = config
|
|
||||||
self.pipe = util.QueuePipe()
|
|
||||||
self.network = network
|
|
||||||
self.lock = threading.RLock()
|
|
||||||
# gui is None is we run as daemon
|
|
||||||
self.gui = None
|
|
||||||
self.wallets = {}
|
|
||||||
|
|
||||||
def load_wallet(self, config):
|
|
||||||
path = config.get_wallet_path()
|
|
||||||
if path in self.wallets:
|
|
||||||
wallet = self.wallets[path]
|
|
||||||
else:
|
|
||||||
storage = WalletStorage(path)
|
|
||||||
wallet = Wallet(storage)
|
|
||||||
wallet.start_threads(self.network)
|
|
||||||
self.wallets[path] = wallet
|
|
||||||
return wallet
|
|
||||||
|
|
||||||
def run(self):
|
|
||||||
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
|
||||||
s.bind(('', 0))
|
|
||||||
lockfile = os.path.join(self.config.path, 'lock')
|
|
||||||
with open(lockfile, 'w') as f:
|
|
||||||
f.write("%d"%s.getsockname()[1])
|
|
||||||
s.listen(5)
|
|
||||||
s.settimeout(0.1)
|
|
||||||
while self.is_running():
|
|
||||||
try:
|
|
||||||
connection, address = s.accept()
|
|
||||||
except socket.timeout:
|
|
||||||
continue
|
|
||||||
client = ClientThread(self, connection)
|
|
||||||
client.start()
|
|
||||||
print_error("Daemon exiting")
|
|
||||||
|
|
||||||
def stop(self):
|
|
||||||
for k, wallet in self.wallets.items():
|
|
||||||
wallet.stop_threads()
|
|
||||||
util.DaemonThread.stop(self)
|
|
||||||
|
|
||||||
|
|
||||||
def get_daemon(config):
|
|
||||||
lockfile = os.path.join(config.path, 'lock')
|
|
||||||
try:
|
|
||||||
with open(lockfile) as f:
|
|
||||||
num = int(f.read())
|
|
||||||
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
|
||||||
s.connect(('', num))
|
|
||||||
return s
|
|
||||||
except:
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
|
|
||||||
# make sure that certificates are here
|
# make sure that certificates are here
|
||||||
@@ -441,20 +324,23 @@ if __name__ == '__main__':
|
|||||||
wallet = Wallet(storage)
|
wallet = Wallet(storage)
|
||||||
else:
|
else:
|
||||||
wallet = None
|
wallet = None
|
||||||
result = run_command(config, cmd, None, wallet, password)
|
result = run_offline_command(config, cmd, wallet, password)
|
||||||
print_msg(json_encode(result))
|
print_msg(json_encode(result))
|
||||||
sys.exit(0)
|
sys.exit(0)
|
||||||
else:
|
else:
|
||||||
config_options['password'] = password
|
config_options['password'] = password
|
||||||
|
|
||||||
# check if daemon is running
|
server = get_daemon(config)
|
||||||
s = get_daemon(config)
|
|
||||||
if s:
|
# daemon is running
|
||||||
p = util.SocketPipe(s)
|
if server is not None:
|
||||||
p.set_timeout(1000000)
|
cmdname = config_options.get('cmd')
|
||||||
p.send(config_options)
|
if cmdname == 'daemon':
|
||||||
result = p.get()
|
result = server.daemon(config_options)
|
||||||
s.close()
|
elif cmdname == 'gui':
|
||||||
|
result = server.gui(config_options)
|
||||||
|
else:
|
||||||
|
result = server.run_cmdline(config_options)
|
||||||
if type(result) in [str, unicode]:
|
if type(result) in [str, unicode]:
|
||||||
print_msg(result)
|
print_msg(result)
|
||||||
elif type(result) is dict and result.get('error'):
|
elif type(result) is dict and result.get('error'):
|
||||||
@@ -470,10 +356,13 @@ if __name__ == '__main__':
|
|||||||
network.start()
|
network.start()
|
||||||
else:
|
else:
|
||||||
network = None
|
network = None
|
||||||
server = NetworkServer(config, network)
|
server = Daemon(config, network)
|
||||||
server.start()
|
server.start()
|
||||||
server.gui = init_gui(config, network, plugins)
|
server.gui = init_gui(config, network, plugins)
|
||||||
server.gui.main()
|
server.gui.main()
|
||||||
|
server.stop()
|
||||||
|
sys.exit(0)
|
||||||
|
|
||||||
elif cmd_name == 'daemon':
|
elif cmd_name == 'daemon':
|
||||||
subcommand = config.get('subcommand')
|
subcommand = config.get('subcommand')
|
||||||
if subcommand in ['status', 'stop']:
|
if subcommand in ['status', 'stop']:
|
||||||
@@ -484,7 +373,7 @@ if __name__ == '__main__':
|
|||||||
if p == 0:
|
if p == 0:
|
||||||
network = Network(config, plugins)
|
network = Network(config, plugins)
|
||||||
network.start()
|
network.start()
|
||||||
server = NetworkServer(config, network)
|
server = Daemon(config, network)
|
||||||
if config.get('websocket_server'):
|
if config.get('websocket_server'):
|
||||||
from electrum import websockets
|
from electrum import websockets
|
||||||
websockets.WebSocketServer(config, network).start()
|
websockets.WebSocketServer(config, network).start()
|
||||||
|
|||||||
@@ -1,72 +0,0 @@
|
|||||||
#!/usr/bin/env python
|
|
||||||
#
|
|
||||||
# Electrum - lightweight Bitcoin client
|
|
||||||
# Copyright (C) 2015 Thomas Voegtlin
|
|
||||||
#
|
|
||||||
# This program is free software: you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU General Public License as published by
|
|
||||||
# the Free Software Foundation, either version 3 of the License, or
|
|
||||||
# (at your option) any later version.
|
|
||||||
#
|
|
||||||
# This program is distributed in the hope that it will be useful,
|
|
||||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
# GNU General Public License for more details.
|
|
||||||
#
|
|
||||||
# You should have received a copy of the GNU General Public License
|
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
|
|
||||||
"""
|
|
||||||
jsonrpc interface for webservers.
|
|
||||||
may be called from your php script.
|
|
||||||
"""
|
|
||||||
|
|
||||||
import socket, os
|
|
||||||
from jsonrpclib.SimpleJSONRPCServer import SimpleJSONRPCServer, SimpleJSONRPCRequestHandler
|
|
||||||
|
|
||||||
from electrum.wallet import WalletStorage, Wallet
|
|
||||||
from electrum.commands import known_commands, Commands
|
|
||||||
|
|
||||||
|
|
||||||
class RequestHandler(SimpleJSONRPCRequestHandler):
|
|
||||||
|
|
||||||
def do_OPTIONS(self):
|
|
||||||
self.send_response(200)
|
|
||||||
self.end_headers()
|
|
||||||
|
|
||||||
def end_headers(self):
|
|
||||||
self.send_header("Access-Control-Allow-Headers",
|
|
||||||
"Origin, X-Requested-With, Content-Type, Accept")
|
|
||||||
self.send_header("Access-Control-Allow-Origin", "*")
|
|
||||||
SimpleJSONRPCRequestHandler.end_headers(self)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class ElectrumGui:
|
|
||||||
|
|
||||||
def __init__(self, config, network, plugins):
|
|
||||||
self.network = network
|
|
||||||
self.config = config
|
|
||||||
storage = WalletStorage(self.config.get_wallet_path())
|
|
||||||
if not storage.file_exists:
|
|
||||||
raise BaseException("Wallet not found")
|
|
||||||
self.wallet = Wallet(storage)
|
|
||||||
self.cmd_runner = Commands(self.config, self.wallet, self.network)
|
|
||||||
host = config.get('rpchost', 'localhost')
|
|
||||||
port = config.get('rpcport', 7777)
|
|
||||||
self.server = SimpleJSONRPCServer((host, port), requestHandler=RequestHandler)
|
|
||||||
self.server.socket.settimeout(1)
|
|
||||||
for cmdname in known_commands:
|
|
||||||
self.server.register_function(getattr(self.cmd_runner, cmdname), cmdname)
|
|
||||||
|
|
||||||
def main(self):
|
|
||||||
self.wallet.start_threads(self.network)
|
|
||||||
while True:
|
|
||||||
try:
|
|
||||||
self.server.handle_request()
|
|
||||||
except socket.timeout:
|
|
||||||
continue
|
|
||||||
except:
|
|
||||||
break
|
|
||||||
self.wallet.stop_threads()
|
|
||||||
158
lib/daemon.py
Normal file
158
lib/daemon.py
Normal file
@@ -0,0 +1,158 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
#
|
||||||
|
# Electrum - lightweight Bitcoin client
|
||||||
|
# Copyright (C) 2015 Thomas Voegtlin
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import socket, os
|
||||||
|
import jsonrpclib
|
||||||
|
from jsonrpclib.SimpleJSONRPCServer import SimpleJSONRPCServer, SimpleJSONRPCRequestHandler
|
||||||
|
|
||||||
|
import util
|
||||||
|
from util import print_msg, print_error, print_stderr, json_encode, json_decode, set_verbosity, InvalidPassword
|
||||||
|
from wallet import WalletStorage, Wallet
|
||||||
|
from commands import known_commands, Commands
|
||||||
|
from simple_config import SimpleConfig
|
||||||
|
from network import Network
|
||||||
|
|
||||||
|
|
||||||
|
def get_daemon(config):
|
||||||
|
host = config.get('rpchost', 'localhost')
|
||||||
|
port = config.get('rpcport', 7777)
|
||||||
|
server = jsonrpclib.Server('http://%s:%d' % (host, port))
|
||||||
|
# check if daemon is running
|
||||||
|
try:
|
||||||
|
server.ping()
|
||||||
|
return server
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class RequestHandler(SimpleJSONRPCRequestHandler):
|
||||||
|
|
||||||
|
def do_OPTIONS(self):
|
||||||
|
self.send_response(200)
|
||||||
|
self.end_headers()
|
||||||
|
|
||||||
|
def end_headers(self):
|
||||||
|
self.send_header("Access-Control-Allow-Headers",
|
||||||
|
"Origin, X-Requested-With, Content-Type, Accept")
|
||||||
|
self.send_header("Access-Control-Allow-Origin", "*")
|
||||||
|
SimpleJSONRPCRequestHandler.end_headers(self)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
class Daemon(util.DaemonThread):
|
||||||
|
|
||||||
|
def __init__(self, config, network):
|
||||||
|
util.DaemonThread.__init__(self)
|
||||||
|
self.config = config
|
||||||
|
self.network = network
|
||||||
|
self.wallets = {}
|
||||||
|
self.wallet = self.load_wallet(config)
|
||||||
|
self.cmd_runner = Commands(self.config, self.wallet, self.network)
|
||||||
|
host = config.get('rpchost', 'localhost')
|
||||||
|
port = config.get('rpcport', 7777)
|
||||||
|
self.server = SimpleJSONRPCServer((host, port), requestHandler=RequestHandler, logRequests=False)
|
||||||
|
self.server.socket.settimeout(1)
|
||||||
|
for cmdname in known_commands:
|
||||||
|
self.server.register_function(getattr(self.cmd_runner, cmdname), cmdname)
|
||||||
|
self.server.register_function(self.run_cmdline, 'run_cmdline')
|
||||||
|
self.server.register_function(self.ping, 'ping')
|
||||||
|
self.server.register_function(self.daemon, 'daemon')
|
||||||
|
self.server.register_function(self.gui, 'gui')
|
||||||
|
|
||||||
|
def ping(self):
|
||||||
|
return True
|
||||||
|
|
||||||
|
def daemon(self, config):
|
||||||
|
sub = config.get('subcommand')
|
||||||
|
assert sub in ['start', 'stop', 'status']
|
||||||
|
if sub == 'start':
|
||||||
|
response = "Daemon already running"
|
||||||
|
elif sub == 'status':
|
||||||
|
p = self.network.get_parameters()
|
||||||
|
response = {
|
||||||
|
'path': self.network.config.path,
|
||||||
|
'server': p[0],
|
||||||
|
'blockchain_height': self.network.get_local_height(),
|
||||||
|
'server_height': self.network.get_server_height(),
|
||||||
|
'nodes': self.network.get_interfaces(),
|
||||||
|
'connected': self.network.is_connected(),
|
||||||
|
'auto_connect': p[4],
|
||||||
|
'wallets': self.wallets.keys(),
|
||||||
|
}
|
||||||
|
elif sub == 'stop':
|
||||||
|
self.stop()
|
||||||
|
response = "Daemon stopped"
|
||||||
|
return response
|
||||||
|
|
||||||
|
def gui(self, config_options):
|
||||||
|
config = SimpleConfig(config_options)
|
||||||
|
if self.gui:
|
||||||
|
if hasattr(self.gui, 'new_window'):
|
||||||
|
path = config.get_wallet_path()
|
||||||
|
self.gui.new_window(path, config.get('url'))
|
||||||
|
response = "ok"
|
||||||
|
else:
|
||||||
|
response = "error: current GUI does not support multiple windows"
|
||||||
|
else:
|
||||||
|
response = "Error: Electrum is running in daemon mode. Please stop the daemon first."
|
||||||
|
return response
|
||||||
|
|
||||||
|
def load_wallet(self, config):
|
||||||
|
path = config.get_wallet_path()
|
||||||
|
if path in self.wallets:
|
||||||
|
wallet = self.wallets[path]
|
||||||
|
else:
|
||||||
|
storage = WalletStorage(path)
|
||||||
|
wallet = Wallet(storage)
|
||||||
|
wallet.start_threads(self.network)
|
||||||
|
self.wallets[path] = wallet
|
||||||
|
return wallet
|
||||||
|
|
||||||
|
def run_cmdline(self, config_options):
|
||||||
|
password = config_options.get('password')
|
||||||
|
config = SimpleConfig(config_options)
|
||||||
|
cmdname = config.get('cmd')
|
||||||
|
cmd = known_commands[cmdname]
|
||||||
|
wallet = self.load_wallet(config) if cmd.requires_wallet else None
|
||||||
|
if wallet:
|
||||||
|
wallet.wait_until_synchronized()
|
||||||
|
# arguments passed to function
|
||||||
|
args = map(lambda x: config.get(x), cmd.params)
|
||||||
|
# decode json arguments
|
||||||
|
args = map(json_decode, args)
|
||||||
|
# options
|
||||||
|
args += map(lambda x: config.get(x), cmd.options)
|
||||||
|
cmd_runner = Commands(config, wallet, self.network)
|
||||||
|
cmd_runner.password = password
|
||||||
|
func = getattr(cmd_runner, cmd.name)
|
||||||
|
result = func(*args)
|
||||||
|
return result
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
while self.is_running():
|
||||||
|
try:
|
||||||
|
self.server.handle_request()
|
||||||
|
except socket.timeout:
|
||||||
|
continue
|
||||||
|
except:
|
||||||
|
break
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
for k, wallet in self.wallets.items():
|
||||||
|
wallet.stop_threads()
|
||||||
|
util.DaemonThread.stop(self)
|
||||||
@@ -118,7 +118,9 @@ class WalletStorage(PrintError):
|
|||||||
self.write()
|
self.write()
|
||||||
|
|
||||||
def write(self):
|
def write(self):
|
||||||
assert not threading.currentThread().isDaemon()
|
if threading.currentThread().isDaemon():
|
||||||
|
self.print_error('warning: daemon thread cannot write wallet')
|
||||||
|
return
|
||||||
if not self.modified:
|
if not self.modified:
|
||||||
return
|
return
|
||||||
s = json.dumps(self.data, indent=4, sort_keys=True)
|
s = json.dumps(self.data, indent=4, sort_keys=True)
|
||||||
|
|||||||
Reference in New Issue
Block a user