Skip to content

Commit

Permalink
lightningd: make shutdown_plugins a two-step function
Browse files Browse the repository at this point in the history
*first* call keeps db_write plugins alive, subscribed plugins have 30s to
finish their business

*second* call notifies subscribed db_write plugins for 2nd time
and gives then 5s to self-terminate

issue ElementsProject#4785
  • Loading branch information
SimonVrouwe committed Sep 29, 2021
1 parent ee5dbc0 commit e636ea5
Show file tree
Hide file tree
Showing 6 changed files with 77 additions and 37 deletions.
7 changes: 5 additions & 2 deletions lightningd/lightningd.c
Original file line number Diff line number Diff line change
Expand Up @@ -1133,8 +1133,8 @@ int main(int argc, char *argv[])
/* We're not going to collect our children. */
remove_sigchild_handler();

/* Tell plugins we're shutting down. */
shutdown_plugins(ld);
/* Tell plugins we're shutting down, except db_write plugins */
shutdown_plugins(ld, true);
shutdown_subdaemons(ld);

/* Clean up the JSON-RPC. This needs to happen in a DB transaction since
Expand All @@ -1144,6 +1144,9 @@ int main(int argc, char *argv[])
tal_free(ld->jsonrpc);
db_commit_transaction(ld->wallet->db);

/* Shutdown remaining plugins and close the db */
shutdown_plugins(ld, false);

