2019-12-03 20:27:13 +00:00
|
|
|
from servicebase import ServiceBase
|
2019-12-06 18:14:19 +00:00
|
|
|
import requests
|
2019-12-09 18:57:22 +00:00
|
|
|
import sys
|
2019-12-06 18:14:19 +00:00
|
|
|
import re
|
|
|
|
import json
|
2019-12-09 18:57:22 +00:00
|
|
|
import urllib3
|
|
|
|
urllib3.disable_warnings()
|
2019-12-03 20:27:13 +00:00
|
|
|
|
|
|
|
class ServiceDelegate(ServiceBase) :
|
2019-12-06 18:14:19 +00:00
|
|
|
|
2020-05-26 18:26:01 +00:00
|
|
|
_return_payload = {}
|
|
|
|
_hosts = None
|
2019-12-09 18:57:22 +00:00
|
|
|
_partition = 'Common'
|
2019-12-06 18:14:19 +00:00
|
|
|
|
2019-12-04 19:00:50 +00:00
|
|
|
def get_arguments(cls) :
|
2020-05-22 18:28:13 +00:00
|
|
|
return ['-cg', '--vip', 'store_true', 'Return VIP information about the subject (f5)']
|
2019-12-03 20:27:13 +00:00
|
|
|
|
2019-12-06 18:14:19 +00:00
|
|
|
def startup(self) :
|
|
|
|
for requirement in ['hosts','username','key'] :
|
2020-07-14 21:36:46 +00:00
|
|
|
if requirement not in self._config or (requirement in self._config and self._config[requirement] == ''):
|
2020-05-26 18:26:01 +00:00
|
|
|
self._error.append('Missing required config option ' + requirement)
|
2019-12-06 18:14:19 +00:00
|
|
|
return
|
2020-05-26 18:26:01 +00:00
|
|
|
self._hosts = self._config['hosts']
|
2020-05-22 18:28:13 +00:00
|
|
|
self.debug('Logging into F5\'s and searching for hosts, this make take some time.',1)
|
2019-12-06 18:14:19 +00:00
|
|
|
|
2020-05-22 02:34:56 +00:00
|
|
|
def lookup(self,subject) :
|
2020-05-26 18:26:01 +00:00
|
|
|
self._return_payload = {}
|
2019-12-06 18:14:19 +00:00
|
|
|
|
2019-12-16 20:47:10 +00:00
|
|
|
logincandidates = self.choose_host(subject)
|
2020-05-28 18:00:50 +00:00
|
|
|
if len(logincandidates) == 0 :
|
2019-12-09 18:57:22 +00:00
|
|
|
# Idk what to do here, do we give up or do we log in to all of them?
|
2020-05-26 18:26:01 +00:00
|
|
|
self._error.append('Unable to find an LB to log into to check host ' + str(subject))
|
2020-05-28 18:00:50 +00:00
|
|
|
# Search all
|
|
|
|
logincandidates = {x:x for x in self._hosts}
|
|
|
|
# Give up
|
|
|
|
# return None
|
2019-12-06 18:14:19 +00:00
|
|
|
|
2019-12-09 18:57:22 +00:00
|
|
|
for host in logincandidates.keys() :
|
2020-05-26 18:26:01 +00:00
|
|
|
auth_header = self.get_iq_auth_header(host,self._config['username'],self._config['key'])
|
2019-12-09 18:57:22 +00:00
|
|
|
if not auth_header :
|
2020-05-26 18:26:01 +00:00
|
|
|
self._error.append('Unable to log in to ' + host)
|
2019-12-09 18:57:22 +00:00
|
|
|
continue
|
|
|
|
# Get all VIPs
|
2020-05-26 18:26:01 +00:00
|
|
|
if 'vip' not in self._return_payload :
|
2019-12-09 18:57:22 +00:00
|
|
|
vips = self.get_vip(host,auth_header);
|
|
|
|
# If it's a VIP, return it and the host and let's be done
|
|
|
|
if vips :
|
2020-05-22 18:28:13 +00:00
|
|
|
self.debug('Got vips from ' + host,3)
|
2019-12-16 20:47:10 +00:00
|
|
|
vip_match = [ x for x in vips if x['destination'].split('/' + self._partition + '/')[1].startswith(subject)]
|
2019-12-09 18:57:22 +00:00
|
|
|
all_vips = [ x['destination'] for x in vips ]
|
|
|
|
if len(vip_match) > 0 :
|
|
|
|
selected_vip = vip_match[0]
|
2020-05-26 18:26:01 +00:00
|
|
|
self._return_payload.update({'vip': selected_vip})
|
2019-12-09 18:57:22 +00:00
|
|
|
pool_name = selected_vip['pool'].split('/' + self._partition + '/')[1]
|
|
|
|
pool_info = self.get_pool(host,auth_header,pool_name)
|
|
|
|
if pool_info :
|
2020-05-26 18:26:01 +00:00
|
|
|
self._return_payload.update(pool_info)
|
2019-12-09 18:57:22 +00:00
|
|
|
break
|
|
|
|
else :
|
2020-05-22 18:28:13 +00:00
|
|
|
self.debug('Unable to find VIP in VIPs from host ' + host,2)
|
2019-12-09 18:57:22 +00:00
|
|
|
else :
|
2019-12-16 20:47:10 +00:00
|
|
|
# Otherwise, get all nodes.
|
|
|
|
# If it's in the nodes, we have to start all over and find where.
|
|
|
|
# Take all vips and get all pools
|
|
|
|
# Check if address is in the pool. If it is, return the node, pool, vip, and host.
|
|
|
|
# Naw, I'll just give up.
|
2020-05-22 18:28:13 +00:00
|
|
|
self.debug('Unable to get VIPs from ' + host,2)
|
2020-07-14 21:36:46 +00:00
|
|
|
if len(self._return_payload) == 0 :
|
2020-05-26 18:26:01 +00:00
|
|
|
self._return_payload = None
|
|
|
|
return self._return_payload
|
2019-12-06 18:14:19 +00:00
|
|
|
|
|
|
|
def shutdown(self) :
|
|
|
|
return
|
|
|
|
|
|
|
|
|
2019-12-09 18:57:22 +00:00
|
|
|
def get_iq_auth_header(self,host,username,password) :
|
2020-05-22 18:28:13 +00:00
|
|
|
assumed_login_provider = 'tmos'
|
2019-12-09 18:57:22 +00:00
|
|
|
token = None
|
|
|
|
try :
|
2020-05-22 18:28:13 +00:00
|
|
|
payload = { 'username' : username, 'password' : password, 'loginProviderName' : assumed_login_provider }
|
2019-12-09 18:57:22 +00:00
|
|
|
login = requests.post('https://' + host + '/mgmt/shared/authn/login', json=payload,verify=False,timeout=2)
|
|
|
|
if 'token' in json.loads(login.text) :
|
|
|
|
token = json.loads(login.text)['token']['token']
|
|
|
|
except Exception as exception:
|
2020-05-22 18:28:13 +00:00
|
|
|
self.debug('Exception getting auth header for host ' + host + ': \n' + str(exception),2)
|
2019-12-09 18:57:22 +00:00
|
|
|
return None
|
|
|
|
if token :
|
2020-05-22 18:28:13 +00:00
|
|
|
return { 'X-F5-Auth-Token' : token }
|
2019-12-06 18:14:19 +00:00
|
|
|
else :
|
|
|
|
return None
|
|
|
|
|
2019-12-09 18:57:22 +00:00
|
|
|
def get_vip(self,host,header) :
|
|
|
|
"""Given a host and an authorization header, gets all the vips for that host"""
|
|
|
|
# Please don't give me empty auth headers, I don't want to check it everywhere
|
2020-05-22 18:28:13 +00:00
|
|
|
self.debug('Trying VIP info for ' + host,2)
|
2019-12-09 18:57:22 +00:00
|
|
|
try :
|
2020-05-22 18:28:13 +00:00
|
|
|
vips = requests.get('https://' + host + '/mgmt/tm/ltm/virtual',headers=header,verify=False,timeout=10)
|
2019-12-09 18:57:22 +00:00
|
|
|
if vips.status_code == 200 :
|
|
|
|
return json.loads(vips.text)['items']
|
2020-05-22 18:28:13 +00:00
|
|
|
self.debug('Response not OK for ' + vips.history[0].url,3)
|
2019-12-09 18:57:22 +00:00
|
|
|
return None
|
|
|
|
except Exception as exception :
|
2020-05-22 18:28:13 +00:00
|
|
|
self.debug('Exception getting VIPs for host ' + host + ': \n' + str(exception),2)
|
2019-12-09 18:57:22 +00:00
|
|
|
return None
|
2019-12-06 18:14:19 +00:00
|
|
|
|
2019-12-09 18:57:22 +00:00
|
|
|
def get_pool(self,host,header,pool) :
|
|
|
|
"""Given a host, auth header, and pool, return the pool and it's members from the host"""
|
|
|
|
try :
|
2020-05-22 18:28:13 +00:00
|
|
|
pools = requests.get('https://' + host + '/mgmt/tm/ltm/pool/~' + self._partition + '~' + pool,headers=header,verify=False,timeout=5)
|
2019-12-09 18:57:22 +00:00
|
|
|
if pools.status_code == 200 :
|
|
|
|
poolobj = json.loads(pools.text)
|
2020-05-22 18:28:13 +00:00
|
|
|
members = requests.get('https://' + host + '/mgmt/tm/ltm/pool/~' + self._partition + '~' + pool + '/members',headers=header,verify=False,timeout=10)
|
2019-12-09 18:57:22 +00:00
|
|
|
if members.status_code == 200 :
|
|
|
|
membersobj = json.loads(members.text)['items']
|
|
|
|
return {'pool':poolobj,'members':membersobj}
|
|
|
|
return {'pool':poolobj}
|
2020-05-22 18:28:13 +00:00
|
|
|
elf.debug('Response not OK for ' + pool.history[0].url,3)
|
2019-12-09 18:57:22 +00:00
|
|
|
return None
|
|
|
|
except Exception as exception:
|
2020-05-22 18:28:13 +00:00
|
|
|
self.debug('Exception getting pool for host ' + host + ': \n' + str(exception),2)
|
2019-12-09 18:57:22 +00:00
|
|
|
return None
|
2019-12-06 18:14:19 +00:00
|
|
|
|
2019-12-16 20:47:10 +00:00
|
|
|
def choose_host(self,subject) :
|
|
|
|
"""Given a subject, return the host that we suspect is responsible for the VIP resides in."""
|
2019-12-06 18:14:19 +00:00
|
|
|
hosts = {}
|
|
|
|
# Guess based on hostname
|
2019-12-16 20:47:10 +00:00
|
|
|
dcmatch = re.search('^[a-zA-Z]*-([a-zA-Z0-9]*)(-.*)*$',subject)
|
2019-12-06 18:14:19 +00:00
|
|
|
if dcmatch :
|
|
|
|
potentialdc = dcmatch.group(1)
|
|
|
|
else :
|
|
|
|
potentialdc = None
|
|
|
|
if potentialdc :
|
2020-05-26 18:26:01 +00:00
|
|
|
for host in self._hosts.keys() :
|
2019-12-06 18:14:19 +00:00
|
|
|
if potentialdc.lower() in host.lower() :
|
2020-05-26 18:26:01 +00:00
|
|
|
hosts.update({host:self._hosts[host]})
|
2020-05-28 18:00:50 +00:00
|
|
|
if len(hosts) > 0 :
|
2019-12-06 18:14:19 +00:00
|
|
|
return hosts
|
2020-05-22 18:28:13 +00:00
|
|
|
self.debug('Unable to determine DC from subject name: ' + subject,3)
|
2019-12-06 18:14:19 +00:00
|
|
|
# No hostname match, maybe we have info from bluecat?
|
2020-05-22 02:34:56 +00:00
|
|
|
if 'bluecat' in self.hints and subject in self.hints['bluecat'] and self.hints['bluecat'][subject]['object'] :
|
|
|
|
if 'locationCode' in self.hints['bluecat'][subject]['object'] :
|
|
|
|
potentialdc = self.hints['bluecat'][subject]['object']['locationCode'].split(' ')[-1]
|
|
|
|
elif 'parent' in self.hints['bluecat'][subject] and 'name' in self.hints['bluecat'][subject]['parent'] :
|
|
|
|
potentialdc = self.hints['bluecat'][subject]['parent']['name'].split(' ')[0]
|
2020-05-26 18:26:01 +00:00
|
|
|
for host in self._hosts.keys() :
|
2019-12-06 18:14:19 +00:00
|
|
|
if potentialdc.lower() in host.lower() :
|
2020-05-26 18:26:01 +00:00
|
|
|
hosts.update({host:self._hosts[host]})
|
2020-05-22 18:28:13 +00:00
|
|
|
self.debug('Got additional host info from the BlueCat plugin!',3)
|
2019-12-06 18:14:19 +00:00
|
|
|
return hosts
|
|
|
|
return hosts
|