2017-10-04 16:27:40 -04:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# 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 csv
|
|
|
|
import optparse
|
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
tabformat = '{0:>15}|{1:>15}|{2:>15}|{3:>36}|{4:>17}|{5:>12}|{6:>48}'
|
|
|
|
columns = ['Node', 'Model', 'Serial', 'UUID', 'Mac Address', 'Type',
|
|
|
|
'Current IP Addresses']
|
|
|
|
delimit = ['-' * 15, '-' * 15, '-' * 15, '-' * 36, '-' * 17, '-' * 12,
|
|
|
|
'-' * 48]
|
|
|
|
|
|
|
|
|
|
|
|
def dumpmacs(procinfo):
|
|
|
|
return ','.join(procinfo['macs']) # + procinfo.get('relatedmacs', []))
|
|
|
|
|
|
|
|
|
|
|
|
def print_disco(options, session, currmac):
|
|
|
|
procinfo = {}
|
|
|
|
for tmpinfo in session.read('/discovery/by-mac/{0}'.format(currmac)):
|
|
|
|
|
|
|
|
procinfo.update(tmpinfo)
|
|
|
|
record = [procinfo['nodename'], procinfo['modelnumber'],
|
|
|
|
procinfo['serialnumber'], procinfo['uuid'], dumpmacs(procinfo),
|
|
|
|
','.join(procinfo['types']),
|
|
|
|
','.join(sorted(procinfo['ipaddrs']))]
|
|
|
|
if options.csv:
|
|
|
|
csv.writer(sys.stdout).writerow(record)
|
|
|
|
else:
|
|
|
|
print(tabformat.format(*record))
|
|
|
|
|
|
|
|
|
2017-10-05 16:55:11 -04:00
|
|
|
def process_header(header):
|
|
|
|
# normalize likely header titles
|
|
|
|
fields = []
|
|
|
|
broken = False
|
|
|
|
for datum in header:
|
|
|
|
datum = datum.lower()
|
|
|
|
if datum.startswith('node') or datum.startswith('name'):
|
|
|
|
fields.append('node')
|
|
|
|
elif datum in ('nodegroup', 'nodegroups', 'group', 'groups'):
|
|
|
|
fields.append('groups')
|
|
|
|
elif datum.startswith('mac') or datum.startswith('ether'):
|
|
|
|
fields.append('mac')
|
|
|
|
elif datum.startswith('serial') or datum in ('sn', 's/n'):
|
|
|
|
fields.append('serial')
|
|
|
|
elif datum == 'uuid':
|
|
|
|
fields.append('uuid')
|
|
|
|
elif datum in ('bmc', 'imm', 'xcc'):
|
|
|
|
fields.append('hardwaremanagement.manager')
|
|
|
|
elif datum in ('bmc gateway', 'xcc gateway', 'imm gateway'):
|
|
|
|
fields.append('net.bmc.gateway')
|
|
|
|
elif datum in ('bmcuser', 'username', 'user'):
|
|
|
|
fields.append('secret.hardwaremanagementuser')
|
|
|
|
elif datum in ('bmcpass', 'password', 'pass'):
|
|
|
|
fields.append('secret.hardwaremanagementpassword')
|
|
|
|
else:
|
|
|
|
print("Unrecognized column name {0}".format(datum))
|
|
|
|
broken = True
|
|
|
|
if broken:
|
|
|
|
sys.exit(1)
|
|
|
|
return tuple(fields)
|
|
|
|
|
|
|
|
|
|
|
|
def datum_complete(datum):
|
|
|
|
if 'node' not in datum or not datum['node']:
|
|
|
|
sys.stderr.write('Nodename is a required field')
|
|
|
|
return False
|
|
|
|
provided = set(datum)
|
|
|
|
required = set(['serial', 'uuid', 'mac'])
|
|
|
|
for field in provided & required:
|
|
|
|
if datum[field]:
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
sys.stderr.write('One of the fields "Serial Number", "UUID", or '
|
|
|
|
'"MAC Address" must be provided')
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
searchkeys = set(['mac', 'serial', 'uuid'])
|
|
|
|
|
|
|
|
|
|
|
|
def search_record(datum, options, session):
|
|
|
|
for searchkey in searchkeys:
|
|
|
|
options.__dict__[searchkey] = None
|
|
|
|
for searchkey in searchkeys & set(datum):
|
|
|
|
options.__dict__[searchkey] = datum[searchkey]
|
|
|
|
return list(list_matching_macs(options, session))
|
|
|
|
|
|
|
|
|
2017-10-06 14:34:34 -04:00
|
|
|
def datum_to_attrib(datum):
|
|
|
|
for key in ('serial', 'uuid', 'mac'):
|
|
|
|
try:
|
|
|
|
del datum[key]
|
|
|
|
except KeyError:
|
|
|
|
pass
|
|
|
|
datum['name'] = datum['node']
|
|
|
|
del datum['node']
|
|
|
|
return datum
|
2017-10-05 16:55:11 -04:00
|
|
|
|
|
|
|
def import_csv(options, session):
|
2017-10-06 14:34:34 -04:00
|
|
|
nodedata = []
|
2017-10-05 16:55:11 -04:00
|
|
|
with open(options.importfile, 'r') as datasrc:
|
|
|
|
records = csv.reader(datasrc)
|
|
|
|
fields = process_header(next(records))
|
|
|
|
for record in records:
|
|
|
|
currfields = list(fields)
|
|
|
|
nodedatum = {}
|
|
|
|
for datum in record:
|
|
|
|
nodedatum[currfields.pop(0)] = datum
|
|
|
|
if not datum_complete(nodedatum):
|
|
|
|
sys.exit(1)
|
|
|
|
if not search_record(nodedatum, options, session):
|
|
|
|
sys.stderr.write(
|
|
|
|
"Could not match the following data: " +
|
|
|
|
repr(nodedatum) + '\n')
|
|
|
|
sys.exit(1)
|
2017-10-06 14:34:34 -04:00
|
|
|
nodedata.append(nodedatum)
|
|
|
|
for datum in nodedata:
|
|
|
|
maclist = search_record(datum, options, session)
|
|
|
|
datum = datum_to_attrib(datum)
|
|
|
|
nodename = datum['name']
|
|
|
|
for res in session.create('/nodes/', datum):
|
|
|
|
if 'error' in res:
|
|
|
|
sys.stderr.write(res['error'] + '\n')
|
|
|
|
continue
|
|
|
|
elif 'created' in res:
|
|
|
|
print('Defined ' + res['created'])
|
|
|
|
else:
|
|
|
|
print(repr(res))
|
|
|
|
for mac in maclist:
|
|
|
|
for res in session.update('/discovery/by-mac/{0}'.format(mac),
|
|
|
|
{'node': nodename}):
|
|
|
|
if 'error' in res:
|
|
|
|
sys.stderr.write(res['error'] + '\n')
|
|
|
|
continue
|
|
|
|
elif 'assigned' in res:
|
|
|
|
print('Discovered ' + res['assigned'])
|
|
|
|
else:
|
|
|
|
print(repr(res))
|
2017-10-05 16:55:11 -04:00
|
|
|
|
|
|
|
|
2017-10-04 16:27:40 -04:00
|
|
|
def list_discovery(options, session):
|
|
|
|
if options.csv:
|
|
|
|
csv.writer(sys.stdout).writerow(columns)
|
|
|
|
else:
|
|
|
|
print(tabformat.format(*columns))
|
|
|
|
print(tabformat.format(*delimit))
|
2017-10-05 13:52:20 -04:00
|
|
|
for mac in list_matching_macs(options, session):
|
|
|
|
print_disco(options, session, mac)
|
|
|
|
|
2018-01-23 15:05:24 -05:00
|
|
|
def clear_discovery(options, session):
|
|
|
|
for mac in list_matching_macs(options, session):
|
|
|
|
for res in session.delete('/discovery/by-mac/{0}'.format(mac)):
|
|
|
|
if 'deleted' in res:
|
|
|
|
print('Cleared info for {0}'.format(res['deleted']))
|
|
|
|
else:
|
|
|
|
print(repr(res))
|
2017-10-05 13:52:20 -04:00
|
|
|
|
|
|
|
def list_matching_macs(options, session):
|
2017-10-04 16:27:40 -04:00
|
|
|
path = '/discovery/'
|
|
|
|
if options.model:
|
|
|
|
path += 'by-model/{0}/'.format(options.model)
|
|
|
|
if options.serial:
|
|
|
|
path += 'by-serial/{0}/'.format(options.serial)
|
|
|
|
if options.uuid:
|
|
|
|
path += 'by-uuid/{0}/'.format(options.uuid)
|
|
|
|
if options.type:
|
|
|
|
path += 'by-type/{0}/'.format(options.type)
|
2018-01-23 16:09:17 -05:00
|
|
|
if options.state:
|
|
|
|
if options.state == 'unknown':
|
|
|
|
options.state = 'unidentified'
|
|
|
|
path += 'by-state/{0}/'.format(options.state).lower()
|
2017-10-04 16:27:40 -04:00
|
|
|
if options.mac:
|
2017-10-05 16:55:11 -04:00
|
|
|
path += 'by-mac/{0}'.format(options.mac)
|
|
|
|
result = list(session.read(path))[0]
|
|
|
|
if 'error' in result:
|
|
|
|
return []
|
2017-10-05 13:52:20 -04:00
|
|
|
return [options.mac.replace(':', '-')]
|
2017-10-04 16:27:40 -04:00
|
|
|
else:
|
|
|
|
path += 'by-mac/'
|
2017-10-05 13:52:20 -04:00
|
|
|
return [x['item']['href'] for x in session.read(path)]
|
|
|
|
|
|
|
|
def assign_discovery(options, session):
|
|
|
|
abort = False
|
2017-10-05 16:55:11 -04:00
|
|
|
if options.importfile:
|
|
|
|
return import_csv(options, session)
|
2017-10-05 13:52:20 -04:00
|
|
|
if not (options.serial or options.uuid or options.mac):
|
|
|
|
sys.stderr.write(
|
|
|
|
"UUID (-u), serial (-s), or ether address (-e) required for "
|
|
|
|
"assignment\n")
|
|
|
|
abort = True
|
|
|
|
if not options.node:
|
|
|
|
sys.stderr.write("Node (-n) must be specified for assignment\n")
|
|
|
|
abort = True
|
|
|
|
if abort:
|
|
|
|
sys.exit(1)
|
|
|
|
matches = list_matching_macs(options, session)
|
|
|
|
if not matches:
|
|
|
|
sys.stderr.write("No matching discovery candidates found\n")
|
|
|
|
sys.exit(1)
|
|
|
|
for res in session.update('/discovery/by-mac/{0}'.format(matches[0]),
|
|
|
|
{'node': options.node}):
|
|
|
|
if 'assigned' in res:
|
|
|
|
print('Assigned: {0}'.format(res['assigned']))
|
|
|
|
else:
|
|
|
|
print(repr(res))
|
|
|
|
|
|
|
|
|
2017-10-04 16:27:40 -04:00
|
|
|
|
|
|
|
def main():
|
|
|
|
parser = optparse.OptionParser(
|
2018-01-23 15:05:24 -05:00
|
|
|
usage='Usage: %prog [list|assign|rescan|clear] [options]')
|
2017-10-05 13:52:20 -04:00
|
|
|
# -a for 'address' maybe?
|
2017-10-06 14:34:34 -04:00
|
|
|
# order by
|
|
|
|
# show state (discovered or..
|
|
|
|
# nodediscover approve?
|
|
|
|
# flush to clear old data out? (e.g. no good way to age pxe data)
|
|
|
|
# also delete discovery datum... more targeted
|
|
|
|
# defect: -t lenovo-imm returns all
|
2017-10-04 16:27:40 -04:00
|
|
|
parser.add_option('-m', '--model', dest='model',
|
|
|
|
help='Operate with nodes matching the specified model '
|
|
|
|
'number', metavar='MODEL')
|
2018-01-23 16:09:17 -05:00
|
|
|
parser.add_option('-d', '--discoverystate', dest='state',
|
|
|
|
help='The discovery state of the entries (discovered, '
|
|
|
|
'identified, and unidentified)')
|
2017-10-04 16:27:40 -04:00
|
|
|
parser.add_option('-s', '--serial', dest='serial',
|
|
|
|
help='Operate against the system matching the specified '
|
|
|
|
'serial number', metavar='SERIAL')
|
|
|
|
parser.add_option('-u', '--uuid', dest='uuid',
|
|
|
|
help='Operate against the system matching the specified '
|
|
|
|
'UUID', metavar='UUID')
|
2017-10-05 13:52:20 -04:00
|
|
|
parser.add_option('-n', '--node', help='Operate with the given nodename')
|
|
|
|
parser.add_option('-e', '--ethaddr', dest='mac',
|
2017-10-04 16:27:40 -04:00
|
|
|
help='Operate against the system with the specified MAC '
|
|
|
|
'address', metavar='MAC')
|
|
|
|
parser.add_option('-t', '--type', dest='type',
|
|
|
|
help='Operate against the system of the specified type',
|
|
|
|
metavar='TYPE')
|
|
|
|
parser.add_option('-c', '--csv', dest='csv',
|
|
|
|
help='Use CSV formatted output', action='store_true')
|
2017-10-05 16:55:11 -04:00
|
|
|
parser.add_option('-i', '--import', dest='importfile',
|
2017-10-04 16:27:40 -04:00
|
|
|
help='Import bulk assignment data from given CSV file',
|
|
|
|
metavar='IMPORT.CSV')
|
|
|
|
(options, args) = parser.parse_args()
|
2018-01-23 15:05:24 -05:00
|
|
|
if len(args) == 0 or args[0] not in ('list', 'assign', 'rescan', 'clear'):
|
2017-10-04 16:27:40 -04:00
|
|
|
parser.print_help()
|
|
|
|
sys.exit(1)
|
|
|
|
session = client.Command()
|
|
|
|
if args[0] == 'list':
|
|
|
|
list_discovery(options, session)
|
2018-01-23 15:05:24 -05:00
|
|
|
if args[0] == 'clear':
|
|
|
|
clear_discovery(options, session)
|
2017-10-04 16:27:40 -04:00
|
|
|
if args[0] == 'assign':
|
2017-10-05 10:05:56 -04:00
|
|
|
assign_discovery(options, session)
|
|
|
|
if args[0] == 'rescan':
|
2017-10-20 13:53:13 -04:00
|
|
|
list(session.update('/discovery/rescan', {'rescan': 'start'}))
|
2017-10-05 10:05:56 -04:00
|
|
|
print("Rescan initiated")
|
2017-10-04 16:27:40 -04:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|