Skip to content

Commit

Permalink
Fix lint
Browse files Browse the repository at this point in the history
Signed-off-by: Tushar <[email protected]>
  • Loading branch information
tgupta3 committed May 11, 2024
1 parent 5f7cedb commit 38eb6f0
Showing 1 changed file with 45 additions and 30 deletions.
75 changes: 45 additions & 30 deletions suzieq/poller/controller/source/vcenter.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,10 +8,8 @@

import asyncio
import logging
from requests.auth import HTTPBasicAuth
from requests.exceptions import RequestException
from typing import Any, Dict, List, Optional, Tuple, Union
from urllib.parse import urljoin, urlparse
from typing import Dict, List, Optional, Union
from urllib.parse import urlparse
from pyVim.connect import SmartConnect, Disconnect
from pyVmomi import vim, vmodl
import ssl
Expand Down Expand Up @@ -84,6 +82,7 @@ def validate_password(cls, password):
except SensitiveLoadError as e:
raise ValueError(e)


class Vcenter(Source, InventoryAsyncPlugin):
def __init__(self, config_data: dict, validate: bool = True) -> None:
self._status = 'init'
Expand All @@ -109,14 +108,17 @@ def _load(self, input_data):
)
self._server = self._data.server
if not self._auth:
raise InventorySourceError(f"{self.name} Vcenter must have an "
"'auth' set in the 'namespaces' section"
)
raise InventorySourceError(
f"{self.name} Vcenter must have an "
"'auth' set in the 'namespaces' section")

def _init_session(self):
"""Initialize the session property"""
context = ssl.SSLContext(ssl.PROTOCOL_SSLv23)
context.verify_mode = ssl.CERT_NONE if not self._data.ssl_verify else ssl.CERT_REQUIRED
context.verify_mode = ssl.CERT_REQUIRED
if not self._data.ssl_verify:
context.verify_mode = ssl.CERT_NONE

try:
self._session = SmartConnect(
host=self._server.host,
Expand All @@ -127,47 +129,56 @@ def _init_session(self):
)
except Exception as e:
self._session = None
raise InventorySourceError(f"Failed to connect to VCenter: {str(e)}")
raise InventorySourceError(
f"Failed to connect to VCenter: {str(e)}")

def _get_custom_keys(self, content, attribute_names):
"""Retrieve custom attribute keys based on their names."""
all_custom_fields = {field.name: field.key for field in content.customFieldsManager.field}
return [all_custom_fields[name] for name in attribute_names if name in all_custom_fields]
all_custom_fields = {field.name: field.key
for field in content.customFieldsManager.field}
return [
all_custom_fields[name]
for name in attribute_names
if name in all_custom_fields
]

def _create_filter_spec(self, view):
"""Create and return a FilterSpec based on provided view and attribute keys."""
"""Return a FilterSpec based on provided view and attribute keys."""
traversal_spec = vmodl.query.PropertyCollector.TraversalSpec(
name='traverseEntities',
path='view',
skip=False,
name='traverseEntities', path='view', skip=False,
type=vim.view.ContainerView,
selectSet=[vmodl.query.PropertyCollector.SelectionSpec(name='traverseEntities')]
)
prop_set = vmodl.query.PropertyCollector.PropertySpec(all=False, type=vim.VirtualMachine)
selectSet=[vmodl.query.PropertyCollector.SelectionSpec(
name='traverseEntities')])
prop_set = vmodl.query.PropertyCollector.PropertySpec(
all=False, type=vim.VirtualMachine)
prop_set.pathSet = ['name', 'guest.ipAddress', 'customValue']
obj_spec = vmodl.query.PropertyCollector.ObjectSpec(obj=view, selectSet=[traversal_spec])
obj_spec = vmodl.query.PropertyCollector.ObjectSpec(
obj=view, selectSet=[traversal_spec])
filter_spec = vmodl.query.PropertyCollector.FilterSpec()
filter_spec.objectSet = [obj_spec]
filter_spec.propSet = [prop_set]
return filter_spec

async def get_inventory_list(self) -> List:
"""
Retrieve VMs that have any of a list of specified custom attribute names using the Property Collector.
Retrieve VMs that have any specified custom attribute names.
This method uses vSphere's Property Collector to fetch only properties that are required.
This is a lot faster than fetching the entire inventory and filtering on attributes.
This method uses vSphere's Property Collector to fetch only
properties that are required. This is a lot faster than
fetching the entire inventory and filtering on attributes.
"""
if not self._session:
self._init_session()

content = self._session.RetrieveContent()
view = content.viewManager.CreateContainerView(content.rootFolder, [vim.VirtualMachine], True)
view = content.viewManager.CreateContainerView(
content.rootFolder, [vim.VirtualMachine], True)
attribute_keys = self._get_custom_keys(content, self._data.attributes)

filter_spec = self._create_filter_spec(view)
retrieve_options = vmodl.query.PropertyCollector.RetrieveOptions()
result = content.propertyCollector.RetrievePropertiesEx([filter_spec], retrieve_options)
result = content.propertyCollector.RetrievePropertiesEx(
[filter_spec], retrieve_options)
vms_with_ip = {}
while result:
for obj in result.objects:
Expand All @@ -180,20 +191,23 @@ async def get_inventory_list(self) -> List:
elif prop.name == 'guest.ipAddress' and prop.val:
vm_ip = prop.val
elif prop.name == 'customValue':
has_custom_attr = any(cv.key in attribute_keys for cv in prop.val)
has_custom_attr = any(
cv.key in attribute_keys for cv in prop.val)
if has_custom_attr and vm_ip:
vms_with_ip[vm_name] = vm_ip

if hasattr(result, 'token') and result.token:
result = content.propertyCollector.ContinueRetrievePropertiesEx(token=result.token)
property_collector = content.propertyCollector
result = property_collector.ContinueRetrievePropertiesEx(
token=result.token)
else:
break

view.Destroy()
logger.info(f'Vcenter: Retrieved {len(vms_with_ip)} VMs with IPs that have any of the specified attribute names')
logger.info(
f'Vcenter: Retrieved {len(vms_with_ip)} VMs with IPs')
return vms_with_ip


def parse_inventory(self, inventory_list: list) -> Dict:
inventory = {}
for name, ip in inventory_list.items():
Expand All @@ -203,7 +217,8 @@ def parse_inventory(self, inventory_list: list) -> Dict:
'namespace': namespace,
'hostname': name,
}
logger.info(f'Vcenter: Acting on inventory of {len(inventory)} devices')
logger.info(
f'Vcenter: Acting on inventory of {len(inventory)} devices')
return inventory

async def _execute(self):
Expand All @@ -217,4 +232,4 @@ async def _execute(self):

async def _stop(self):
if self._session:
Disconnect(self._session)
Disconnect(self._session)

0 comments on commit 38eb6f0

Please sign in to comment.