/* Clean our our HTLC maps, since they use malloc. */
htlc_in_map_clear(&ld->htlcs_in);
htlc_out_map_clear(&ld->htlcs_out);
Expand Down
75 changes: 47 additions & 28 deletions lightningd/plugin.c
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,8 @@ static const char *state_desc(const struct plugin *plugin)
return "before replying to init";
case INIT_COMPLETE:
return "during normal operation";
case SHUTDOWN:
return "during shutdown";
}
fatal("Invalid plugin state %i for %s",
plugin->plugin_state, plugin->cmd);
Expand All @@ -82,7 +84,6 @@ struct plugins *plugins_new(const tal_t *ctx, struct log_book *log_book,
p->startup = true;
p->plugin_cmds = tal_arr(p, struct plugin_command *, 0);
p->blacklist = tal_arr(p, const char *, 0);
p->shutdown = false;
p->plugin_idx = 0;
#if DEVELOPER
p->dev_builtin_plugins_unimportant = false;
Expand Down Expand Up @@ -193,9 +194,9 @@ static void destroy_plugin(struct plugin *p)

/* If we are shutting down, do not continue to checking if
* the dying plugin is important. */
if (p->plugins->shutdown) {
/* But return if this was the last plugin! */
if (list_empty(&p->plugins->plugins))
if (p->plugin_state == SHUTDOWN) {
/* At shutdown we wait for plugins to self-terminate */
if (!plugins_any_in_state(p->plugins, SHUTDOWN))
io_break(p->plugins);
return;
}
Expand Down Expand Up @@ -722,7 +723,7 @@ static void plugin_conn_finish(struct io_conn *conn, struct plugin *plugin)
{
/* This is expected at shutdown of course. */
plugin_kill(plugin,
plugin->plugins->shutdown
plugin->plugin_state == SHUTDOWN
? LOG_DBG : LOG_INFORM,
"exited %s", state_desc(plugin));
}
Expand Down Expand Up @@ -1981,12 +1982,10 @@ void plugins_notify(struct plugins *plugins,
if (taken(n))
tal_steal(tmpctx, n);

/* If we're shutting down, ld->plugins will be NULL */
if (plugins) {
list_for_each(&plugins->plugins, p, list) {
plugin_single_notify(p, n);
}
}
list_for_each(&plugins->plugins, p, list) {
if (p->plugin_state != SHUTDOWN)
plugin_single_notify(p, n);
}
}

static void destroy_request(struct jsonrpc_request *req,
Expand Down Expand Up @@ -2087,43 +2086,63 @@ static void plugin_shutdown_timeout(struct lightningd *ld)
io_break(ld->plugins);
}

void shutdown_plugins(struct lightningd *ld)
void shutdown_plugins(struct lightningd *ld, bool first)
{
struct plugin *p, *next;

/* This makes sure we don't complain about important plugins
* vanishing! */
ld->plugins->shutdown = true;
/* First mark a set of plugins we want to shutdown in this call */
list_for_each(&ld->plugins->plugins, p, list) {
if (first && plugin_registered_db_write_hook(p))
continue;
/* don't complain about important plugins vanishing and stop sending
* it any (other) notifications */
plugin_set_state(p, SHUTDOWN);
}

/* Tell them all to shutdown; if they care. */
/* Notify **all** subscribed plugins */
list_for_each_safe(&ld->plugins->plugins, p, next, list) {
/* Kill immediately, deletes self from list. */
if (!notify_plugin_shutdown(ld, p))
if (notify_plugin_shutdown(ld, p))
continue;
/* or kill immediately those in the set */
else if (p->plugin_state == SHUTDOWN)
tal_free(p);
}

/* If anyone was interested in shutdown, give them time. */
if (!list_empty(&ld->plugins->plugins)) {
if (plugins_any_in_state(ld->plugins, SHUTDOWN)) {
struct oneshot *t;

/* 30 seconds should do it. */
/* Give them 30 or 5 seconds to self-terminate, the last one in
* the set calls io_break when destroyed */
t = new_reltimer(ld->timers, ld,
time_from_sec(30),
time_from_sec(first ? 30 : 5),
plugin_shutdown_timeout, ld);

io_loop_with_timers(ld);
tal_free(t);

/* Report and free remaining plugins. */
while (!list_empty(&ld->plugins->plugins)) {
p = list_pop(&ld->plugins->plugins, struct plugin, list);
/* Report and free plugins that didn't self-terminate */
list_for_each_safe(&ld->plugins->plugins, p, next, list) {
if (p->plugin_state != SHUTDOWN)
continue;

log_debug(ld->log,
"%s: failed to shutdown, killing.",
p->shortname);
tal_free(p);
}
}

/* NULL stops notifications trying to access plugins. */
ld->plugins = tal_free(ld->plugins);
/* In 2nd call all should be gone, close the db */
if (!first) {
assert(list_empty(&ld->plugins->plugins));
ld->plugins = tal_free(ld->plugins);
tal_free(ld->wallet);
return;
}

/* In first call, remove JSON-RPC methods of remaining plugins */
list_for_each(&ld->plugins->plugins, p, list) {
for (size_t i=0; i<tal_count(p->methods); i++) {
jsonrpc_command_del(ld->jsonrpc, p->methods[i]);
}
}
}
17 changes: 11 additions & 6 deletions lightningd/plugin.h
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,9 @@ enum plugin_state {
/* We have to get `init` response */
AWAITING_INIT_RESPONSE,
/* We have `init` response. */
INIT_COMPLETE
INIT_COMPLETE,
/* We are shutting down, maybe waiting for it to self-terminate */
SHUTDOWN
};

/**
Expand Down Expand Up @@ -113,9 +115,6 @@ struct plugins {
/* Blacklist of plugins from --disable-plugin */
const char **blacklist;

/* Whether we are shutting down (`plugins_free` is called) */
bool shutdown;

/* Index to show what order they were added in */
u64 plugin_idx;

Expand Down Expand Up @@ -236,9 +235,15 @@ void plugin_kill(struct plugin *plugin, enum log_level loglevel,
const char *fmt, ...);

/**
* Tell all the plugins we're shutting down, and free them.
* The @first=true call sends all subscribed plugins a shutdown notification and
* gives those 30s to self-terminate, others are killed immediate. Plugins
* that registered the db_write hook are kept alive, but their JSON RPC methods
* are removed.
*
* The 2nd (first=false) call repeats above for any remaining plugins, so
* db_write plugins can be notified twice, gives them 5s to self-terminate.
*/
void shutdown_plugins(struct lightningd *ld);
void shutdown_plugins(struct lightningd *ld, bool first);

/**
* Returns the plugin which registers the command with name {cmd_name}
Expand Down
10 changes: 10 additions & 0 deletions lightningd/plugin_hook.c
Original file line number Diff line number Diff line change
Expand Up @@ -290,6 +290,16 @@ struct db_write_hook_req {
size_t *num_hooks;
};

bool plugin_registered_db_write_hook(struct plugin *plugin) {
const struct plugin_hook *hook = &db_write_hook;

for (size_t i = 0; i < tal_count(hook->hooks); i++)
if (hook->hooks[i]->plugin == plugin)
return true;

return false;
}

static void db_hook_response(const char *buffer, const jsmntok_t *toks,
const jsmntok_t *idtok,
struct db_write_hook_req *dwh_req)
Expand Down
3 changes: 3 additions & 0 deletions lightningd/plugin_hook.h
Original file line number Diff line number Diff line change
Expand Up @@ -108,6 +108,9 @@ bool plugin_hook_continue(void *arg, const char *buffer, const jsmntok_t *toks);
struct plugin_hook *plugin_hook_register(struct plugin *plugin,
const char *method);

/* Helper to tell if plugin registered the db_write hook */
bool plugin_registered_db_write_hook(struct plugin *plugin);

/* Special sync plugin hook for db. */
void plugin_hook_db_sync(struct db *db);

Expand Down
2 changes: 1 addition & 1 deletion lightningd/test/run-find_my_abspath.c
Original file line number Diff line number Diff line change
Expand Up @@ -199,7 +199,7 @@ void setup_topology(struct chain_topology *topology UNNEEDED,
u32 min_blockheight UNNEEDED, u32 max_blockheight UNNEEDED)
{ fprintf(stderr, "setup_topology called!\n"); abort(); }
/* Generated stub for shutdown_plugins */
void shutdown_plugins(struct lightningd *ld UNNEEDED)
void shutdown_plugins(struct lightningd *ld UNNEEDED, bool first UNNEEDED)
{ fprintf(stderr, "shutdown_plugins called!\n"); abort(); }
/* Generated stub for stop_topology */
void stop_topology(struct chain_topology *topo UNNEEDED)
Expand Down

0 comments on commit e636ea5

Please sign in to comment.