|
|
|
#!/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 traceback
|
|
|
|
import sys
|
|
|
|
import os
|
|
|
|
import imp
|
|
|
|
import pkgutil
|
|
|
|
|
|
|
|
from util import *
|
|
|
|
from i18n import _
|
|
|
|
from util import print_error, profiler
|
|
|
|
|
|
|
|
plugins = {}
|
|
|
|
descriptions = []
|
|
|
|
loader = None
|
|
|
|
|
|
|
|
def is_available(name, w):
|
|
|
|
for d in descriptions:
|
|
|
|
if d.get('name') == name:
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
deps = d.get('requires', [])
|
|
|
|
for dep, s in deps:
|
|
|
|
try:
|
|
|
|
__import__(dep)
|
|
|
|
except ImportError:
|
|
|
|
return False
|
|
|
|
wallet_types = d.get('requires_wallet_type')
|
|
|
|
if wallet_types:
|
|
|
|
if w.wallet_type not in wallet_types:
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
@profiler
|
|
|
|
def init_plugins(config, is_local, gui_name):
|
|
|
|
global plugins, descriptions, loader
|
|
|
|
if is_local:
|
|
|
|
fp, pathname, description = imp.find_module('plugins')
|
|
|
|
electrum_plugins = imp.load_module('electrum_plugins', fp, pathname, description)
|
|
|
|
loader = lambda name: imp.load_source('electrum_plugins.' + name, os.path.join(pathname, name + '.py'))
|
|
|
|
else:
|
|
|
|
electrum_plugins = __import__('electrum_plugins')
|
|
|
|
loader = lambda name: __import__('electrum_plugins.' + name, fromlist=['electrum_plugins'])
|
|
|
|
|
|
|
|
def constructor(name, storage):
|
|
|
|
if plugins.get(name) is None:
|
|
|
|
try:
|
|
|
|
print_error(_("Loading plugin by constructor:"), name)
|
|
|
|
p = loader(name)
|
|
|
|
plugins[name] = p.Plugin(config, name)
|
|
|
|
except:
|
|
|
|
print_msg(_("Error: cannot initialize plugin"), name)
|
|
|
|
return
|
|
|
|
return plugins[name].constructor(storage)
|
|
|
|
|
|
|
|
def register_wallet_type(name, x, constructor):
|
|
|
|
import wallet
|
|
|
|
x += (lambda storage: constructor(name, storage),)
|
|
|
|
wallet.wallet_types.append(x)
|
|
|
|
|
|
|
|
descriptions = electrum_plugins.descriptions
|
|
|
|
for item in descriptions:
|
|
|
|
name = item['name']
|
|
|
|
if gui_name not in item.get('available_for', []):
|
|
|
|
continue
|
|
|
|
x = item.get('registers_wallet_type')
|
|
|
|
if x:
|
|
|
|
register_wallet_type(name, x, constructor)
|
|
|
|
if not config.get('use_' + name):
|
|
|
|
continue
|
|
|
|
try:
|
|
|
|
p = loader(name)
|
|
|
|
plugins[name] = p.Plugin(config, name)
|
|
|
|
except Exception:
|
|
|
|
print_msg(_("Error: cannot initialize plugin"), name)
|
|
|
|
traceback.print_exc(file=sys.stdout)
|
|
|
|
|
|
|
|
|
|
|
|
hook_names = set()
|
|
|
|
hooks = {}
|
|
|
|
|
|
|
|
def hook(func):
|
|
|
|
hook_names.add(func.func_name)
|
|
|
|
return func
|
|
|
|
|
|
|
|
def run_hook(name, *args):
|
|
|
|
return _run_hook(name, False, *args)
|
|
|
|
|
|
|
|
def always_hook(name, *args):
|
|
|
|
return _run_hook(name, True, *args)
|
|
|
|
|
|
|
|
def _run_hook(name, always, *args):
|
|
|
|
results = []
|
|
|
|
f_list = hooks.get(name, [])
|
|
|
|
for p, f in f_list:
|
|
|
|
if name == 'load_wallet':
|
|
|
|
p.wallet = args[0]
|
|
|
|
if name == 'init_qt':
|
|
|
|
gui = args[0]
|
|
|
|
p.window = gui.main_window
|
|
|
|
if always or p.is_enabled():
|
|
|
|
try:
|
|
|
|
r = f(*args)
|
|
|
|
except Exception:
|
|
|
|
print_error("Plugin error")
|
|
|
|
traceback.print_exc(file=sys.stdout)
|
|
|
|
r = False
|
|
|
|
if r:
|
|
|
|
results.append(r)
|
|
|
|
if name == 'close_wallet':
|
|
|
|
p.wallet = None
|
|
|
|
|
|
|
|
if results:
|
|
|
|
assert len(results) == 1, results
|
|
|
|
return results[0]
|
|
|
|
|
|
|
|
|
|
|
|
class BasePlugin:
|
|
|
|
|
|
|
|
def __init__(self, config, name):
|
|
|
|
self.name = name
|
|
|
|
self.config = config
|
|
|
|
self.wallet = None
|
|
|
|
# add self to hooks
|
|
|
|
for k in dir(self):
|
|
|
|
if k in hook_names:
|
|
|
|
l = hooks.get(k, [])
|
|
|
|
l.append((self, getattr(self, k)))
|
|
|
|
hooks[k] = l
|
|
|
|
|
|
|
|
def close(self):
|
|
|
|
# remove self from hooks
|
|
|
|
for k in dir(self):
|
|
|
|
if k in hook_names:
|
|
|
|
l = hooks.get(k, [])
|
|
|
|
l.remove((self, getattr(self, k)))
|
|
|
|
hooks[k] = l
|
|
|
|
|
|
|
|
def print_error(self, *msg):
|
|
|
|
print_error("[%s]"%self.name, *msg)
|
|
|
|
|
|
|
|
def requires_settings(self):
|
|
|
|
return False
|
|
|
|
|
|
|
|
def enable(self):
|
|
|
|
self.set_enabled(True)
|
|
|
|
return True
|
|
|
|
|
|
|
|
def disable(self):
|
|
|
|
self.set_enabled(False)
|
|
|
|
return True
|
|
|
|
|
|
|
|
@hook
|
|
|
|
def load_wallet(self, wallet, window): pass
|
|
|
|
|
|
|
|
@hook
|
|
|
|
def close_wallet(self): pass
|
|
|
|
|
|
|
|
#def init(self): pass
|
|
|
|
|
|
|
|
def is_enabled(self):
|
|
|
|
return self.is_available() and self.config.get('use_'+self.name) is True
|
|
|
|
|
|
|
|
def is_available(self):
|
|
|
|
return True
|
|
|
|
|
|
|
|
def set_enabled(self, enabled):
|
|
|
|
self.config.set_key('use_'+self.name, enabled, True)
|
|
|
|
|
|
|
|
def settings_dialog(self):
|
|
|
|
pass
|