@ -66,24 +66,12 @@ struct plugin {
char * * subscriptions ;
char * * subscriptions ;
} ;
} ;
struct plugin_request {
u64 id ;
struct json_stream * stream ;
/* The response handler to be called when plugin gives us an object. */
void ( * cb ) ( const char * buffer ,
const jsmntok_t * toks ,
const jsmntok_t * idtok ,
void * ) ;
void * arg ;
} ;
struct plugins {
struct plugins {
struct list_head plugins ;
struct list_head plugins ;
size_t pending_manifests ;
size_t pending_manifests ;
/* Currently pending requests by their request ID */
/* Currently pending requests by their request ID */
UINTMAP ( struct plugin _request * ) pending_requests ;
UINTMAP ( struct jsonrpc_request * ) pending_requests ;
struct log * log ;
struct log * log ;
struct log_book * log_book ;
struct log_book * log_book ;
@ -192,39 +180,29 @@ static void PRINTF_FMT(2,3) plugin_kill(struct plugin *plugin, char *fmt, ...)
*
*
* The caller needs to add the request to req - > stream .
* The caller needs to add the request to req - > stream .
*/
*/
static struct plugin _request *
static struct jsonrpc _request *
plugin_request_new_ ( struct plugin * plugin ,
plugin_request_new_ ( struct plugin * plugin ,
const char * method ,
void ( * cb ) ( const char * buffer ,
void ( * cb ) ( const char * buffer ,
const jsmntok_t * toks ,
const jsmntok_t * toks ,
const jsmntok_t * idtok ,
const jsmntok_t * idtok ,
void * ) ,
void * ) ,
void * arg )
void * arg )
{
{
static u64 next_request_id = 0 ;
struct jsonrpc_request * req = jsonrpc_request_start ( plugin , method , cb , arg ) ;
struct plugin_request * req = tal ( plugin , struct plugin_request ) ;
req - > id = next_request_id + + ;
req - > cb = cb ;
req - > arg = arg ;
/* We will not concurrently drain, if we do we must set the
* writer to non - NULL */
req - > stream = new_json_stream ( req , NULL ) ;
/* Add to map so we can find it later when routing the response */
/* Add to map so we can find it later when routing the response */
uintmap_add ( & plugin - > plugins - > pending_requests , req - > id , req ) ;
uintmap_add ( & plugin - > plugins - > pending_requests , req - > id , req ) ;
return req ;
return req ;
}
}
# define plugin_request_new(plugin, method, response_cb, response_cb_arg) \
# define plugin_request_new(plugin, cb, arg) \
plugin_request_new_ ( ( plugin ) , ( method ) , \
plugin_request_new_ ( \
typesafe_cb_preargs ( void , void * , ( response_cb ) , \
( plugin ) , \
( response_cb_arg ) , \
typesafe_cb_preargs ( void , void * , ( cb ) , ( arg ) , \
const char * buffer , \
const char * buffer , \
const jsmntok_t * toks , \
const jsmntok_t * toks , \
const jsmntok_t * idtok ) , \
const jsmntok_t * idtok ) , \
( response_cb_arg ) )
( arg ) )
/**
/**
* Send a JSON - RPC message ( request or notification ) to the plugin .
* Send a JSON - RPC message ( request or notification ) to the plugin .
*/
*/
@ -303,7 +281,7 @@ static void plugin_response_handle(struct plugin *plugin,
const jsmntok_t * toks ,
const jsmntok_t * toks ,
const jsmntok_t * idtok )
const jsmntok_t * idtok )
{
{
struct plugin _request * request ;
struct jsonrpc _request * request ;
u64 id ;
u64 id ;
/* We only send u64 ids, so if this fails it's a critical error (note
/* We only send u64 ids, so if this fails it's a critical error (note
* that this also works if id is inside a JSON string ! ) . */
* that this also works if id is inside a JSON string ! ) . */
@ -323,7 +301,7 @@ static void plugin_response_handle(struct plugin *plugin,
}
}
/* We expect the request->cb to copy if needed */
/* We expect the request->cb to copy if needed */
request - > cb ( plugin - > buffer , toks , idtok , request - > arg ) ;
request - > response_ cb( plugin - > buffer , toks , idtok , request - > response_cb_ arg) ;
uintmap_del ( & plugin - > plugins - > pending_requests , id ) ;
uintmap_del ( & plugin - > plugins - > pending_requests , id ) ;
tal_free ( request ) ;
tal_free ( request ) ;
@ -666,7 +644,7 @@ static struct command_result *plugin_rpcmethod_dispatch(struct command *cmd,
{
{
const jsmntok_t * idtok ;
const jsmntok_t * idtok ;
struct plugin * plugin ;
struct plugin * plugin ;
struct plugin _request * req ;
struct jsonrpc _request * req ;
char id [ STR_MAX_CHARS ( u64 ) ] ;
char id [ STR_MAX_CHARS ( u64 ) ] ;
if ( cmd - > mode = = CMD_USAGE | | cmd - > mode = = CMD_CHECK ) {
if ( cmd - > mode = = CMD_USAGE | | cmd - > mode = = CMD_CHECK ) {
@ -681,7 +659,7 @@ static struct command_result *plugin_rpcmethod_dispatch(struct command *cmd,
idtok = json_get_member ( buffer , toks , " id " ) ;
idtok = json_get_member ( buffer , toks , " id " ) ;
assert ( idtok ! = NULL ) ;
assert ( idtok ! = NULL ) ;
req = plugin_request_new ( plugin , plugin_rpcmethod_cb , cmd ) ;
req = plugin_request_new ( plugin , NULL , plugin_rpcmethod_cb , cmd ) ;
snprintf ( id , ARRAY_SIZE ( id ) , " % " PRIu64 , req - > id ) ;
snprintf ( id , ARRAY_SIZE ( id ) , " % " PRIu64 , req - > id ) ;
json_stream_forward_change_id ( req - > stream , buffer , toks , idtok , id ) ;
json_stream_forward_change_id ( req - > stream , buffer , toks , idtok , id ) ;
@ -920,30 +898,13 @@ void clear_plugins(struct plugins *plugins)
tal_free ( p ) ;
tal_free ( p ) ;
}
}
/* For our own "getmanifest" and "init" requests: starts params[] */
static void start_simple_request ( struct plugin_request * req , const char * reqname )
{
json_object_start ( req - > stream , NULL ) ;
json_add_string ( req - > stream , " jsonrpc " , " 2.0 " ) ;
json_add_string ( req - > stream , " method " , reqname ) ;
json_add_u64 ( req - > stream , " id " , req - > id ) ;
}
static void end_simple_request ( struct plugin * plugin , struct plugin_request * req )
{
json_object_end ( req - > stream ) ;
json_stream_append ( req - > stream , " \n \n " ) ;
plugin_send ( plugin , req - > stream ) ;
req - > stream = NULL ;
}
void plugins_init ( struct plugins * plugins , const char * dev_plugin_debug )
void plugins_init ( struct plugins * plugins , const char * dev_plugin_debug )
{
{
struct plugin * p ;
struct plugin * p ;
char * * cmd ;
char * * cmd ;
int stdin , stdout ;
int stdin , stdout ;
struct timer * expired ;
struct timer * expired ;
struct plugin _request * req ;
struct jsonrpc_request * req ;
plugins - > pending_manifests = 0 ;
plugins - > pending_manifests = 0 ;
uintmap_init ( & plugins - > pending_requests ) ;
uintmap_init ( & plugins - > pending_requests ) ;
@ -969,11 +930,10 @@ void plugins_init(struct plugins *plugins, const char *dev_plugin_debug)
* write - only on p - > stdout */
* write - only on p - > stdout */
io_new_conn ( p , stdout , plugin_stdout_conn_init , p ) ;
io_new_conn ( p , stdout , plugin_stdout_conn_init , p ) ;
io_new_conn ( p , stdin , plugin_stdin_conn_init , p ) ;
io_new_conn ( p , stdin , plugin_stdin_conn_init , p ) ;
req = plugin_request_new ( p , plugin_manifest_cb , p ) ;
req = plugin_request_new ( p , " getmanifest " , plugin_manifest_cb , p ) ;
start_simple_request ( req , " getmanifest " ) ;
jsonrpc_request_end ( req ) ;
json_array_start ( req - > stream , " params " ) ;
plugin_send ( p , req - > stream ) ;
json_array_end ( req - > stream ) ;
end_simple_request ( p , req ) ;
plugins - > pending_manifests + + ;
plugins - > pending_manifests + + ;
/* Don't timeout if they're running a debugger. */
/* Don't timeout if they're running a debugger. */
if ( debug )
if ( debug )
@ -1011,13 +971,11 @@ static void plugin_config(struct plugin *plugin)
{
{
struct plugin_opt * opt ;
struct plugin_opt * opt ;
const char * name ;
const char * name ;
struct plugin _request * req ;
struct jsonrpc _request * req ;
struct lightningd * ld = plugin - > plugins - > ld ;
struct lightningd * ld = plugin - > plugins - > ld ;
/* No writer since we don't flush concurrently. */
/* No writer since we don't flush concurrently. */
req = plugin_request_new ( plugin , plugin_config_cb , plugin ) ;
req = plugin_request_new ( plugin , " init " , plugin_config_cb , plugin ) ;
start_simple_request ( req , " init " ) ;
json_object_start ( req - > stream , " params " ) ; /* start of .params */
/* Add .params.options */
/* Add .params.options */
json_object_start ( req - > stream , " options " ) ;
json_object_start ( req - > stream , " options " ) ;
@ -1034,8 +992,8 @@ static void plugin_config(struct plugin *plugin)
json_add_string ( req - > stream , " rpc-file " , ld - > rpc_filename ) ;
json_add_string ( req - > stream , " rpc-file " , ld - > rpc_filename ) ;
json_object_end ( req - > stream ) ;
json_object_end ( req - > stream ) ;
json_objec t_end ( req - > stream ) ; /* end of .params */
jsonrpc_reques t_end ( req ) ;
end_simple_request ( plugin , req ) ;
plugin_send ( plugin , req - > stream ) ;
}
}
void plugins_config ( struct plugins * plugins )
void plugins_config ( struct plugins * plugins )