2019-09-23 11:04:52 -04:00
|
|
|
#!/usr/bin/python2
|
2017-11-01 16:48:17 -04:00
|
|
|
# vim: tabstop=4 shiftwidth=4 softtabstop=4
|
|
|
|
|
|
|
|
# Copyright 2017 Lenovo
|
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
|
|
|
|
|
|
|
|
import os
|
|
|
|
import signal
|
|
|
|
import optparse
|
2018-08-02 16:53:56 -04:00
|
|
|
import shlex
|
2017-11-01 16:48:17 -04:00
|
|
|
import sys
|
|
|
|
|
|
|
|
try:
|
|
|
|
signal.signal(signal.SIGPIPE, signal.SIG_DFL)
|
|
|
|
except AttributeError:
|
|
|
|
pass
|
|
|
|
path = os.path.dirname(os.path.realpath(__file__))
|
|
|
|
path = os.path.realpath(os.path.join(path, '..', 'lib', 'python'))
|
|
|
|
if path.startswith('/opt'):
|
|
|
|
sys.path.append(path)
|
|
|
|
|
|
|
|
import confluent.client as client
|
|
|
|
|
|
|
|
class NullOpt(object):
|
|
|
|
blame = None
|
|
|
|
clear = None
|
|
|
|
|
|
|
|
|
|
|
|
def bailout(msg, code=1):
|
|
|
|
sys.stderr.write(msg + '\n')
|
|
|
|
sys.exit(code)
|
|
|
|
|
|
|
|
|
2019-02-21 15:02:53 -05:00
|
|
|
argparser = optparse.OptionParser(usage="Usage: %prog [options] noderange [option|option=value]")
|
2018-02-06 16:31:20 -05:00
|
|
|
argparser.add_option('-c', '--comparedefault', dest='comparedefault',
|
|
|
|
action='store_true', default=False,
|
|
|
|
help='Compare given settings to default or list settings '
|
|
|
|
'that are non default')
|
2018-08-02 16:53:56 -04:00
|
|
|
argparser.add_option('-b', '--batch', dest='batch', metavar='settings.batch',
|
|
|
|
default=False, help='Provide settings in a batch file')
|
2018-02-06 16:52:13 -05:00
|
|
|
argparser.add_option('-d', '--detail', dest='detail',
|
|
|
|
action='store_true', default=False,
|
|
|
|
help='Provide verbose information as available, such as '
|
|
|
|
'help text and possible valid values')
|
2020-01-29 10:15:32 -05:00
|
|
|
argparser.add_option('-e', '--extra', dest='extra',
|
|
|
|
action='store_true', default=False,
|
|
|
|
help='Access extra configuration. Extra configuration is generally '
|
|
|
|
'reserved for unpopular or redundant options that may be slow to '
|
|
|
|
'read. Notably the IMM category on Lenovo settings is considered '
|
|
|
|
'to be extra configuration')
|
2018-02-07 15:46:08 -05:00
|
|
|
argparser.add_option('-x', '--exclude', dest='exclude',
|
|
|
|
action='store_true', default=False,
|
|
|
|
help='Treat positional arguments as items to not '
|
|
|
|
'examine, compare, or restore default')
|
2018-08-17 11:16:11 -04:00
|
|
|
argparser.add_option('-a', '--advanced', dest='advanced',
|
|
|
|
action='store_true', default=False,
|
|
|
|
help='Include advanced settings, which are normally not '
|
|
|
|
'intended to be used without direction from the '
|
|
|
|
'relevant server vendor.')
|
2018-08-29 14:49:19 -04:00
|
|
|
argparser.add_option('-r', '--restoredefault', default=False,
|
2018-08-30 09:35:16 -04:00
|
|
|
dest='restoredefault', metavar="COMPONENT",
|
|
|
|
help='Restore the configuration of the node '
|
|
|
|
'to factory default for given component. '
|
|
|
|
'Currently only uefi is supported')
|
2019-10-23 14:20:40 -04:00
|
|
|
argparser.add_option('-m', '--maxnodes', type='int',
|
|
|
|
help='Specify a maximum number of '
|
|
|
|
'nodes to configure, '
|
2020-01-29 10:15:32 -05:00
|
|
|
'prompting if over the threshold')
|
2017-11-01 16:48:17 -04:00
|
|
|
(options, args) = argparser.parse_args()
|
|
|
|
|
|
|
|
cfgpaths = {
|
2017-11-03 10:36:32 -04:00
|
|
|
'bmc.ipv4_address': (
|
2017-11-01 16:48:17 -04:00
|
|
|
'configuration/management_controller/net_interfaces/management',
|
|
|
|
'ipv4_address'),
|
|
|
|
'bmc.ipv4_method': (
|
|
|
|
'configuration/management_controller/net_interfaces/management',
|
|
|
|
'ipv4_configuration'),
|
|
|
|
'bmc.ipv4_gateway': (
|
|
|
|
'configuration/management_controller/net_interfaces/management',
|
|
|
|
'ipv4_gateway'),
|
2018-03-27 16:32:37 -04:00
|
|
|
'bmc.hostname': (
|
|
|
|
'configuration/management_controller/hostname', 'hostname'),
|
2017-11-01 16:48:17 -04:00
|
|
|
}
|
|
|
|
|
2017-11-03 10:31:00 -04:00
|
|
|
autodeps = {
|
2017-11-03 10:36:32 -04:00
|
|
|
'bmc.ipv4_address': (('bmc.ipv4_method', 'static'),)
|
2017-11-03 10:31:00 -04:00
|
|
|
}
|
|
|
|
|
2017-11-01 16:48:17 -04:00
|
|
|
try:
|
|
|
|
noderange = args[0]
|
|
|
|
except IndexError:
|
|
|
|
argparser.print_help()
|
|
|
|
sys.exit(1)
|
2017-11-13 11:49:40 -05:00
|
|
|
client.check_globbing(noderange)
|
2017-11-01 16:48:17 -04:00
|
|
|
setmode = None
|
|
|
|
assignment = {}
|
|
|
|
queryparms = {}
|
2018-02-02 17:17:02 -05:00
|
|
|
printsys = []
|
2019-08-27 14:56:19 -04:00
|
|
|
printbmc = []
|
2020-01-29 14:20:56 -05:00
|
|
|
printextbmc = []
|
2019-08-29 15:24:14 -04:00
|
|
|
printallbmc = False
|
2018-02-02 17:17:02 -05:00
|
|
|
setsys = {}
|
2018-02-08 19:39:39 -05:00
|
|
|
forceset = False
|
|
|
|
needval = None
|
2017-11-02 17:07:44 -04:00
|
|
|
|
2018-02-07 15:46:08 -05:00
|
|
|
if len(args) == 1 or options.exclude:
|
|
|
|
if not options.exclude:
|
|
|
|
printsys = 'all'
|
2017-11-03 10:36:32 -04:00
|
|
|
for candidate in cfgpaths:
|
|
|
|
path, attrib = cfgpaths[candidate]
|
|
|
|
path = '/noderange/{0}/{1}'.format(noderange, path)
|
|
|
|
if path not in queryparms:
|
|
|
|
queryparms[path] = {}
|
|
|
|
queryparms[path][attrib] = candidate
|
2018-02-08 19:39:39 -05:00
|
|
|
|
|
|
|
|
|
|
|
def _assign_value():
|
|
|
|
if key not in cfgpaths:
|
|
|
|
setsys[key] = value
|
|
|
|
for depkey, depval in autodeps.get(key, []):
|
|
|
|
assignment[depkey] = depval
|
|
|
|
assignment[key] = value
|
|
|
|
|
|
|
|
|
2020-09-22 12:25:25 -04:00
|
|
|
def parse_config_line(arguments, single=False):
|
2019-08-29 15:24:14 -04:00
|
|
|
global setmode, printallbmc, forceset, key, value, needval, candidate, path, attrib
|
2020-09-22 12:25:25 -04:00
|
|
|
for pidx in range(0, len(arguments)):
|
|
|
|
param = arguments[pidx]
|
2018-08-02 16:53:56 -04:00
|
|
|
if param == 'show':
|
|
|
|
continue # forgive muscle memory of pasu users
|
|
|
|
if param == 'set':
|
2017-11-01 16:48:17 -04:00
|
|
|
setmode = True
|
2018-08-02 16:53:56 -04:00
|
|
|
forceset = True
|
|
|
|
continue
|
|
|
|
if needval:
|
|
|
|
key = needval
|
2019-01-16 11:31:06 -05:00
|
|
|
needval = None
|
2020-09-22 12:25:25 -04:00
|
|
|
if single:
|
|
|
|
value = ' '.join(arguments[pidx:])
|
|
|
|
_assign_value()
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
value = param
|
2018-02-08 19:39:39 -05:00
|
|
|
_assign_value()
|
2018-08-02 16:53:56 -04:00
|
|
|
continue
|
|
|
|
if '=' in param or param[-1] == ':' or forceset:
|
|
|
|
if setmode is None:
|
|
|
|
setmode = True
|
|
|
|
if setmode != True:
|
2021-05-18 10:33:27 -04:00
|
|
|
bailout('Cannot do set and query in same command: Query detected but "{0}" appears to be set'.format(param))
|
2018-08-02 16:53:56 -04:00
|
|
|
if '=' in param:
|
|
|
|
key, _, value = param.partition('=')
|
|
|
|
_assign_value()
|
|
|
|
elif param[-1] == ':':
|
|
|
|
needval = param[:-1]
|
|
|
|
else:
|
|
|
|
needval = param
|
2018-02-08 19:39:39 -05:00
|
|
|
else:
|
2018-08-02 16:53:56 -04:00
|
|
|
if setmode is None:
|
|
|
|
setmode = False
|
|
|
|
if setmode != False:
|
2021-05-18 10:33:27 -04:00
|
|
|
bailout('Cannot do set and query in same command: Set mode detected but "{0}" appears to be a query'.format(param))
|
2018-08-02 16:53:56 -04:00
|
|
|
if '.' not in param:
|
2019-08-29 15:24:14 -04:00
|
|
|
if param == 'bmc':
|
|
|
|
printallbmc = True
|
2018-08-02 16:53:56 -04:00
|
|
|
matchedparms = False
|
|
|
|
for candidate in cfgpaths:
|
|
|
|
if candidate.startswith('{0}.'.format(param)):
|
|
|
|
matchedparms = True
|
|
|
|
if not options.exclude:
|
|
|
|
path, attrib = cfgpaths[candidate]
|
|
|
|
path = '/noderange/{0}/{1}'.format(noderange, path)
|
|
|
|
if path not in queryparms:
|
|
|
|
queryparms[path] = {}
|
|
|
|
queryparms[path][attrib] = candidate
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
del queryparms[path]
|
|
|
|
except KeyError:
|
|
|
|
pass
|
2020-01-29 14:20:56 -05:00
|
|
|
if param.lower() == 'imm':
|
|
|
|
printextbmc.append(param)
|
|
|
|
options.extra = True
|
|
|
|
elif not matchedparms:
|
2018-08-02 16:53:56 -04:00
|
|
|
printsys.append(param)
|
|
|
|
elif param not in cfgpaths:
|
2019-08-27 14:56:19 -04:00
|
|
|
if param.startswith('bmc.'):
|
|
|
|
printbmc.append(param.replace('bmc.', ''))
|
2020-01-29 14:20:56 -05:00
|
|
|
elif param.lower().startswith('imm'):
|
|
|
|
options.extra = True
|
|
|
|
printextbmc.append(param)
|
2019-08-27 14:56:19 -04:00
|
|
|
else:
|
|
|
|
printsys.append(param)
|
2018-08-02 16:53:56 -04:00
|
|
|
else:
|
|
|
|
path, attrib = cfgpaths[param]
|
|
|
|
path = '/noderange/{0}/{1}'.format(noderange, path)
|
|
|
|
if path not in queryparms:
|
|
|
|
queryparms[path] = {}
|
|
|
|
queryparms[path][attrib] = param
|
|
|
|
|
|
|
|
if options.batch:
|
|
|
|
printsys = []
|
|
|
|
argfile = open(options.batch, 'r')
|
|
|
|
argset = argfile.readline()
|
|
|
|
while argset:
|
2019-02-12 11:12:47 -05:00
|
|
|
try:
|
|
|
|
argset = argset[:argset.index('#')]
|
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
argset = argset.strip()
|
|
|
|
if argset:
|
2020-09-22 12:25:25 -04:00
|
|
|
parse_config_line(shlex.split(argset), single=True)
|
2018-08-02 16:53:56 -04:00
|
|
|
argset = argfile.readline()
|
|
|
|
else:
|
|
|
|
parse_config_line(args[1:])
|
2017-11-01 16:48:17 -04:00
|
|
|
session = client.Command()
|
2018-02-05 15:23:13 -05:00
|
|
|
rcode = 0
|
2019-08-27 14:56:19 -04:00
|
|
|
if options.restoredefault:
|
2019-10-23 14:20:40 -04:00
|
|
|
session.stop_if_noderange_over(noderange, options.maxnodes)
|
2019-08-27 14:56:19 -04:00
|
|
|
if options.restoredefault.lower() in (
|
2018-08-30 09:35:16 -04:00
|
|
|
'sys', 'system', 'uefi', 'bios'):
|
2019-08-27 14:56:19 -04:00
|
|
|
for fr in session.update(
|
|
|
|
'/noderange/{0}/configuration/system/clear'.format(noderange),
|
|
|
|
{'clear': True}):
|
|
|
|
rcode |= client.printerror(fr)
|
|
|
|
sys.exit(rcode)
|
|
|
|
elif options.restoredefault.lower() in (
|
|
|
|
'bmc', 'imm', 'xcc'):
|
|
|
|
for fr in session.update(
|
|
|
|
'/noderange/{0}/configuration/management_controller/clear'.format(noderange),
|
|
|
|
{'clear': True}):
|
|
|
|
rcode |= client.printerror(fr)
|
|
|
|
sys.exit(rcode)
|
|
|
|
else:
|
|
|
|
sys.stderr.write(
|
|
|
|
'Unrecognized component to restore defaults: {0}\n'.format(
|
|
|
|
options.restoredefault))
|
|
|
|
sys.exit(1)
|
2017-11-01 16:48:17 -04:00
|
|
|
if setmode:
|
2019-10-23 14:20:40 -04:00
|
|
|
session.stop_if_noderange_over(noderange, options.maxnodes)
|
2018-02-07 15:46:08 -05:00
|
|
|
if options.exclude:
|
|
|
|
sys.stderr.write('Cannot use exclude and assign at the same time\n')
|
|
|
|
sys.exit(1)
|
2017-11-02 17:07:44 -04:00
|
|
|
updatebypath = {}
|
|
|
|
attrnamebypath = {}
|
|
|
|
for key in assignment:
|
|
|
|
if key not in cfgpaths:
|
2019-08-27 14:56:19 -04:00
|
|
|
if key.startswith('bmc.'):
|
|
|
|
path = 'configuration/management_controller/extended/all'
|
|
|
|
attrib = key.replace('bmc.', '')
|
|
|
|
else:
|
|
|
|
path = 'configuration/system/all'
|
|
|
|
attrib = key
|
2018-02-02 17:17:02 -05:00
|
|
|
else:
|
|
|
|
path, attrib = cfgpaths[key]
|
2017-11-02 17:07:44 -04:00
|
|
|
if path not in updatebypath:
|
|
|
|
updatebypath[path] = {}
|
|
|
|
attrnamebypath[path] = {}
|
|
|
|
updatebypath[path][attrib] = assignment[key]
|
|
|
|
attrnamebypath[path][attrib] = key
|
|
|
|
# well, we want to expand things..
|
|
|
|
# check ipv4, if requested change method to static
|
|
|
|
for path in updatebypath:
|
2018-02-05 15:23:13 -05:00
|
|
|
for fr in session.update('/noderange/{0}/{1}'.format(noderange, path),
|
2017-11-02 17:07:44 -04:00
|
|
|
updatebypath[path]):
|
2018-05-16 11:21:26 -04:00
|
|
|
rcode |= client.printerror(fr)
|
|
|
|
for node in fr.get('databynode', []):
|
2018-02-05 15:23:13 -05:00
|
|
|
r = fr['databynode'][node]
|
|
|
|
if 'value' not in r:
|
|
|
|
continue
|
|
|
|
keyval = r['value']
|
2017-11-02 17:07:44 -04:00
|
|
|
key, val = keyval.split('=')
|
|
|
|
if key in attrnamebypath[path]:
|
|
|
|
key = attrnamebypath[path][key]
|
|
|
|
print('{0}: {1}: {2}'.format(node, key, val))
|
2017-11-01 16:48:17 -04:00
|
|
|
else:
|
|
|
|
for path in queryparms:
|
2018-02-06 16:31:20 -05:00
|
|
|
if options.comparedefault:
|
|
|
|
continue
|
2018-06-01 16:48:19 -04:00
|
|
|
rc = client.print_attrib_path(path, session, list(queryparms[path]),
|
|
|
|
NullOpt(), queryparms[path])
|
|
|
|
if rc:
|
|
|
|
sys.exit(rc)
|
2020-01-29 14:20:56 -05:00
|
|
|
if printsys == 'all' or printextbmc or printbmc or printallbmc:
|
|
|
|
if printbmc or not printextbmc:
|
|
|
|
rcode = client.print_attrib_path(
|
|
|
|
'/noderange/{0}/configuration/management_controller/extended/all'.format(noderange),
|
|
|
|
session, printbmc, options, attrprefix='bmc.')
|
2020-01-29 10:15:32 -05:00
|
|
|
if options.extra:
|
|
|
|
rcode |= client.print_attrib_path(
|
|
|
|
'/noderange/{0}/configuration/management_controller/extended/extra'.format(noderange),
|
2020-01-29 14:20:56 -05:00
|
|
|
session, printextbmc, options)
|
2018-02-07 15:46:08 -05:00
|
|
|
if printsys or options.exclude:
|
2018-02-02 17:17:02 -05:00
|
|
|
if printsys == 'all':
|
|
|
|
printsys = []
|
2018-08-30 09:58:15 -04:00
|
|
|
if (options.comparedefault or printsys == []) and not options.advanced:
|
2018-08-17 11:16:11 -04:00
|
|
|
path = '/noderange/{0}/configuration/system/all'.format(noderange)
|
|
|
|
else:
|
|
|
|
path = '/noderange/{0}/configuration/system/advanced'.format(
|
|
|
|
noderange)
|
2018-06-01 16:48:19 -04:00
|
|
|
rcode = client.print_attrib_path(path, session, printsys,
|
|
|
|
options)
|
2018-08-30 09:35:16 -04:00
|
|
|
sys.exit(rcode)
|