1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
|
# 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
|