|
|
|
#include <ccan/io/io.h>
|
|
|
|
#include <ccan/list/list.h>
|
|
|
|
#include <common/configdir.h>
|
|
|
|
#include <common/memleak.h>
|
|
|
|
#include <lightningd/jsonrpc.h>
|
|
|
|
#include <lightningd/plugin_hook.h>
|
|
|
|
#include <wallet/db.h>
|
|
|
|
#include <wallet/db_common.h>
|
|
|
|
|
|
|
|
/* Struct containing all the information needed to deserialize and
|
|
|
|
* dispatch an eventual plugin_hook response. */
|
|
|
|
struct plugin_hook_request {
|
|
|
|
struct list_head call_chain;
|
|
|
|
struct plugin *plugin;
|
|
|
|
const struct plugin_hook *hook;
|
|
|
|
void *cb_arg;
|
|
|
|
struct db *db;
|
|
|
|
struct lightningd *ld;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct hook_instance {
|
|
|
|
/* What plugin registered */
|
|
|
|
struct plugin *plugin;
|
|
|
|
|
|
|
|
/* Dependencies it asked for. */
|
|
|
|
const char **before, **after;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* A link in the plugin_hook call chain (there's a joke in there about
|
|
|
|
* computer scientists and naming...). The purpose is to act both as a list
|
|
|
|
* from which elements can be popped off as we progress along the chain as
|
|
|
|
* well as have a way for plugins to notify about their untimely demise during
|
|
|
|
* a hook call.
|
|
|
|
*/
|
|
|
|
struct plugin_hook_call_link {
|
|
|
|
struct list_node list;
|
|
|
|
struct plugin *plugin;
|
|
|
|
struct plugin_hook_request *req;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct plugin_hook *plugin_hook_by_name(const char *name)
|
|
|
|
{
|
|
|
|
static struct plugin_hook **hooks = NULL;
|
|
|
|
static size_t num_hooks;
|
|
|
|
if (!hooks)
|
|
|
|
hooks = autodata_get(hooks, &num_hooks);
|
|
|
|
|
|
|
|
for (size_t i=0; i<num_hooks; i++)
|
|
|
|
if (streq(hooks[i]->name, name))
|
|
|
|
return hooks[i];
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* When we destroy a plugin, we remove its hooks */
|
|
|
|
static void destroy_hook_instance(struct hook_instance *h,
|
|
|
|
struct plugin_hook *hook)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < tal_count(hook->hooks); i++) {
|
|
|
|
if (h == hook->hooks[i]) {
|
|
|
|
tal_arr_remove(&hook->hooks, i);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
struct plugin_hook *plugin_hook_register(struct plugin *plugin, const char *method)
|
|
|
|
{
|
|
|
|
struct hook_instance *h;
|
|
|
|
struct plugin_hook *hook = plugin_hook_by_name(method);
|
|
|
|
if (!hook) {
|
|
|
|
/* No such hook name registered */
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Make sure the hook_elements array is initialized. */
|
|
|
|
if (hook->hooks == NULL)
|
|
|
|
hook->hooks = notleak(tal_arr(NULL, struct hook_instance *, 0));
|
|
|
|
|
|
|
|
/* If this is a single type hook and we have a plugin registered we
|
|
|
|
* must fail this attempt to add the plugin to the hook. */
|
|
|
|
if (hook->type == PLUGIN_HOOK_SINGLE && tal_count(hook->hooks) > 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* Ensure we don't register the same plugin multple times. */
|
|
|
|
for (size_t i=0; i<tal_count(hook->hooks); i++)
|
|
|
|
if (hook->hooks[i]->plugin == plugin)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* Ok, we're sure they can register and they aren't yet registered, so
|
|
|
|
* register them. */
|
|
|
|
h = tal(plugin, struct hook_instance);
|
|
|
|
h->plugin = plugin;
|
|
|
|
h->before = tal_arr(h, const char *, 0);
|
|
|
|
h->after = tal_arr(h, const char *, 0);
|
|
|
|
tal_add_destructor2(h, destroy_hook_instance, hook);
|
|
|
|
|
|
|
|
tal_arr_expand(&hook->hooks, h);
|
|
|
|
return hook;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Mutual recursion */
|
|
|
|
static void plugin_hook_call_next(struct plugin_hook_request *ph_req);
|
|
|
|
static void plugin_hook_callback(const char *buffer, const jsmntok_t *toks,
|
|
|
|
const jsmntok_t *idtok,
|
|
|
|
struct plugin_hook_request *r);
|
|
|
|
|
|
|
|
/* We get notified if a plugin was killed while it was part of a call
|
|
|
|
* chain. If it was still to be called we just remove it from the list,
|
|
|
|
* otherwise it was the plugin that was currently handling the hook call, and
|
|
|
|
* we need to fail over to the next plugin.
|
|
|
|
*/
|
|
|
|
static void plugin_hook_killed(struct plugin_hook_call_link *link)
|
|
|
|
{
|
|
|
|
struct plugin_hook_call_link *head;
|
|
|
|
|
|
|
|
head = list_top(&link->req->call_chain, struct plugin_hook_call_link,
|
|
|
|
list);
|
|
|
|
|
|
|
|
/* If we are the head of the call chain, then the plugin died while it
|
|
|
|
* was handling the hook call. Pretend it didn't get the memo by
|
|
|
|
* calling the next one instead. This is correct since it is
|
|
|
|
* equivalent to the plugin dying before the hook invokation, assuming
|
|
|
|
* the plugin has not commmitted any changes internally. This is the
|
|
|
|
* weakest assumption we can make short of restarting the plugin and
|
|
|
|
* calling the hook again (potentially crashing the plugin the same
|
|
|
|
* way again.
|
|
|
|
*/
|
|
|
|
if (link == head) {
|
|
|
|
/* Call next will unlink, so we don't need to. This is treated
|
|
|
|
* equivalent to the plugin returning a continue-result.
|
|
|
|
*/
|
|
|
|
plugin_hook_callback(NULL, NULL, NULL, link->req);
|
|
|
|
} else {
|
|
|
|
/* The plugin is in the list waiting to be called, just remove
|
|
|
|
* it from the list. */
|
|
|
|
list_del(&link->list);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool plugin_hook_continue(void *unused, const char *buffer, const jsmntok_t *toks)
|
|
|
|
{
|
|
|
|
const jsmntok_t *resrestok = json_get_member(buffer, toks, "result");
|
|
|
|
return resrestok && json_tok_streq(buffer, resrestok, "continue");
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback to be passed to the jsonrpc_request.
|
|
|
|
*
|
|
|
|
* Unbundles the arguments, deserializes the response and dispatches
|
|
|
|
* it to the hook callback.
|
|
|
|
*/
|
|
|
|
static void plugin_hook_callback(const char *buffer, const jsmntok_t *toks,
|
|
|
|
const jsmntok_t *idtok,
|
|
|
|
struct plugin_hook_request *r)
|
|
|
|
{
|
|
|
|
const jsmntok_t *resulttok;
|
|
|
|
struct db *db = r->db;
|
|
|
|
struct plugin_hook_call_link *last, *it;
|
|
|
|
bool in_transaction = false;
|
|
|
|
|
|
|
|
if (r->ld->state == LD_STATE_SHUTDOWN) {
|
|
|
|
log_debug(r->ld->log,
|
|
|
|
"Abandoning plugin hook call due to shutdown");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/* Pop the head off the call chain and continue with the next */
|
|
|
|
last = list_pop(&r->call_chain, struct plugin_hook_call_link, list);
|
|
|
|
assert(last != NULL);
|
|
|
|
tal_del_destructor(last, plugin_hook_killed);
|
|
|
|
tal_free(last);
|
|
|
|
|
|
|
|
if (buffer) {
|
|
|
|
resulttok = json_get_member(buffer, toks, "result");
|
|
|
|
|
|
|
|
if (!resulttok)
|
|
|
|
fatal("Plugin for %s returned non-result response %.*s",
|
|
|
|
r->hook->name, toks->end - toks->start,
|
|
|
|
buffer + toks->start);
|
|
|
|
|
|
|
|
if (r->hook->type == PLUGIN_HOOK_CHAIN) {
|
|
|
|
db_begin_transaction(db);
|
|
|
|
if (!r->hook->deserialize_cb(r->cb_arg, buffer,
|
|
|
|
resulttok)) {
|
|
|
|
tal_free(r->cb_arg);
|
|
|
|
db_commit_transaction(db);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
in_transaction = true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* plugin died */
|
|
|
|
resulttok = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!list_empty(&r->call_chain)) {
|
|
|
|
if (in_transaction)
|
|
|
|
db_commit_transaction(db);
|
|
|
|
plugin_hook_call_next(r);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We optimize for the case where we already called deserialize_cb */
|
|
|
|
if (!in_transaction)
|
|
|
|
db_begin_transaction(db);
|
|
|
|
if (r->hook->type == PLUGIN_HOOK_CHAIN)
|
|
|
|
r->hook->final_cb(r->cb_arg);
|
|
|
|
else
|
|
|
|
r->hook->single_response_cb(r->cb_arg, buffer, resulttok);
|
|
|
|
db_commit_transaction(db);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
/* We need to remove the destructors from the remaining
|
|
|
|
* call-chain, otherwise they'd still be called when the
|
|
|
|
* plugin dies or we shut down. */
|
|
|
|
list_for_each(&r->call_chain, it, list) {
|
|
|
|
tal_del_destructor(it, plugin_hook_killed);
|
|
|
|
tal_steal(r, it);
|
|
|
|
}
|
|
|
|
|
|
|
|
tal_free(r);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void plugin_hook_call_next(struct plugin_hook_request *ph_req)
|
|
|
|
{
|
|
|
|
struct jsonrpc_request *req;
|
|
|
|
const struct plugin_hook *hook = ph_req->hook;
|
|
|
|
assert(!list_empty(&ph_req->call_chain));
|
|
|
|
ph_req->plugin = list_top(&ph_req->call_chain, struct plugin_hook_call_link, list)->plugin;
|
|
|
|
|
|
|
|
req = jsonrpc_request_start(NULL, hook->name,
|
|
|
|
plugin_get_log(ph_req->plugin),
|
|
|
|
NULL,
|
|
|
|
plugin_hook_callback, ph_req);
|
|
|
|
|
|
|
|
hook->serialize_payload(ph_req->cb_arg, req->stream);
|
|
|
|
jsonrpc_request_end(req);
|
|
|
|
plugin_request_send(ph_req->plugin, req);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool plugin_hook_call_(struct lightningd *ld, const struct plugin_hook *hook,
|
|
|
|
tal_t *cb_arg STEALS)
|
|
|
|
{
|
|
|
|
struct plugin_hook_request *ph_req;
|
|
|
|
struct plugin_hook_call_link *link;
|
|
|
|
if (tal_count(hook->hooks)) {
|
|
|
|
/* If we have a plugin that has registered for this
|
|
|
|
* hook, serialize and call it */
|
|
|
|
/* FIXME: technically this is a leak, but we don't
|
|
|
|
* currently have a list to store these. We might want
|
|
|
|
* to eventually to inspect in-flight requests. */
|
|
|
|
ph_req = notleak(tal(hook->hooks, struct plugin_hook_request));
|
|
|
|
ph_req->hook = hook;
|
|
|
|
ph_req->cb_arg = tal_steal(ph_req, cb_arg);
|
|
|
|
ph_req->db = ld->wallet->db;
|
|
|
|
ph_req->ld = ld;
|
|
|
|
|
|
|
|
list_head_init(&ph_req->call_chain);
|
|
|
|
for (size_t i=0; i<tal_count(hook->hooks); i++) {
|
|
|
|
/* We allocate this off of the plugin so we get notified if the plugin dies. */
|
|
|
|
link = tal(hook->hooks[i]->plugin,
|
|
|
|
struct plugin_hook_call_link);
|
|
|
|
link->plugin = hook->hooks[i]->plugin;
|
|
|
|
link->req = ph_req;
|
|
|
|
tal_add_destructor(link, plugin_hook_killed);
|
|
|
|
list_add_tail(&ph_req->call_chain, &link->list);
|
|
|
|
}
|
|
|
|
plugin_hook_call_next(ph_req);
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
/* If no plugin has registered for this hook, just
|
|
|
|
* call the callback with a NULL result. Saves us the
|
|
|
|
* roundtrip to the serializer and deserializer. If we
|
|
|
|
* were expecting a default response it should have
|
|
|
|
* been part of the `cb_arg`. */
|
|
|
|
if (hook->type == PLUGIN_HOOK_CHAIN)
|
|
|
|
hook->final_cb(cb_arg);
|
|
|
|
else
|
|
|
|
hook->single_response_cb(cb_arg, NULL, NULL);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We open-code this, because it's just different and special enough to be
|
|
|
|
* annoying, and to make it clear that it's totally synchronous. */
|
|
|
|
|
|
|
|
/* Special synchronous hook for db */
|
|
|
|
static struct plugin_hook db_write_hook = {"db_write", PLUGIN_HOOK_SINGLE, NULL,
|
|
|
|
NULL, NULL};
|
|
|
|
AUTODATA(hooks, &db_write_hook);
|
|
|
|
|
|
|
|
static void db_hook_response(const char *buffer, const jsmntok_t *toks,
|
|
|
|
const jsmntok_t *idtok,
|
|
|
|
struct plugin_hook_request *ph_req)
|
|
|
|
{
|
|
|
|
const jsmntok_t *resulttok;
|
|
|
|
|
|
|
|
resulttok = json_get_member(buffer, toks, "result");
|
|
|
|
if (!resulttok)
|
|
|
|
fatal("Plugin returned an invalid response to the db_write "
|
|
|
|
"hook: %s", buffer);
|
|
|
|
|
|
|
|
/* We expect result: { 'result' : 'continue' }.
|
|
|
|
* Anything else we abort.
|
|
|
|
*/
|
|
|
|
resulttok = json_get_member(buffer, resulttok, "result");
|
|
|
|
if (resulttok) {
|
|
|
|
if (!json_tok_streq(buffer, resulttok, "continue"))
|
|
|
|
fatal("Plugin returned failed db_write: %s.", buffer);
|
|
|
|
} else
|
|
|
|
fatal("Plugin returned an invalid result to the db_write "
|
|
|
|
"hook: %s", buffer);
|
|
|
|
|
|
|
|
/* We're done, exit exclusive loop. */
|
|
|
|
io_break(ph_req);
|
|
|
|
}
|
|
|
|
|
|
|
|
void plugin_hook_db_sync(struct db *db)
|
|
|
|
{
|
|
|
|
const struct plugin_hook *hook = &db_write_hook;
|
|
|
|
struct jsonrpc_request *req;
|
|
|
|
struct plugin_hook_request *ph_req;
|
|
|
|
void *ret;
|
|
|
|
struct plugin *plugin;
|
|
|
|
|
|
|
|
const char **changes = db_changes(db);
|
|
|
|
if (tal_count(hook->hooks) == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ph_req = notleak(tal(hook->hooks, struct plugin_hook_request));
|
|
|
|
/* FIXME: do IO logging for this! */
|
|
|
|
req = jsonrpc_request_start(NULL, hook->name, NULL, NULL,
|
|
|
|
db_hook_response,
|
|
|
|
ph_req);
|
|
|
|
|
|
|
|
ph_req->hook = hook;
|
|
|
|
ph_req->db = db;
|
|
|
|
plugin = ph_req->plugin = hook->hooks[0]->plugin;
|
|
|
|
|
|
|
|
json_add_num(req->stream, "data_version", db_data_version_get(db));
|
|
|
|
|
|
|
|
json_array_start(req->stream, "writes");
|
|
|
|
for (size_t i = 0; i < tal_count(changes); i++)
|
|
|
|
json_add_string(req->stream, NULL, changes[i]);
|
|
|
|
json_array_end(req->stream);
|
|
|
|
jsonrpc_request_end(req);
|
|
|
|
|
|
|
|
plugin_request_send(ph_req->plugin, req);
|
|
|
|
|
|
|
|
/* We can be called on way out of an io_loop, which is already breaking.
|
|
|
|
* That will make this immediately return; save the break value and call
|
|
|
|
* again, then hand it onwards. */
|
|
|
|
ret = plugin_exclusive_loop(plugin);
|
|
|
|
if (ret != ph_req) {
|
|
|
|
void *ret2 = plugin_exclusive_loop(plugin);
|
|
|
|
assert(ret2 == ph_req);
|
|
|
|
io_break(ret);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void add_deps(const char ***arr,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *arrtok)
|
|
|
|
{
|
|
|
|
const jsmntok_t *t;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (!arrtok)
|
|
|
|
return;
|
|
|
|
|
|
|
|
json_for_each_arr(i, t, arrtok)
|
|
|
|
tal_arr_expand(arr, json_strdup(*arr, buffer, t));
|
|
|
|
}
|
|
|
|
|
|
|
|
void plugin_hook_add_deps(struct plugin_hook *hook,
|
|
|
|
struct plugin *plugin,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *before,
|
|
|
|
const jsmntok_t *after)
|
|
|
|
{
|
|
|
|
struct hook_instance *h = NULL;
|
|
|
|
|
|
|
|
/* We just added this, it must exist */
|
|
|
|
for (size_t i = 0; i < tal_count(hook->hooks); i++) {
|
|
|
|
if (hook->hooks[i]->plugin == plugin) {
|
|
|
|
h = hook->hooks[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(h);
|
|
|
|
|
|
|
|
add_deps(&h->before, buffer, before);
|
|
|
|
add_deps(&h->after, buffer, after);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* From https://en.wikipedia.org/wiki/Topological_sorting#Kahn's_algorithm
|
|
|
|
* (https://creativecommons.org/licenses/by-sa/3.0/):
|
|
|
|
* L ← Empty list that will contain the sorted elements
|
|
|
|
* S ← Set of all nodes with no incoming edge
|
|
|
|
*
|
|
|
|
* while S is not empty do
|
|
|
|
* remove a node n from S
|
|
|
|
* add n to L
|
|
|
|
* for each node m with an edge e from n to m do
|
|
|
|
* remove edge e from the graph
|
|
|
|
* if m has no other incoming edges then
|
|
|
|
* insert m into S
|
|
|
|
*
|
|
|
|
* if graph has edges then
|
|
|
|
* return error (graph has at least one cycle)
|
|
|
|
* else
|
|
|
|
* return L (a topologically sorted order)
|
|
|
|
*/
|
|
|
|
struct hook_node {
|
|
|
|
struct hook_instance *hook;
|
|
|
|
size_t num_incoming;
|
|
|
|
struct hook_node **outgoing;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct hook_node *find_hook(struct hook_node *graph, const char *name)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < tal_count(graph); i++) {
|
|
|
|
if (plugin_paths_match(graph[i].hook->plugin->cmd, name))
|
|
|
|
return graph + i;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *plugin_hook_make_ordered(const tal_t *ctx, struct plugin_hook *hook)
|
|
|
|
{
|
|
|
|
struct hook_node *graph;
|
|
|
|
struct hook_node **l, **s;
|
|
|
|
char *ret;
|
|
|
|
|
|
|
|
/* Populate graph nodes */
|
|
|
|
graph = tal_arr(tmpctx, struct hook_node, tal_count(hook->hooks));
|
|
|
|
for (size_t i = 0; i < tal_count(graph); i++) {
|
|
|
|
graph[i].hook = hook->hooks[i];
|
|
|
|
graph[i].num_incoming = 0;
|
|
|
|
graph[i].outgoing = tal_arr(graph, struct hook_node *, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add edges. */
|
|
|
|
for (size_t i = 0; i < tal_count(graph); i++) {
|
|
|
|
for (size_t j = 0; j < tal_count(graph[i].hook->before); j++) {
|
|
|
|
struct hook_node *n = find_hook(graph,
|
|
|
|
graph[i].hook->before[j]);
|
|
|
|
if (!n) {
|
|
|
|
/* This is useful for typos! */
|
|
|
|
log_debug(graph[i].hook->plugin->log,
|
|
|
|
"hook %s before unknown plugin %s",
|
|
|
|
hook->name,
|
|
|
|
graph[i].hook->before[j]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
tal_arr_expand(&graph[i].outgoing, n);
|
|
|
|
n->num_incoming++;
|
|
|
|
}
|
|
|
|
for (size_t j = 0; j < tal_count(graph[i].hook->after); j++) {
|
|
|
|
struct hook_node *n = find_hook(graph,
|
|
|
|
graph[i].hook->after[j]);
|
|
|
|
if (!n) {
|
|
|
|
/* This is useful for typos! */
|
|
|
|
log_debug(graph[i].hook->plugin->log,
|
|
|
|
"hook %s after unknown plugin %s",
|
|
|
|
hook->name,
|
|
|
|
graph[i].hook->after[j]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
tal_arr_expand(&n->outgoing, &graph[i]);
|
|
|
|
graph[i].num_incoming++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Populate array of ready-to-go nodes. */
|
|
|
|
s = tal_arr(graph, struct hook_node *, 0);
|
|
|
|
for (size_t i = 0; i < tal_count(graph); i++) {
|
|
|
|
if (graph[i].num_incoming == 0)
|
|
|
|
tal_arr_expand(&s, &graph[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
l = tal_arr(graph, struct hook_node *, 0);
|
|
|
|
while (tal_count(s)) {
|
|
|
|
struct hook_node *n = s[0];
|
|
|
|
tal_arr_expand(&l, n);
|
|
|
|
tal_arr_remove(&s, 0);
|
|
|
|
|
|
|
|
/* for each node m with an edge e from n to m do
|
|
|
|
* remove edge e from the graph
|
|
|
|
* if m has no other incoming edges then
|
|
|
|
* insert m into S
|
|
|
|
*/
|
|
|
|
for (size_t i = 0; i < tal_count(n->outgoing); i++) {
|
|
|
|
if (--n->outgoing[i]->num_incoming == 0)
|
|
|
|
tal_arr_expand(&s, n->outgoing[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check for any left over */
|
|
|
|
ret = tal_strdup(ctx, "");
|
|
|
|
for (size_t i = 0; i < tal_count(graph); i++) {
|
|
|
|
if (graph[i].num_incoming)
|
|
|
|
tal_append_fmt(&ret, "%s ", graph[i].hook->plugin->cmd);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strlen(ret) == 0) {
|
|
|
|
/* Success! Write them back in order. */
|
|
|
|
assert(tal_count(l) == tal_count(hook->hooks));
|
|
|
|
for (size_t i = 0; i < tal_count(hook->hooks); i++)
|
|
|
|
hook->hooks[i] = l[i]->hook;
|
|
|
|
|
|
|
|
return tal_free(ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|