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
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
|
# (c) Copyright IBM Corp. 2020 All Rights Reserved
#
# Author: Aman Kumar Sinha <amansi26@in.ibm.com>
#
# This file is part of cloud-init. See LICENSE file for license information.
"""
Refresh IPv6 interface and RMC
------------------------------
**Summary:** Ensure Network Manager is not managing IPv6 interface
This module is IBM PowerVM Hypervisor specific
Reliable Scalable Cluster Technology (RSCT) is a set of software components
that together provide a comprehensive clustering environment(RAS features)
for IBM PowerVM based virtual machines. RSCT includes the Resource
Monitoring and Control (RMC) subsystem. RMC is a generalized framework used
for managing, monitoring, and manipulating resources. RMC runs as a daemon
process on individual machines and needs creation of unique node id and
restarts during VM boot.
More details refer
https://www.ibm.com/support/knowledgecenter/en/SGVKBA_3.2/admin/bl503_ovrv.htm
This module handles
- Refreshing RMC
- Disabling NetworkManager from handling IPv6 interface, as IPv6 interface
is used for communication between RMC daemon and PowerVM hypervisor.
**Internal name:** ``cc_refresh_rmc_and_interface``
**Module frequency:** per always
**Supported distros:** RHEL
"""
from cloudinit import log as logging
from cloudinit.settings import PER_ALWAYS
from cloudinit import util
from cloudinit import subp
from cloudinit import netinfo
import errno
frequency = PER_ALWAYS
LOG = logging.getLogger(__name__)
# Ensure that /opt/rsct/bin has been added to standard PATH of the
# distro. The symlink to rmcctrl is /usr/sbin/rsct/bin/rmcctrl .
RMCCTRL = 'rmcctrl'
def handle(name, _cfg, _cloud, _log, _args):
if not subp.which(RMCCTRL):
LOG.debug("No '%s' in path, disabled", RMCCTRL)
return
LOG.debug(
'Making the IPv6 up explicitly. '
'Ensuring IPv6 interface is not being handled by NetworkManager '
'and it is restarted to re-establish the communication with '
'the hypervisor')
ifaces = find_ipv6_ifaces()
# Setting NM_CONTROLLED=no for IPv6 interface
# making it down and up
if len(ifaces) == 0:
LOG.debug("Did not find any interfaces with ipv6 addresses.")
else:
for iface in ifaces:
refresh_ipv6(iface)
disable_ipv6(sysconfig_path(iface))
restart_network_manager()
def find_ipv6_ifaces():
info = netinfo.netdev_info()
ifaces = []
for iface, data in info.items():
if iface == "lo":
LOG.debug('Skipping localhost interface')
if len(data.get("ipv4", [])) != 0:
# skip this interface, as it has ipv4 addrs
continue
ifaces.append(iface)
return ifaces
def refresh_ipv6(interface):
# IPv6 interface is explicitly brought up, subsequent to which the
# RMC services are restarted to re-establish the communication with
# the hypervisor.
subp.subp(['ip', 'link', 'set', interface, 'down'])
subp.subp(['ip', 'link', 'set', interface, 'up'])
def sysconfig_path(iface):
return '/etc/sysconfig/network-scripts/ifcfg-' + iface
def restart_network_manager():
subp.subp(['systemctl', 'restart', 'NetworkManager'])
def disable_ipv6(iface_file):
# Ensuring that the communication b/w the hypervisor and VM is not
# interrupted due to NetworkManager. For this purpose, as part of
# this function, the NM_CONTROLLED is explicitly set to No for IPV6
# interface and NetworkManager is restarted.
try:
contents = util.load_file(iface_file)
except IOError as e:
if e.errno == errno.ENOENT:
LOG.debug("IPv6 interface file %s does not exist\n",
iface_file)
else:
raise e
if 'IPV6INIT' not in contents:
LOG.debug("Interface file %s did not have IPV6INIT", iface_file)
return
LOG.debug("Editing interface file %s ", iface_file)
# Dropping any NM_CONTROLLED or IPV6 lines from IPv6 interface file.
lines = contents.splitlines()
lines = [line for line in lines if not search(line)]
lines.append("NM_CONTROLLED=no")
with open(iface_file, "w") as fp:
fp.write("\n".join(lines) + "\n")
def search(contents):
# Search for any NM_CONTROLLED or IPV6 lines in IPv6 interface file.
return(
contents.startswith("IPV6ADDR") or
contents.startswith("IPADDR6") or
contents.startswith("IPV6INIT") or
contents.startswith("NM_CONTROLLED"))
def refresh_rmc():
# To make a healthy connection between RMC daemon and hypervisor we
# refresh RMC. With refreshing RMC we are ensuring that making IPv6
# down and up shouldn't impact communication between RMC daemon and
# hypervisor.
# -z : stop Resource Monitoring & Control subsystem and all resource
# managers, but the command does not return control to the user
# until the subsystem and all resource managers are stopped.
# -s : start Resource Monitoring & Control subsystem.
try:
subp.subp([RMCCTRL, '-z'])
subp.subp([RMCCTRL, '-s'])
except Exception:
util.logexc(LOG, 'Failed to refresh the RMC subsystem.')
raise
|