@ -15,6 +15,20 @@ class MethodType(Enum):
HOOK = 1
HOOK = 1
class Method ( object ) :
""" Description of methods that are registered with the plugin.
These can be one of the following :
- RPC exposed by RPC passthrough
- HOOK registered to be called synchronously by lightningd
"""
def __init__ ( self , name , func , mtype = MethodType . RPCMETHOD ) :
self . name = name
self . func = func
self . mtype = mtype
class Plugin ( object ) :
class Plugin ( object ) :
""" Controls interactions with lightningd, and bundles functionality.
""" Controls interactions with lightningd, and bundles functionality.
@ -25,7 +39,7 @@ class Plugin(object):
"""
"""
def __init__ ( self , stdout = None , stdin = None , autopatch = True ) :
def __init__ ( self , stdout = None , stdin = None , autopatch = True ) :
self . methods = { ' init ' : ( self . _init , MethodType . RPCMETHOD ) }
self . methods = { ' init ' : Method ( ' init ' , self . _init , MethodType . RPCMETHOD ) }
self . options = { }
self . options = { }
# A dict from topics to handler functions
# A dict from topics to handler functions
@ -72,7 +86,8 @@ class Plugin(object):
)
)
# Register the function with the name
# Register the function with the name
self . methods [ name ] = ( func , MethodType . RPCMETHOD )
method = Method ( name , func , MethodType . RPCMETHOD )
self . methods [ name ] = method
def add_subscription ( self , topic , func ) :
def add_subscription ( self , topic , func ) :
""" Add a subscription to our list of subscriptions.
""" Add a subscription to our list of subscriptions.
@ -146,7 +161,8 @@ class Plugin(object):
raise ValueError (
raise ValueError (
" Method {} was already registered " . format ( name , self . methods [ name ] )
" Method {} was already registered " . format ( name , self . methods [ name ] )
)
)
self . methods [ name ] = ( func , MethodType . HOOK )
method = Method ( name , func , MethodType . HOOK )
self . methods [ name ] = method
def hook ( self , method_name ) :
def hook ( self , method_name ) :
""" Decorator to add a plugin hook to the dispatch table.
""" Decorator to add a plugin hook to the dispatch table.
@ -211,13 +227,13 @@ class Plugin(object):
if name not in self . methods :
if name not in self . methods :
raise ValueError ( " No method {} found. " . format ( name ) )
raise ValueError ( " No method {} found. " . format ( name ) )
func , _ = self . methods [ name ]
method = self . methods [ name ]
try :
try :
result = {
result = {
' jsonrpc ' : ' 2.0 ' ,
' jsonrpc ' : ' 2.0 ' ,
' id ' : request [ ' id ' ] ,
' id ' : request [ ' id ' ] ,
' result ' : self . _exec_func ( func , request )
' result ' : self . _exec_func ( method . func , request )
}
}
except Exception as e :
except Exception as e :
result = {
result = {
@ -291,27 +307,26 @@ class Plugin(object):
def _getmanifest ( self , * * kwargs ) :
def _getmanifest ( self , * * kwargs ) :
methods = [ ]
methods = [ ]
hooks = [ ]
hooks = [ ]
for name , entry in self . methods . items ( ) :
for method in self . methods . values ( ) :
func , typ = entry
# Skip the builtin ones, they don't get reported
# Skip the builtin ones, they don't get reported
if name in [ ' getmanifest ' , ' init ' ] :
if method . name in [ ' getmanifest ' , ' init ' ] :
continue
continue
if typ == MethodType . HOOK :
if method . m type == MethodType . HOOK :
hooks . append ( name )
hooks . append ( method . name )
continue
continue
doc = inspect . getdoc ( func )
doc = inspect . getdoc ( method . func )
if not doc :
if not doc :
self . log (
self . log (
' RPC method \' {} \' does not have a docstring. ' . format ( name )
' RPC method \' {} \' does not have a docstring. ' . format ( method . name )
)
)
doc = " Undocumented RPC method from a plugin. "
doc = " Undocumented RPC method from a plugin. "
doc = re . sub ( ' \n + ' , ' ' , doc )
doc = re . sub ( ' \n + ' , ' ' , doc )
# Handles out-of-order use of parameters like:
# Handles out-of-order use of parameters like:
# def hello_obfus(arg1, arg2, plugin, thing3, request=None, thing5='at', thing6=21)
# def hello_obfus(arg1, arg2, plugin, thing3, request=None, thing5='at', thing6=21)
argspec = inspect . getargspec ( func )
argspec = inspect . getargspec ( method . func )
defaults = argspec . defaults
defaults = argspec . defaults
num_defaults = len ( defaults ) if defaults else 0
num_defaults = len ( defaults ) if defaults else 0
start_kwargs_idx = len ( argspec . args ) - num_defaults
start_kwargs_idx = len ( argspec . args ) - num_defaults
@ -327,7 +342,7 @@ class Plugin(object):
args . append ( " [ %s ] " % arg )
args . append ( " [ %s ] " % arg )
methods . append ( {
methods . append ( {
' name ' : name ,
' name ' : method . name ,
' usage ' : " " . join ( args ) ,
' usage ' : " " . join ( args ) ,
' description ' : doc
' description ' : doc
} )
} )