# This file is part of cloud-init. See LICENSE file for license information. """Run the dhclient hook to record network info.""" import argparse import os from cloudinit import atomic_helper from cloudinit import log as logging from cloudinit import stages LOG = logging.getLogger(__name__) NAME = "dhclient-hook" UP = "up" DOWN = "down" EVENTS = (UP, DOWN) def _get_hooks_dir(): i = stages.Init() return os.path.join(i.paths.get_runpath(), 'dhclient.hooks') def _filter_env_vals(info): """Given info (os.environ), return a dictionary with lower case keys for each entry starting with DHCP4_ or new_.""" new_info = {} for k, v in info.items(): if k.startswith("DHCP4_") or k.startswith("new_"): key = (k.replace('DHCP4_', '').replace('new_', '')).lower() new_info[key] = v return new_info def run_hook(interface, event, data_d=None, env=None): if event not in EVENTS: raise ValueError("Unexpected event '%s'. Expected one of: %s" % (event, EVENTS)) if data_d is None: data_d = _get_hooks_dir() if env is None: env = os.environ hook_file = os.path.join(data_d, interface + ".json") if event == UP: if not os.path.exists(data_d): os.makedirs(data_d) atomic_helper.write_json(hook_file, _filter_env_vals(env)) LOG.debug("Wrote dhclient options in %s", hook_file) elif event == DOWN: if os.path.exists(hook_file): os.remove(hook_file) LOG.debug("Removed dhclient options file %s", hook_file) def get_parser(parser=None): if parser is None: parser = argparse.ArgumentParser(prog=NAME, description=__doc__) parser.add_argument( "event", help='event taken on the interface', choices=EVENTS) parser.add_argument( "interface", help='the network interface being acted upon') # cloud-init main uses 'action' parser.set_defaults(action=(NAME, handle_args)) return parser def handle_args(name, args, data_d=None): """Handle the Namespace args. Takes 'name' as passed by cloud-init main. not used here.""" return run_hook(interface=args.interface, event=args.event, data_d=data_d) if __name__ == '__main__': import sys parser = get_parser() args = parser.parse_args(args=sys.argv[1:]) return_value = handle_args( NAME, args, data_d=os.environ.get('_CI_DHCP_HOOK_DATA_D')) if return_value: sys.exit(return_value) # vi: ts=4 expandtab