ifupdown-ng/libifupdown/lifecycle.c

244 lines
6.1 KiB
C
Raw Permalink Normal View History

2020-07-23 14:51:22 +00:00
/*
* libifupdown/lifecycle.c
* Purpose: management of interface lifecycle (bring up, takedown, reload)
*
* Copyright (c) 2020 Ariadne Conill <ariadne@dereferenced.org>
*
* Permission to use, copy, modify, and/or distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* This software is provided 'as is' and without any warranty, express or
* implied. In no event shall the authors be liable for any damages arising
* from the use of this software.
*/
#include <ctype.h>
2020-07-23 14:51:22 +00:00
#include <string.h>
#include "libifupdown/environment.h"
#include "libifupdown/execute.h"
#include "libifupdown/interface.h"
#include "libifupdown/lifecycle.h"
#include "libifupdown/state.h"
#include "libifupdown/tokenize.h"
2020-07-23 14:51:22 +00:00
static bool
handle_commands_for_phase(const struct lif_execute_opts *opts, char *const envp[], struct lif_interface *iface, const char *phase)
2020-07-23 14:51:22 +00:00
{
struct lif_node *iter;
LIF_DICT_FOREACH(iter, &iface->vars)
{
struct lif_dict_entry *entry = iter->data;
if (strcmp(entry->key, phase))
continue;
const char *cmd = entry->data;
if (!lif_execute_fmt(opts, envp, "%s", cmd))
return false;
}
return true;
}
static bool
handle_executors_for_phase(const struct lif_execute_opts *opts, char *const envp[], struct lif_interface *iface)
{
struct lif_node *iter;
LIF_DICT_FOREACH(iter, &iface->vars)
{
struct lif_dict_entry *entry = iter->data;
if (strcmp(entry->key, "use"))
continue;
const char *cmd = entry->data;
if (!lif_maybe_run_executor(opts, envp, cmd))
return false;
}
return true;
}
2020-07-23 14:51:22 +00:00
bool
lif_lifecycle_run_phase(const struct lif_execute_opts *opts, struct lif_interface *iface, const char *phase, const char *lifname, bool up)
{
char **envp = NULL;
lif_environment_push(&envp, "IFACE", lifname);
2020-07-23 14:51:22 +00:00
lif_environment_push(&envp, "PHASE", phase);
/* try to provide $METHOD for ifupdown1 scripts if we can */
2020-07-26 08:59:15 +00:00
if (iface->is_dhcp)
2020-07-23 14:51:22 +00:00
lif_environment_push(&envp, "METHOD", "dhcp");
/* same for $MODE */
if (up)
lif_environment_push(&envp, "MODE", "start");
else
lif_environment_push(&envp, "MODE", "stop");
if (opts->verbose)
lif_environment_push(&envp, "VERBOSE", "1");
if (opts->interfaces_file)
lif_environment_push(&envp, "INTERFACES_FILE", opts->interfaces_file);
struct lif_node *iter;
bool did_address = false, did_gateway = false;
LIF_DICT_FOREACH(iter, &iface->vars)
{
struct lif_dict_entry *entry = iter->data;
if (!strcmp(entry->key, "address"))
{
if (did_address)
continue;
struct lif_address *addr = entry->data;
char addrbuf[4096];
if (!lif_address_unparse(addr, addrbuf, sizeof addrbuf, true))
continue;
lif_environment_push(&envp, "IF_ADDRESS", addrbuf);
did_address = true;
continue;
}
else if (!strcmp(entry->key, "gateway"))
{
if (did_gateway)
continue;
did_gateway = true;
}
else if (!strcmp(entry->key, "requires"))
{
if (iface->is_bridge)
lif_environment_push(&envp, "IF_BRIDGE_PORTS", (const char *) entry->data);
if (iface->is_bond)
lif_environment_push(&envp, "IF_BOND_SLAVES", (const char *) entry->data);
}
char envkey[4096] = "IF_";
strlcat(envkey, entry->key, sizeof envkey);
char *ep = envkey + 2;
while (*ep++)
{
*ep = toupper(*ep);
if (*ep == '-')
*ep = '_';
}
lif_environment_push(&envp, envkey, (const char *) entry->data);
}
if (!handle_executors_for_phase(opts, envp, iface))
goto handle_error;
2020-07-23 14:51:22 +00:00
if (!handle_commands_for_phase(opts, envp, iface, phase))
goto handle_error;
2020-07-23 14:51:22 +00:00
/* we should do error handling here, but ifupdown1 doesn't */
lif_execute_fmt(opts, envp, "/bin/run-parts /etc/network/if-%s.d", phase);
2020-07-23 14:51:22 +00:00
lif_environment_free(&envp);
return true;
handle_error:
2020-07-23 14:51:22 +00:00
lif_environment_free(&envp);
return false;
}
2020-07-24 09:42:34 +00:00
static bool
handle_dependents(const struct lif_execute_opts *opts, struct lif_interface *parent, struct lif_dict *collection, struct lif_dict *state, bool up)
{
struct lif_dict_entry *requires = lif_dict_find(&parent->vars, "requires");
/* no dependents, nothing to worry about */
if (requires == NULL)
return true;
char require_ifs[4096] = {};
2020-07-24 09:42:34 +00:00
strlcpy(require_ifs, requires->data, sizeof require_ifs);
char *bufp = require_ifs;
for (char *tokenp = lif_next_token(&bufp); *tokenp; tokenp = lif_next_token(&bufp))
2020-07-24 09:42:34 +00:00
{
struct lif_interface *iface = lif_interface_collection_find(collection, tokenp);
/* already up or down, skip */
if (up == iface->is_up)
continue;
if (opts->verbose)
fprintf(stderr, "ifupdown: changing state of dependent interface %s (of %s) to %s\n",
iface->ifname, parent->ifname, up ? "up" : "down");
2020-07-24 09:42:34 +00:00
if (!lif_lifecycle_run(opts, iface, collection, state, iface->ifname, up))
return false;
}
return true;
}
2020-07-23 14:51:22 +00:00
bool
2020-07-24 09:42:34 +00:00
lif_lifecycle_run(const struct lif_execute_opts *opts, struct lif_interface *iface, struct lif_dict *collection, struct lif_dict *state, const char *lifname, bool up)
2020-07-23 14:51:22 +00:00
{
if (lifname == NULL)
lifname = iface->ifname;
if (up)
{
2020-07-24 09:42:34 +00:00
/* when going up, dependents go up first. */
if (!handle_dependents(opts, iface, collection, state, up))
return false;
2020-07-23 14:51:22 +00:00
/* XXX: we should try to recover (take the iface down) if bringing it up fails.
* but, right now neither debian ifupdown or busybox ifupdown do any recovery,
* so we wont right now.
*/
if (!lif_lifecycle_run_phase(opts, iface, "pre-up", lifname, up))
return false;
if (!lif_lifecycle_run_phase(opts, iface, "up", lifname, up))
return false;
if (!lif_lifecycle_run_phase(opts, iface, "post-up", lifname, up))
return false;
2020-07-23 14:51:22 +00:00
lif_state_upsert(state, lifname, iface);
2020-07-24 09:42:34 +00:00
iface->is_up = true;
2020-07-23 14:51:22 +00:00
}
else
{
if (!lif_lifecycle_run_phase(opts, iface, "pre-down", lifname, up))
return false;
2020-07-23 14:51:22 +00:00
if (!lif_lifecycle_run_phase(opts, iface, "down", lifname, up))
return false;
if (!lif_lifecycle_run_phase(opts, iface, "post-down", lifname, up))
return false;
2020-07-24 09:42:34 +00:00
/* when going up, dependents go down last. */
if (!handle_dependents(opts, iface, collection, state, up))
return false;
2020-07-23 14:51:22 +00:00
lif_state_delete(state, lifname);
2020-07-24 09:42:34 +00:00
iface->is_up = false;
2020-07-23 14:51:22 +00:00
}
return true;
}