From ab139be008693ecfe8b56646d4c5c328a59b82f0 Mon Sep 17 00:00:00 2001 From: Shalin Patel Date: Tue, 6 Aug 2024 19:01:51 -0700 Subject: [PATCH 1/4] feat: fetch nutanix VMs --- plugins/nutanix/MANIFEST.in | 1 + plugins/nutanix/Makefile | 52 +++++ plugins/nutanix/README.md | 32 +++ .../nutanix/fix_plugin_nutanix/__init__.py | 209 ++++++++++++++++++ .../nutanix/fix_plugin_nutanix/collector.py | 2 + plugins/nutanix/pyproject.toml | 44 ++++ plugins/nutanix/setup.cfg | 10 + plugins/nutanix/test/test_args.py | 13 ++ plugins/nutanix/test/test_collector.py | 22 ++ plugins/nutanix/test/test_config.py | 11 + plugins/nutanix/tox.ini | 39 ++++ requirements-all.txt | 5 +- 12 files changed, 439 insertions(+), 1 deletion(-) create mode 100644 plugins/nutanix/MANIFEST.in create mode 100644 plugins/nutanix/Makefile create mode 100644 plugins/nutanix/README.md create mode 100644 plugins/nutanix/fix_plugin_nutanix/__init__.py create mode 100644 plugins/nutanix/fix_plugin_nutanix/collector.py create mode 100644 plugins/nutanix/pyproject.toml create mode 100644 plugins/nutanix/setup.cfg create mode 100644 plugins/nutanix/test/test_args.py create mode 100644 plugins/nutanix/test/test_collector.py create mode 100644 plugins/nutanix/test/test_config.py create mode 100644 plugins/nutanix/tox.ini diff --git a/plugins/nutanix/MANIFEST.in b/plugins/nutanix/MANIFEST.in new file mode 100644 index 0000000000..bb3ec5f0d4 --- /dev/null +++ b/plugins/nutanix/MANIFEST.in @@ -0,0 +1 @@ +include README.md diff --git a/plugins/nutanix/Makefile b/plugins/nutanix/Makefile new file mode 100644 index 0000000000..83c19145d1 --- /dev/null +++ b/plugins/nutanix/Makefile @@ -0,0 +1,52 @@ +.PHONY: clean clean-test clean-pyc clean-build docs help test +.DEFAULT_GOAL := help +.SILENT: clean clean-build clean-pyc clean-test + +clean: clean-build clean-pyc clean-test ## remove all build, test, coverage and Python artifacts + +clean-build: ## remove build artifacts + rm -fr build/ + rm -fr out/ + rm -fr gen/ + rm -fr dist/ + rm -fr .eggs/ + rm -fr .hypothesis/ + rm -fr .mypy_cache/ + find . -name '*.egg-info' -exec rm -fr {} + + find . -name '*.egg' -exec rm -fr {} + + +clean-pyc: ## remove Python file artifacts + find . -name '*.pyc' -exec rm -f {} + + find . -name '*.pyo' -exec rm -f {} + + find . -name '*~' -exec rm -f {} + + find . -name '__pycache__' -exec rm -fr {} + + +clean-test: ## remove test and coverage artifacts + rm -fr .tox/ + rm -f .coverage + rm -fr htmlcov/ + rm -fr .pytest_cache + +lint: ## static code analysis + black --line-length 120 --check fix_plugin_nutanix test + flake8 fix_plugin_nutanix + mypy --python-version 3.11 --strict --install-types fix_plugin_nutanix test + +test: ## run tests quickly with the default Python + pytest + +test-all: ## run tests on every Python version with tox + tox + +coverage: ## check code coverage quickly with the default Python + coverage run --source fix_plugin_aws -m pytest + coverage combine + coverage report -m + coverage html + $(BROWSER) htmlcov/index.html + +list-outdated: + pip list --outdated + +install-latest: + pip list --outdated --format=freeze | grep -v '^\-e' | cut -d = -f 1 | xargs -n1 pip install -U diff --git a/plugins/nutanix/README.md b/plugins/nutanix/README.md new file mode 100644 index 0000000000..41168c27ed --- /dev/null +++ b/plugins/nutanix/README.md @@ -0,0 +1,32 @@ +# fix-plugin-aws +An AWS collector plugin for Fix. + +## Usage +For details on how to edit configuration, please see [the documentation](https://inventory.fix.security/docs/getting-started/configuring-fix). + +When the collector is enabled (`fixworker.collector = [aws]`) it will automatically collect any accounts the AWS boto3 SDK can authenticate for. +By default it will check for environment variables like `AWS_ACCESS_KEY_ID`, `AWS_SECRET_ACCESS_KEY` or `AWS_SESSION_TOKEN`. + +If Fix should assume an IAM role that role can be given via `fixworker.aws.role = SomeRoleName`. + +The collector will scrape resources in all regions unless regions are specified using e.g. `fixworker.aws.region = [us-east-1, us-west-2]`. + + +## Scraping multiple accounts +If the given credentials are allowed to assume the specified role in other accounts of your AWS organisation, Fix +can collect multiple accounts at the same time. To do so provide the account IDs to the `fixworker.aws.account` configuration. + +## Scraping the entire organisation +Instead of giving a list of account IDs manually you could also specify `fixworker.aws.scrape_org`, which will make Fix try to get the list of all accounts using the [ListAccounts](https://docs.aws.amazon.com/organizations/latest/APIReference/API_ListAccounts.html) API. + +If certain accounts are to be excluded from that list they can be specified using the `fixworker.aws.scrape_exclude_account` config option. + +## Miscellaneous Options +When collecting multiple accounts Fix by default will collect the accounts it finds in the org as well as the one it is currently authenticated as. +If you do not want it to scrape the account that was used to get the list of all org accounts (e.g. your root account) you can specify `fixworker.aws.dont_scrape_current`. + +If instead of using the current credentials you would like Fix to assume the specified role (`fixworker.aws.role`) even for the current account you can specify the options +`fixworker.aws.assume_current` and `fixworker.aws.dont_scrape_current`. This would make it so that Fix does not scrape the current account using default credentials but instead assume the specified IAM role even for the current account. + +## License +See [LICENSE](../../LICENSE) for details. diff --git a/plugins/nutanix/fix_plugin_nutanix/__init__.py b/plugins/nutanix/fix_plugin_nutanix/__init__.py new file mode 100644 index 0000000000..c94143ea9b --- /dev/null +++ b/plugins/nutanix/fix_plugin_nutanix/__init__.py @@ -0,0 +1,209 @@ +import fixlib.logger +import os +import ntnx_prism_py_client +import ntnx_clustermgmt_py_client +import ntnx_vmm_py_client + +from ntnx_prism_py_client import Configuration as PrismConfiguration +from ntnx_prism_py_client import ApiClient as PrismClient + +from ntnx_clustermgmt_py_client import Configuration as ClusterConfiguration +from ntnx_clustermgmt_py_client import ApiClient as ClusterClient + +from ntnx_vmm_py_client import Configuration as VMMConfiguration +from ntnx_vmm_py_client import ApiClient as VMMClient + +from attrs import define, field +from datetime import datetime +from typing import ClassVar, Dict, List, Optional +from fixlib.baseplugin import BaseCollectorPlugin +from fixlib.graph import ByNodeId, Graph, EdgeType, BySearchCriteria +from fixlib.args import ArgumentParser +from fixlib.config import Config +from fixlib.baseresources import ( + BaseAccount, + BaseRegion, + BaseInstance, + BaseNetwork, + BaseResource, + BaseVolume, + InstanceStatus, + VolumeStatus, +) + +log = fixlib.logger.getLogger("fix." + __name__) + + +class NutanixCollectorPlugin(BaseCollectorPlugin): + # The cloud attribute is used to identify the cloud provider in the fix data model + # The BaseCollectorPlugin will use this create a new cloud node in the graph + cloud = "nutanix" + + def collect(self) -> None: + """This method is being called by fix whenever the collector runs + + It is responsible for querying the cloud APIs for remote resources and adding + them to the plugin graph. + The graph root (self.graph.root) must always be followed by one or more + accounts. An account must always be followed by a region. + A region can contain arbitrary resources. + """ + log.debug("plugin: collecting nutanix resources") + + sherlockDev = PrismCentral(id="sherlock_dev", name="Sherlock Dev", tags={"url": "https://prismcentral.dev.ntnxsherlock.com:9440/"}) + self.graph.add_resource(self.graph.root, sherlockDev) + + bowser = PrismElement(id="bowser", name="bowser", tags={"Some Tag": "Some Value"}) + self.graph.add_resource(sherlockDev, bowser) + self.collect_virtual_machines() + log.info(f"graph: {self.graph.export_model()}") + + @staticmethod + def add_args(arg_parser: ArgumentParser) -> None: + """Example of how to use the ArgumentParser + + Can be accessed via ArgumentParser.args.example_arg + Note: almost all plugin config should be done via add_config() + so it can be changed centrally and at runtime. + """ + # arg_parser.add_argument( + # "--example-arg", + # help="Example Argument", + # dest="example_arg", + # type=str, + # default=None, + # nargs="+", + # ) + pass + + @staticmethod + def add_config(config: Config) -> None: + """Add any plugin config to the global config store. + + Method called by the PluginLoader upon plugin initialization. + Can be used to introduce plugin config arguments to the global config store. + """ + # config.add_config(ExampleConfig) + pass + + def collect_virtual_machines(self) -> None: + """Example of how to collect resources in a region""" + # Get all virtual machines in the PE + vmmClient = vmm_client("prismcentral.dev.ntnxsherlock.com", False) + vmm_instance = ntnx_vmm_py_client.api.VmApi(vmmClient) + print("\nGet all virtual machines in the PE\n") + response = vmm_instance.list_vms() + log.info(f"Found virtual machines: {response.metadata.total_available_results}") + for vm in response.data: + log.info(f"VM: {vm.name}" f", uuid: {vm.ext_id}" f", power_state: {vm.power_state}" f", create_time: {vm.create_time}") + + + + +@define +class ExampleConfig: + """Example of how to use the fixcore config service + + Can be accessed via Config.example.region + """ + + kind: ClassVar[str] = "example" + region: Optional[List[str]] = field(default=None, metadata={"description": "Example Region"}) + + +@define(eq=False, slots=False) +class PrismCentral(BaseAccount): + """Prism Central Account""" + + kind: ClassVar[str] = "prism_central" + + def delete(self, graph: Graph) -> bool: + return NotImplemented + + +@define(eq=False, slots=False) +class PrismElement(BaseRegion): + """Prism Element""" + + kind: ClassVar[str] = "prism_element" + + def delete(self, graph: Graph) -> bool: + """PEs can usually not be deleted so we return NotImplemented""" + return NotImplemented + + +@define(eq=False, slots=False) +class NutanixResource(BaseResource): + """A class that implements the abstract method delete() as well as update_tag() + and delete_tag(). + + delete() must be implemented. update_tag() and delete_tag() are optional. + """ + + kind: ClassVar[str] = "nutanix_resource" + kind_display: ClassVar[str] = "Nutanix Resource" + + def delete(self, graph: Graph) -> bool: + """Delete a resource in the cloud""" + log.debug(f"Deleting resource {self.id} in account {self.account(graph).id} region {self.region(graph).id}") + return True + + def update_tag(self, key, value) -> bool: + """Update a resource tag in the cloud""" + log.debug(f"Updating or setting tag {key}: {value} on resource {self.id}") + return True + + def delete_tag(self, key) -> bool: + """Delete a resource tag in the cloud""" + log.debug(f"Deleting tag {key} on resource {self.id}") + return True + + +@define(eq=False, slots=False) +class NutanixVMs(NutanixResource): + """An Example Instance Resource""" + + kind: ClassVar[str] = "ViMachines" + + + +def prism_client(endpoint: str, insecure: bool) -> PrismClient: + prism_config = PrismConfiguration() + prism_config.host = endpoint + prism_config.verify_ssl = not insecure + prism_config.username = os.environ.get("NUTANIX_USER") + prism_config.password = os.environ.get("NUTANIX_PASSWORD") + prism_client = PrismClient(configuration=prism_config) + prism_client.add_default_header( + header_name="Accept-Encoding", header_value="gzip, deflate, br" + ) + prism_client.add_default_header(header_name="Content-Type", header_value="application/json") + return prism_client + + +def cluster_client(endpoint: str, insecure: bool) -> ClusterClient: + cluster_config = ClusterConfiguration() + cluster_config.host = endpoint + cluster_config.verify_ssl = not insecure + cluster_config.username = os.getenv("NUTANIX_USER") + cluster_config.password = os.getenv("NUTANIX_PASSWORD") + cluster_client = PrismClient(configuration=cluster_config) + cluster_client.add_default_header( + header_name="Accept-Encoding", header_value="gzip, deflate, br" + ) + cluster_client.add_default_header(header_name="Content-Type", header_value="application/json") + return cluster_client + +def vmm_client(endpoint: str, insecure: bool) -> VMMClient: + vmm_config = VMMConfiguration() + vmm_config.host = endpoint + vmm_config.port = 9440 + vmm_config.verify_ssl = not insecure + vmm_config.username = os.getenv("NUTANIX_USER") + vmm_config.password = os.getenv("NUTANIX_PASSWORD") + vmm_client = VMMClient(configuration=vmm_config) + vmm_client.add_default_header( + header_name="Accept-Encoding", header_value="gzip, deflate, br" + ) + vmm_client.add_default_header(header_name="Content-Type", header_value="application/json") + return vmm_client \ No newline at end of file diff --git a/plugins/nutanix/fix_plugin_nutanix/collector.py b/plugins/nutanix/fix_plugin_nutanix/collector.py new file mode 100644 index 0000000000..139597f9cb --- /dev/null +++ b/plugins/nutanix/fix_plugin_nutanix/collector.py @@ -0,0 +1,2 @@ + + diff --git a/plugins/nutanix/pyproject.toml b/plugins/nutanix/pyproject.toml new file mode 100644 index 0000000000..15bafbcb66 --- /dev/null +++ b/plugins/nutanix/pyproject.toml @@ -0,0 +1,44 @@ +[project] +name = "fixinventory-plugin-nutanix" +version = "0.1.0" +authors = [{name="supershal"}] +description = "Runs collector plugins and sends the result to fixcore." +license = { text="AGPLv3" } +requires-python = ">=3.11" +classifiers = [ + # Current project status + "Development Status :: 5 - Production/Stable", + # Audience + "Intended Audience :: System Administrators", + "Intended Audience :: Information Technology", + # License information + "License :: OSI Approved :: GNU Affero General Public License v3 or later (AGPLv3+)", + # Supported python versions + "Programming Language :: Python :: 3.11", + # Supported OS's + "Operating System :: POSIX :: Linux", + "Operating System :: Unix", + # Extra metadata + "Environment :: Console", + "Natural Language :: English", + "Topic :: Security", + "Topic :: Utilities", +] +readme = {file="README.md", content-type="text/markdown"} + +dependencies = [ + "fixinventorylib==4.1.0", + "fixinventorydata", + "retrying", +] + +[project.entry-points."fix.plugins"] +nutanix = "fix_plugin_nutanix:NutanixCollectorPlugin" + +[project.urls] +Documentation = "https://inventory.fix.security" +Source = "https://github.com/someengineering/fix/tree/main/plugins/nutanix" + +[build-system] +requires = ["setuptools>=67.8.0", "wheel>=0.40.0", "build>=0.10.0"] +build-backend = "setuptools.build_meta" diff --git a/plugins/nutanix/setup.cfg b/plugins/nutanix/setup.cfg new file mode 100644 index 0000000000..a8ae35ab70 --- /dev/null +++ b/plugins/nutanix/setup.cfg @@ -0,0 +1,10 @@ +[options] +packages = find: +include_package_data = True +zip_safe = False + +[aliases] +test=pytest + +[mypy] +ignore_missing_imports = True diff --git a/plugins/nutanix/test/test_args.py b/plugins/nutanix/test/test_args.py new file mode 100644 index 0000000000..4e8abba4be --- /dev/null +++ b/plugins/nutanix/test/test_args.py @@ -0,0 +1,13 @@ +# from fixlib.args import get_arg_parser +# from fix_plugin_nutanix import NutanixCollectorPlugin + +# # from fixlib.args import ArgumentParser + + +# def test_args(): +# arg_parser = get_arg_parser() +# NutanixCollectorPlugin.add_args(arg_parser) +# arg_parser.parse_args() + + +# # assert ArgumentParser.args.example_arg is None diff --git a/plugins/nutanix/test/test_collector.py b/plugins/nutanix/test/test_collector.py new file mode 100644 index 0000000000..4ed14f63b6 --- /dev/null +++ b/plugins/nutanix/test/test_collector.py @@ -0,0 +1,22 @@ +import logging +from typing import Dict, Any, List, cast + +from fixlib.baseresources import GraphRoot + +from fixlib.graph import Graph +from fixlib.graph import sanitize +from fix_plugin_nutanix import NutanixCollectorPlugin + + +log = logging.getLogger("fix." + __name__) + +def prepare_graph() -> Graph: + plugin_instance = NutanixCollectorPlugin() + plugin_instance.collect() + graph = Graph(root=GraphRoot(id="root", tags={})) + graph.merge(plugin_instance.graph) + sanitize(graph) + return graph + +def test_collect_nutanix() -> None: + graph = prepare_graph() \ No newline at end of file diff --git a/plugins/nutanix/test/test_config.py b/plugins/nutanix/test/test_config.py new file mode 100644 index 0000000000..f93272b805 --- /dev/null +++ b/plugins/nutanix/test/test_config.py @@ -0,0 +1,11 @@ +# from fixlib.config import Config +# from fix_plugin_nutanix import NutanixCollectorPlugin + + +# def test_config(): +# config = Config("dummy", "dummy") +# NutanixCollectorPlugin.add_config(config) +# Config.init_default_config() + + +# # assert Config.example.region is None diff --git a/plugins/nutanix/tox.ini b/plugins/nutanix/tox.ini new file mode 100644 index 0000000000..100bd097db --- /dev/null +++ b/plugins/nutanix/tox.ini @@ -0,0 +1,39 @@ +[tox] +#env_list = syntax, tests, black, mypy +env_list = syntax, tests + +[flake8] +max-line-length=120 +exclude = .git,.tox,__pycache__,.idea,.pytest_cache +ignore=F403, F405, E722, N806, N813, E266, W503, E203, F811, E501 + +[pytest] +testpaths = test +asyncio_mode=auto +log_cli = true +log_cli_level = INFO + +[testenv] +usedevelop = true +deps = + -r../../requirements-all.txt + --editable=file:///{toxinidir}/../../fixlib + --editable=file:///{toxinidir}/../nutanix +# until this is fixed: https://github.com/pypa/setuptools/issues/3518 +setenv = + SETUPTOOLS_ENABLE_FEATURES = legacy-editable +passenv = + NUTANIX_USER + NUTANIX_PASSWORD + +[testenv:syntax] +commands = flake8 + +[testenv:tests] +commands= pytest + +[testenv:black] +commands = black --line-length 120 --check --diff --target-version py39 . + +[testenv:mypy] +commands= python -m mypy --install-types --non-interactive --python-version 3.11 --strict fix_plugin_nutanix test diff --git a/requirements-all.txt b/requirements-all.txt index d47262cb86..6d1b31d591 100644 --- a/requirements-all.txt +++ b/requirements-all.txt @@ -27,7 +27,7 @@ build==1.2.1 cachetools==5.3.3 cattrs==23.2.3 cerberus==1.3.5 -certifi==2024.7.4 +certifi==2020.4.5.1 cffi==1.16.0 chardet==5.2.0 charset-normalizer==3.3.2 @@ -190,3 +190,6 @@ wrapt==1.16.0 yarl==1.9.4 zc-lockfile==3.0.post1 zipp==3.19.2 +ntnx_vmm_py_client==4.0.3a1 +ntnx_prism_py_client==4.0.3a2 +ntnx_clustermgmt_py_client==4.0.2a2 \ No newline at end of file From ad884a40cfc5453bb01e3663001bb6e6ee0ec847 Mon Sep 17 00:00:00 2001 From: Shalin Patel Date: Fri, 9 Aug 2024 17:02:55 -0700 Subject: [PATCH 2/4] fix: prism collector config --- .../nutanix/fix_plugin_nutanix/__init__.py | 213 +++++------------- .../nutanix/fix_plugin_nutanix/collector.py | 130 +++++++++++ plugins/nutanix/fix_plugin_nutanix/config.py | 37 +++ .../nutanix/fix_plugin_nutanix/resources.py | 72 ++++++ plugins/nutanix/test/test_args.py | 13 -- plugins/nutanix/test/test_collector.py | 36 ++- plugins/nutanix/test/test_config.py | 17 +- plugins/nutanix/tox.ini | 1 + requirements-extra.txt | 5 +- 9 files changed, 341 insertions(+), 183 deletions(-) create mode 100644 plugins/nutanix/fix_plugin_nutanix/config.py create mode 100644 plugins/nutanix/fix_plugin_nutanix/resources.py delete mode 100644 plugins/nutanix/test/test_args.py diff --git a/plugins/nutanix/fix_plugin_nutanix/__init__.py b/plugins/nutanix/fix_plugin_nutanix/__init__.py index c94143ea9b..4af331146d 100644 --- a/plugins/nutanix/fix_plugin_nutanix/__init__.py +++ b/plugins/nutanix/fix_plugin_nutanix/__init__.py @@ -1,11 +1,5 @@ import fixlib.logger import os -import ntnx_prism_py_client -import ntnx_clustermgmt_py_client -import ntnx_vmm_py_client - -from ntnx_prism_py_client import Configuration as PrismConfiguration -from ntnx_prism_py_client import ApiClient as PrismClient from ntnx_clustermgmt_py_client import Configuration as ClusterConfiguration from ntnx_clustermgmt_py_client import ApiClient as ClusterClient @@ -14,30 +8,21 @@ from ntnx_vmm_py_client import ApiClient as VMMClient from attrs import define, field -from datetime import datetime -from typing import ClassVar, Dict, List, Optional +from typing import List, Optional, cast from fixlib.baseplugin import BaseCollectorPlugin -from fixlib.graph import ByNodeId, Graph, EdgeType, BySearchCriteria +from fixlib.graph import Graph from fixlib.args import ArgumentParser from fixlib.config import Config -from fixlib.baseresources import ( - BaseAccount, - BaseRegion, - BaseInstance, - BaseNetwork, - BaseResource, - BaseVolume, - InstanceStatus, - VolumeStatus, -) + +from fix_plugin_nutanix.collector import PrismCentralCollector +from fix_plugin_nutanix.resources import PrismCentralAccount +from fix_plugin_nutanix.config import PrismCentalCredentials, PrismCentralColletorConfig log = fixlib.logger.getLogger("fix." + __name__) class NutanixCollectorPlugin(BaseCollectorPlugin): - # The cloud attribute is used to identify the cloud provider in the fix data model - # The BaseCollectorPlugin will use this create a new cloud node in the graph - cloud = "nutanix" + """Nutanix Collector Plugin""" def collect(self) -> None: """This method is being called by fix whenever the collector runs @@ -50,13 +35,37 @@ def collect(self) -> None: """ log.debug("plugin: collecting nutanix resources") - sherlockDev = PrismCentral(id="sherlock_dev", name="Sherlock Dev", tags={"url": "https://prismcentral.dev.ntnxsherlock.com:9440/"}) - self.graph.add_resource(self.graph.root, sherlockDev) + pcConfigs = cast( + List[PrismCentalCredentials], Config.PrismCentralColletorConfig.credentials + ) + for pc in pcConfigs: + pcAccount = PrismCentralAccount( + id=pc.name.replace(" ", "_"), + name=pc.name, + endpoint=pc.endpoint, + username=pc.username, + password=pc.password, + ) + pc_graph = self.collect_pc(pcAccount) + self.graph.add_resource(self.graph.root, pc_graph) + + def collect_pc(self, prismCentral: PrismCentralAccount) -> Optional[Graph]: + log.info(f"Collecting data from Nutanix Prism Central {prismCentral.name}") + vmmClient = vmm_client(prismCentral) + clusterClient = cluster_client(prismCentral) + prismCentralCollector = PrismCentralCollector( + prismCentral, vmmClient, clusterClient + ) + return prismCentralCollector.collect() - bowser = PrismElement(id="bowser", name="bowser", tags={"Some Tag": "Some Value"}) - self.graph.add_resource(sherlockDev, bowser) - self.collect_virtual_machines() - log.info(f"graph: {self.graph.export_model()}") + @staticmethod + def add_config(config: Config) -> None: + """Add any plugin config to the global config store. + + Method called by the PluginLoader upon plugin initialization. + Can be used to introduce plugin config arguments to the global config store. + """ + config.add_config(PrismCentralColletorConfig) @staticmethod def add_args(arg_parser: ArgumentParser) -> None: @@ -66,144 +75,38 @@ def add_args(arg_parser: ArgumentParser) -> None: Note: almost all plugin config should be done via add_config() so it can be changed centrally and at runtime. """ - # arg_parser.add_argument( - # "--example-arg", - # help="Example Argument", - # dest="example_arg", - # type=str, - # default=None, - # nargs="+", - # ) - pass - - @staticmethod - def add_config(config: Config) -> None: - """Add any plugin config to the global config store. - - Method called by the PluginLoader upon plugin initialization. - Can be used to introduce plugin config arguments to the global config store. - """ - # config.add_config(ExampleConfig) pass - - def collect_virtual_machines(self) -> None: - """Example of how to collect resources in a region""" - # Get all virtual machines in the PE - vmmClient = vmm_client("prismcentral.dev.ntnxsherlock.com", False) - vmm_instance = ntnx_vmm_py_client.api.VmApi(vmmClient) - print("\nGet all virtual machines in the PE\n") - response = vmm_instance.list_vms() - log.info(f"Found virtual machines: {response.metadata.total_available_results}") - for vm in response.data: - log.info(f"VM: {vm.name}" f", uuid: {vm.ext_id}" f", power_state: {vm.power_state}" f", create_time: {vm.create_time}") - - - - -@define -class ExampleConfig: - """Example of how to use the fixcore config service - - Can be accessed via Config.example.region - """ - - kind: ClassVar[str] = "example" - region: Optional[List[str]] = field(default=None, metadata={"description": "Example Region"}) - - -@define(eq=False, slots=False) -class PrismCentral(BaseAccount): - """Prism Central Account""" - kind: ClassVar[str] = "prism_central" - def delete(self, graph: Graph) -> bool: - return NotImplemented - - -@define(eq=False, slots=False) -class PrismElement(BaseRegion): - """Prism Element""" - - kind: ClassVar[str] = "prism_element" - - def delete(self, graph: Graph) -> bool: - """PEs can usually not be deleted so we return NotImplemented""" - return NotImplemented - - -@define(eq=False, slots=False) -class NutanixResource(BaseResource): - """A class that implements the abstract method delete() as well as update_tag() - and delete_tag(). - - delete() must be implemented. update_tag() and delete_tag() are optional. - """ - - kind: ClassVar[str] = "nutanix_resource" - kind_display: ClassVar[str] = "Nutanix Resource" - - def delete(self, graph: Graph) -> bool: - """Delete a resource in the cloud""" - log.debug(f"Deleting resource {self.id} in account {self.account(graph).id} region {self.region(graph).id}") - return True - - def update_tag(self, key, value) -> bool: - """Update a resource tag in the cloud""" - log.debug(f"Updating or setting tag {key}: {value} on resource {self.id}") - return True - - def delete_tag(self, key) -> bool: - """Delete a resource tag in the cloud""" - log.debug(f"Deleting tag {key} on resource {self.id}") - return True - - -@define(eq=False, slots=False) -class NutanixVMs(NutanixResource): - """An Example Instance Resource""" - - kind: ClassVar[str] = "ViMachines" - - - -def prism_client(endpoint: str, insecure: bool) -> PrismClient: - prism_config = PrismConfiguration() - prism_config.host = endpoint - prism_config.verify_ssl = not insecure - prism_config.username = os.environ.get("NUTANIX_USER") - prism_config.password = os.environ.get("NUTANIX_PASSWORD") - prism_client = PrismClient(configuration=prism_config) - prism_client.add_default_header( - header_name="Accept-Encoding", header_value="gzip, deflate, br" - ) - prism_client.add_default_header(header_name="Content-Type", header_value="application/json") - return prism_client - - -def cluster_client(endpoint: str, insecure: bool) -> ClusterClient: +def cluster_client(pc: PrismCentralAccount) -> ClusterClient: cluster_config = ClusterConfiguration() - cluster_config.host = endpoint - cluster_config.verify_ssl = not insecure - cluster_config.username = os.getenv("NUTANIX_USER") - cluster_config.password = os.getenv("NUTANIX_PASSWORD") - cluster_client = PrismClient(configuration=cluster_config) + cluster_config.host = pc.endpoint + cluster_config.port = pc.port + cluster_config.verify_ssl = not pc.insecure + cluster_config.username = pc.username + cluster_config.password = pc.password + cluster_client = ClusterClient(configuration=cluster_config) cluster_client.add_default_header( header_name="Accept-Encoding", header_value="gzip, deflate, br" ) - cluster_client.add_default_header(header_name="Content-Type", header_value="application/json") + cluster_client.add_default_header( + header_name="Content-Type", header_value="application/json" + ) return cluster_client -def vmm_client(endpoint: str, insecure: bool) -> VMMClient: + +def vmm_client(pc: PrismCentralAccount) -> VMMClient: vmm_config = VMMConfiguration() - vmm_config.host = endpoint - vmm_config.port = 9440 - vmm_config.verify_ssl = not insecure - vmm_config.username = os.getenv("NUTANIX_USER") - vmm_config.password = os.getenv("NUTANIX_PASSWORD") + vmm_config.host = pc.endpoint + vmm_config.port = pc.port + vmm_config.verify_ssl = not pc.insecure + vmm_config.username = pc.username + vmm_config.password = pc.password vmm_client = VMMClient(configuration=vmm_config) vmm_client.add_default_header( header_name="Accept-Encoding", header_value="gzip, deflate, br" ) - vmm_client.add_default_header(header_name="Content-Type", header_value="application/json") - return vmm_client \ No newline at end of file + vmm_client.add_default_header( + header_name="Content-Type", header_value="application/json" + ) + return vmm_client diff --git a/plugins/nutanix/fix_plugin_nutanix/collector.py b/plugins/nutanix/fix_plugin_nutanix/collector.py index 139597f9cb..75092e4b25 100644 --- a/plugins/nutanix/fix_plugin_nutanix/collector.py +++ b/plugins/nutanix/fix_plugin_nutanix/collector.py @@ -1,2 +1,132 @@ +import logging +import json +import ntnx_clustermgmt_py_client +from ntnx_clustermgmt_py_client import Configuration as ClusterConfiguration +from ntnx_clustermgmt_py_client import ApiClient as ClusterClient +from typing import Tuple, Type, List, Dict, Callable, Any, Optional, cast, DefaultDict + +import ntnx_vmm_py_client +from ntnx_vmm_py_client import Configuration as VMMConfiguration +from ntnx_vmm_py_client import ApiClient as VMMClient +from fix_plugin_nutanix.resources import ( + PrismCentralAccount, + PrismElement, + ViraualMachine, +) +from fixlib.graph import Graph + +log = logging.getLogger("fix." + __name__) + + +class PrismCentralCollector: + """ + Collects data from single Nutanix Prism Central instance + """ + + def __init__( + self, + prismCentral: PrismCentralAccount, + vmmClient: VMMClient, + clusterClient: ClusterClient, + ) -> None: + self.vmmClient = vmmClient + self.clusterClient = clusterClient + self.prismCentral = prismCentral + # pe_collectors collectors that are always collected + self.mandatoryCollectors: List[Tuple[str, Callable[..., None]]] = [ + ("prism_elements", self.collect_prism_element) + ] + # Global collectors are resources that are specified across all PE + self.globalCollectors: List[Tuple[str, Callable[..., None]]] = [ + # ("images", self.collect_images) + ] + # Prism Element collectors are resources that are specific to PE + self.prismElementCollectors: List[Tuple[str, Callable[..., None]]] = [ + ("virtual_machines", self.collect_virtual_machines) + ] + self.allCollectors = dict(self.mandatoryCollectors) + self.allCollectors.update(dict(self.globalCollectors)) + self.allCollectors.update(dict(self.prismElementCollectors)) + self.collecter_set = set(self.allCollectors.keys()) + self.graph = Graph(root=self.prismCentral) + + def collect(self): + """ + Runs resource collectors across all PE + Resource collectors add their resources to the local `self.graph` graph + """ + log.info("Collecting data from Nutanix Prism Central") + collectors = set(self.collecter_set) + for collectorName, collector in self.mandatoryCollectors: + if collectorName in collectors: + log.info( + f"Running collector: {collectorName} in {self.prismCentral.name}" + ) + collector() + + prismElements = [pe for pe in self.graph.nodes if isinstance(pe, PrismElement)] + for collectorName, collector in self.prismElementCollectors: + for pe in prismElements: + if collectorName in collectors: + log.info(f"Running collector: {collectorName} in {pe.name}") + collector(pe) + + return self.graph + + def collect_prism_element(self) -> None: + """ + Collects data from all Prism Elements + """ + log.info("Collecting data from all Prism Elements") + clusterApi = ntnx_clustermgmt_py_client.api.ClusterApi(self.clusterClient) + clusters = clusterApi.get_clusters() + log.info(f"Found clusters: {clusters.metadata.total_available_results}") + for cluster in clusters.data: + # log.info(f"Cluster: {cluster}") + # log.info(f"uuid: {cluster['extId']}" f", name: {cluster['name']}") + pe = PrismElement( + id=cluster["extId"], + name=cluster["name"], + tags={"cluster_uuid": cluster["extId"]}, + ) + self.graph.add_resource(self.prismCentral, pe) + + def collect_images(self) -> None: + """ + Collects data from all images + """ + log.info("Collecting data from all images") + + def collect_virtual_machines(self, pe: PrismElement) -> None: + """ + Collects data from all virtual machines + """ + log.info("Collecting data from all virtual machines") + # Get all virtual machines in the PE + vmm_instance = ntnx_vmm_py_client.api.VmApi(self.vmmClient) + filter = f"contains(cluster/extId, '{pe.id}')" + response = vmm_instance.list_vms(_filter=filter) + log.info( + f"{pe.name}: Found virtual machines: {response.metadata.total_available_results}" + ) + if response.metadata.total_available_results == 0: + return + for vm in response.data: + # log.info( + # f"VM: {vm.name}" + # f", uuid: {vm.ext_id}" + # f", power_state: {vm.power_state}" + # f", create_time: {vm.create_time}" + # ) + # log.info(f"VM: {vm}") + vm = ViraualMachine( + id=vm.ext_id, + name=vm.name, + tags={"power_state": vm.power_state}, + power_state=vm.power_state, + ctime=vm.create_time, + mtime=vm.update_time, + ) + self.graph.add_resource(pe, vm) diff --git a/plugins/nutanix/fix_plugin_nutanix/config.py b/plugins/nutanix/fix_plugin_nutanix/config.py new file mode 100644 index 0000000000..f6c8bae6f4 --- /dev/null +++ b/plugins/nutanix/fix_plugin_nutanix/config.py @@ -0,0 +1,37 @@ +from attrs import define, field +from typing import List, ClassVar, Optional + + +@define +class PrismCentalCredentials: + kind: ClassVar[str] = "prism_central_credentials" + name: str = field( + metadata={"description": "A NickName/Identifier for Prism Central"} + ) + endpoint: str = field( + metadata={ + "description": "Nutanix Prism Central endpoint, without scheme or port." + } + ) + username: str = field(metadata={"description": "Nutanix Prism Central username."}) + password: str = field(metadata={"description": "Nutanix Prism Central password."}) + insecure: Optional[bool] = field( + default=False, + metadata={"description": "Whether to ignore SSL certificate errors."}, + ) + port: Optional[int] = field( + default=9440, + metadata={"description": "Nutanix Prism Central port."}, + ) + + +@define +class PrismCentralColletorConfig: + kind: ClassVar[str] = "prism_central" + credentials: List[PrismCentalCredentials] = field( + factory=list, + metadata={ + "description": "Nutanix Prism Central credentials for the resources to be collected." + "Expected format: [{ 'name': 'my_pc', 'endpoint': 'mypc.example.com', 'insecure': 'true', 'port': '9440', 'username':'my_pc_username', 'password':'my_pc_password'}]." + }, + ) diff --git a/plugins/nutanix/fix_plugin_nutanix/resources.py b/plugins/nutanix/fix_plugin_nutanix/resources.py new file mode 100644 index 0000000000..61e3b76b3b --- /dev/null +++ b/plugins/nutanix/fix_plugin_nutanix/resources.py @@ -0,0 +1,72 @@ +import logging +from typing import ClassVar, Dict, List, Optional, Tuple, Any +from attrs import define +from fixlib.baseresources import BaseAccount, BaseResource, ModelReference +from fixlib.graph import Graph + +log = logging.getLogger("fix." + __name__) + + +@define(eq=False, slots=False) +class PrismCentralAccount(BaseAccount): + """PrismCentral Account""" + + kind: ClassVar[str] = "prismcenteral_account" + kind_display: ClassVar[str] = "Prism Central" + kind_description: ClassVar[str] = ( + "Prism Central is a multi-cluster management solution that enables you to manage multiple Nutanix clusters " + ) + reference_kinds: ClassVar[ModelReference] = { + "successors": { + "default": [ + "prism_element", + ], + "delete": [], + } + } + endpoint: str + username: str + password: str + port: Optional[str] = "9440" + insecure: Optional[bool] = False + + +@define(eq=False, slots=False) +class PrismElement(BaseResource): + """Prism Element""" + + kind: ClassVar[str] = "prism_element" + kind_display: ClassVar[str] = "Prism Element" + kind_description: ClassVar[str] = ( + "A Nutanix Prism Element is like a region in public clouds" + ) + reference_kinds: ClassVar[ModelReference] = { + "successors": { + "default": [ + "virtual_machine", + ], + "delete": [], + } + } + + def delete(self, graph: Graph) -> bool: + """PEs can usually not be deleted so we return NotImplemented""" + return NotImplemented + + +@define(eq=False, slots=False) +class ViraualMachine(BaseResource): + kind: ClassVar[str] = "virtual_machine" + kind_display: ClassVar[str] = "Virtual Machine" + kind_description: ClassVar[str] = "A virtual machine in Nutanix" + reference_kinds: ClassVar[ModelReference] = { + "predecessors": { + "default": ["prism_element"], + "delete": [], + } + } + power_state: str + + def delete(self, graph: Graph) -> bool: + log.info(f"delete virtual machine {self.id}: {self.name}") + return NotImplemented diff --git a/plugins/nutanix/test/test_args.py b/plugins/nutanix/test/test_args.py deleted file mode 100644 index 4e8abba4be..0000000000 --- a/plugins/nutanix/test/test_args.py +++ /dev/null @@ -1,13 +0,0 @@ -# from fixlib.args import get_arg_parser -# from fix_plugin_nutanix import NutanixCollectorPlugin - -# # from fixlib.args import ArgumentParser - - -# def test_args(): -# arg_parser = get_arg_parser() -# NutanixCollectorPlugin.add_args(arg_parser) -# arg_parser.parse_args() - - -# # assert ArgumentParser.args.example_arg is None diff --git a/plugins/nutanix/test/test_collector.py b/plugins/nutanix/test/test_collector.py index 4ed14f63b6..4d87eaa858 100644 --- a/plugins/nutanix/test/test_collector.py +++ b/plugins/nutanix/test/test_collector.py @@ -1,22 +1,48 @@ import logging +import os from typing import Dict, Any, List, cast -from fixlib.baseresources import GraphRoot +from fixlib.baseresources import GraphRoot, Cloud from fixlib.graph import Graph from fixlib.graph import sanitize -from fix_plugin_nutanix import NutanixCollectorPlugin +from fix_plugin_nutanix.collector import PrismCentralCollector +from fix_plugin_nutanix.resources import PrismCentralAccount +import fix_plugin_nutanix log = logging.getLogger("fix." + __name__) + def prepare_graph() -> Graph: - plugin_instance = NutanixCollectorPlugin() + sherlockDevAccount = PrismCentralAccount( + id="sherlock_dev", + name="Sherlock Dev", + endpoint="prismcentral.dev.ntnxsherlock.com", + username=os.getenv("NUTANIX_USER"), + password=os.getenv("NUTANIX_PASSWORD"), + tags={"url": "https://prismcentral.dev.ntnxsherlock.com:9440/"}, + ) + vmmClient = fix_plugin_nutanix.vmm_client(sherlockDevAccount) + clusterClient = fix_plugin_nutanix.cluster_client(sherlockDevAccount) + plugin_instance = PrismCentralCollector( + sherlockDevAccount, vmmClient, clusterClient + ) plugin_instance.collect() + cloud = Cloud(id="nutanix_test") + cloud_graph = Graph(root=cloud) + cloud_graph.merge(plugin_instance.graph) + # create root and add cloud graph. graph = Graph(root=GraphRoot(id="root", tags={})) - graph.merge(plugin_instance.graph) + graph.merge(cloud_graph) sanitize(graph) + for node in graph.nodes: + log.info(f"Node ID: {node.id}" + f"Node name: {node.name}") + for node_from, node_to, edge in graph.edges: + log.info(f"Edge from: {node_from.name} to {node_to.name} with type {edge}") + return graph + def test_collect_nutanix() -> None: - graph = prepare_graph() \ No newline at end of file + graph = prepare_graph() diff --git a/plugins/nutanix/test/test_config.py b/plugins/nutanix/test/test_config.py index f93272b805..5ff6191635 100644 --- a/plugins/nutanix/test/test_config.py +++ b/plugins/nutanix/test/test_config.py @@ -1,11 +1,10 @@ -# from fixlib.config import Config -# from fix_plugin_nutanix import NutanixCollectorPlugin +from numpy import where +from fixlib.config import Config +from fix_plugin_nutanix import NutanixCollectorPlugin -# def test_config(): -# config = Config("dummy", "dummy") -# NutanixCollectorPlugin.add_config(config) -# Config.init_default_config() - - -# # assert Config.example.region is None +def test_config(): + config = Config("dummy", "dummy") + NutanixCollectorPlugin.add_config(config) + Config.init_default_config() + assert Config.prism_central.credentials == [] diff --git a/plugins/nutanix/tox.ini b/plugins/nutanix/tox.ini index 100bd097db..84c6673d29 100644 --- a/plugins/nutanix/tox.ini +++ b/plugins/nutanix/tox.ini @@ -13,6 +13,7 @@ asyncio_mode=auto log_cli = true log_cli_level = INFO + [testenv] usedevelop = true deps = diff --git a/requirements-extra.txt b/requirements-extra.txt index 614beeb6a6..cd81dc29b0 100644 --- a/requirements-extra.txt +++ b/requirements-extra.txt @@ -24,7 +24,7 @@ brotli==1.1.0 cachetools==5.3.3 cattrs==23.2.3 cerberus==1.3.5 -certifi==2024.7.4 +certifi==2020.4.5.1 cffi==1.16.0 charset-normalizer==3.3.2 cheroot==10.0.1 @@ -147,3 +147,6 @@ wrapt==1.16.0 yarl==1.9.4 zc-lockfile==3.0.post1 zipp==3.19.2 +ntnx_vmm_py_client==4.0.3a1 +ntnx_prism_py_client==4.0.3a2 +ntnx_clustermgmt_py_client==4.0.2a2 \ No newline at end of file From 2ba9634bce4cfc7b0b196347ad281814666ecca4 Mon Sep 17 00:00:00 2001 From: Shalin Patel Date: Sun, 11 Aug 2024 09:55:56 -0700 Subject: [PATCH 3/4] build: build docker images --- Makefile | 9 +++++++++ docker-compose.yaml | 4 ++-- 2 files changed, 11 insertions(+), 2 deletions(-) diff --git a/Makefile b/Makefile index bc626d8d2b..fd86bd12c7 100644 --- a/Makefile +++ b/Makefile @@ -25,3 +25,12 @@ setup: clean requirements: python3 tools/requirements.py + +DOCKER_IMAGE := ghcr.io/someengineering +docker-build: + docker build -t $(DOCKER_IMAGE)/fixinventorybase:$(IMAGE_TAG) . -f Dockerfile.fixinventorybase + docker build -t $(DOCKER_IMAGE)/fixcore:$(IMAGE_TAG) . -f Dockerfile.fixcore + docker build -t $(DOCKER_IMAGE)/fixworker:$(IMAGE_TAG) . -f Dockerfile.fixworker + docker build -t $(DOCKER_IMAGE)/fixmetrics:$(IMAGE_TAG) . -f Dockerfile.fixmetrics + docker build -t $(DOCKER_IMAGE)/fixshell:$(IMAGE_TAG) . -f Dockerfile.fixshell + diff --git a/docker-compose.yaml b/docker-compose.yaml index a7d29b6b94..bea0bb9a06 100644 --- a/docker-compose.yaml +++ b/docker-compose.yaml @@ -1,6 +1,6 @@ services: graphdb-upgrade: - image: arangodb:3.10.1 + image: arangodb:3.8.9 container_name: graphdb-upgrade environment: - ARANGO_ROOT_PASSWORD= @@ -11,7 +11,7 @@ services: command: - --database.auto-upgrade graphdb: - image: arangodb:3.10.1 + image: arangodb:3.8.9 depends_on: graphdb-upgrade: condition: service_completed_successfully From cdeccfc4845705ad1db92bff8aa0d9e51212488b Mon Sep 17 00:00:00 2001 From: Shalin Patel Date: Mon, 12 Aug 2024 18:29:05 -0700 Subject: [PATCH 4/4] fix: set prism account graph --- Makefile | 4 +- .../nutanix/fix_plugin_nutanix/__init__.py | 14 ++++--- .../nutanix/fix_plugin_nutanix/collector.py | 42 +++++++------------ plugins/nutanix/fix_plugin_nutanix/config.py | 2 +- .../nutanix/fix_plugin_nutanix/resources.py | 4 +- plugins/nutanix/test/test_collector.py | 8 ++-- plugins/nutanix/test/test_config.py | 2 +- plugins/nutanix/tox.ini | 1 + 8 files changed, 33 insertions(+), 44 deletions(-) diff --git a/Makefile b/Makefile index fd86bd12c7..82a9201370 100644 --- a/Makefile +++ b/Makefile @@ -26,9 +26,9 @@ setup: clean requirements: python3 tools/requirements.py -DOCKER_IMAGE := ghcr.io/someengineering +DOCKER_IMAGE := somecr.io/someengineering docker-build: - docker build -t $(DOCKER_IMAGE)/fixinventorybase:$(IMAGE_TAG) . -f Dockerfile.fixinventorybase + docker build -t ghcr.io/someengineering/fixinventorybase:$(IMAGE_TAG) . -f Dockerfile.fixinventorybase docker build -t $(DOCKER_IMAGE)/fixcore:$(IMAGE_TAG) . -f Dockerfile.fixcore docker build -t $(DOCKER_IMAGE)/fixworker:$(IMAGE_TAG) . -f Dockerfile.fixworker docker build -t $(DOCKER_IMAGE)/fixmetrics:$(IMAGE_TAG) . -f Dockerfile.fixmetrics diff --git a/plugins/nutanix/fix_plugin_nutanix/__init__.py b/plugins/nutanix/fix_plugin_nutanix/__init__.py index 4af331146d..fe804fd50a 100644 --- a/plugins/nutanix/fix_plugin_nutanix/__init__.py +++ b/plugins/nutanix/fix_plugin_nutanix/__init__.py @@ -1,5 +1,4 @@ import fixlib.logger -import os from ntnx_clustermgmt_py_client import Configuration as ClusterConfiguration from ntnx_clustermgmt_py_client import ApiClient as ClusterClient @@ -24,6 +23,8 @@ class NutanixCollectorPlugin(BaseCollectorPlugin): """Nutanix Collector Plugin""" + cloud = "nutanix" + def collect(self) -> None: """This method is being called by fix whenever the collector runs @@ -35,9 +36,7 @@ def collect(self) -> None: """ log.debug("plugin: collecting nutanix resources") - pcConfigs = cast( - List[PrismCentalCredentials], Config.PrismCentralColletorConfig.credentials - ) + pcConfigs = cast(List[PrismCentalCredentials], Config.nutanix.credentials) for pc in pcConfigs: pcAccount = PrismCentralAccount( id=pc.name.replace(" ", "_"), @@ -45,11 +44,14 @@ def collect(self) -> None: endpoint=pc.endpoint, username=pc.username, password=pc.password, + port=pc.port, + insecure=pc.insecure, ) pc_graph = self.collect_pc(pcAccount) - self.graph.add_resource(self.graph.root, pc_graph) + if pc_graph: + self.send_account_graph(pc_graph) - def collect_pc(self, prismCentral: PrismCentralAccount) -> Optional[Graph]: + def collect_pc(self, prismCentral: PrismCentralAccount) -> Graph: log.info(f"Collecting data from Nutanix Prism Central {prismCentral.name}") vmmClient = vmm_client(prismCentral) clusterClient = cluster_client(prismCentral) diff --git a/plugins/nutanix/fix_plugin_nutanix/collector.py b/plugins/nutanix/fix_plugin_nutanix/collector.py index 75092e4b25..bf457cfa3f 100644 --- a/plugins/nutanix/fix_plugin_nutanix/collector.py +++ b/plugins/nutanix/fix_plugin_nutanix/collector.py @@ -1,14 +1,11 @@ import logging -import json import ntnx_clustermgmt_py_client -from ntnx_clustermgmt_py_client import Configuration as ClusterConfiguration from ntnx_clustermgmt_py_client import ApiClient as ClusterClient -from typing import Tuple, Type, List, Dict, Callable, Any, Optional, cast, DefaultDict +from typing import Tuple, List, Dict, Callable import ntnx_vmm_py_client -from ntnx_vmm_py_client import Configuration as VMMConfiguration from ntnx_vmm_py_client import ApiClient as VMMClient from fix_plugin_nutanix.resources import ( PrismCentralAccount, @@ -18,7 +15,7 @@ from fixlib.graph import Graph log = logging.getLogger("fix." + __name__) - + class PrismCentralCollector: """ @@ -52,18 +49,16 @@ def __init__( self.collecter_set = set(self.allCollectors.keys()) self.graph = Graph(root=self.prismCentral) - def collect(self): + def collect(self) -> Graph: """ Runs resource collectors across all PE Resource collectors add their resources to the local `self.graph` graph """ - log.info("Collecting data from Nutanix Prism Central") + log.info(f"Collecting data from Nutanix Prism Central: {self.prismCentral.name}") collectors = set(self.collecter_set) for collectorName, collector in self.mandatoryCollectors: if collectorName in collectors: - log.info( - f"Running collector: {collectorName} in {self.prismCentral.name}" - ) + log.info(f"Running collector: {collectorName} in {self.prismCentral.name}") collector() prismElements = [pe for pe in self.graph.nodes if isinstance(pe, PrismElement)] @@ -72,20 +67,15 @@ def collect(self): if collectorName in collectors: log.info(f"Running collector: {collectorName} in {pe.name}") collector(pe) - return self.graph def collect_prism_element(self) -> None: - """ - Collects data from all Prism Elements - """ log.info("Collecting data from all Prism Elements") clusterApi = ntnx_clustermgmt_py_client.api.ClusterApi(self.clusterClient) clusters = clusterApi.get_clusters() - log.info(f"Found clusters: {clusters.metadata.total_available_results}") + log.info(f"Found PEs: {clusters.metadata.total_available_results}") for cluster in clusters.data: - # log.info(f"Cluster: {cluster}") - # log.info(f"uuid: {cluster['extId']}" f", name: {cluster['name']}") + log.debug(f"Processing PE. uuid: {cluster['extId']}" f", name: {cluster['name']}") pe = PrismElement( id=cluster["extId"], name=cluster["name"], @@ -100,10 +90,7 @@ def collect_images(self) -> None: log.info("Collecting data from all images") def collect_virtual_machines(self, pe: PrismElement) -> None: - """ - Collects data from all virtual machines - """ - log.info("Collecting data from all virtual machines") + log.info(f"Collecting data from all virtual machines in {pe.name}") # Get all virtual machines in the PE vmm_instance = ntnx_vmm_py_client.api.VmApi(self.vmmClient) filter = f"contains(cluster/extId, '{pe.id}')" @@ -114,13 +101,12 @@ def collect_virtual_machines(self, pe: PrismElement) -> None: if response.metadata.total_available_results == 0: return for vm in response.data: - # log.info( - # f"VM: {vm.name}" - # f", uuid: {vm.ext_id}" - # f", power_state: {vm.power_state}" - # f", create_time: {vm.create_time}" - # ) - # log.info(f"VM: {vm}") + log.debug( + f"VM: {vm.name}" + f", uuid: {vm.ext_id}" + f", power_state: {vm.power_state}" + f", create_time: {vm.create_time}" + ) vm = ViraualMachine( id=vm.ext_id, name=vm.name, diff --git a/plugins/nutanix/fix_plugin_nutanix/config.py b/plugins/nutanix/fix_plugin_nutanix/config.py index f6c8bae6f4..0f1a748ce8 100644 --- a/plugins/nutanix/fix_plugin_nutanix/config.py +++ b/plugins/nutanix/fix_plugin_nutanix/config.py @@ -27,7 +27,7 @@ class PrismCentalCredentials: @define class PrismCentralColletorConfig: - kind: ClassVar[str] = "prism_central" + kind: ClassVar[str] = "nutanix" credentials: List[PrismCentalCredentials] = field( factory=list, metadata={ diff --git a/plugins/nutanix/fix_plugin_nutanix/resources.py b/plugins/nutanix/fix_plugin_nutanix/resources.py index 61e3b76b3b..ed69cebaf6 100644 --- a/plugins/nutanix/fix_plugin_nutanix/resources.py +++ b/plugins/nutanix/fix_plugin_nutanix/resources.py @@ -11,7 +11,7 @@ class PrismCentralAccount(BaseAccount): """PrismCentral Account""" - kind: ClassVar[str] = "prismcenteral_account" + kind: ClassVar[str] = "prismcentral_account" kind_display: ClassVar[str] = "Prism Central" kind_description: ClassVar[str] = ( "Prism Central is a multi-cluster management solution that enables you to manage multiple Nutanix clusters " @@ -27,7 +27,7 @@ class PrismCentralAccount(BaseAccount): endpoint: str username: str password: str - port: Optional[str] = "9440" + port: Optional[int] = 9440 insecure: Optional[bool] = False diff --git a/plugins/nutanix/test/test_collector.py b/plugins/nutanix/test/test_collector.py index 4d87eaa858..9ccd762c04 100644 --- a/plugins/nutanix/test/test_collector.py +++ b/plugins/nutanix/test/test_collector.py @@ -18,20 +18,20 @@ def prepare_graph() -> Graph: sherlockDevAccount = PrismCentralAccount( id="sherlock_dev", name="Sherlock Dev", - endpoint="prismcentral.dev.ntnxsherlock.com", + endpoint=os.getenv("NUTANIX_ENDPOINT"), username=os.getenv("NUTANIX_USER"), password=os.getenv("NUTANIX_PASSWORD"), - tags={"url": "https://prismcentral.dev.ntnxsherlock.com:9440/"}, + tags={"url": os.getenv("NUTANIX_ENDPOINT")}, ) vmmClient = fix_plugin_nutanix.vmm_client(sherlockDevAccount) clusterClient = fix_plugin_nutanix.cluster_client(sherlockDevAccount) plugin_instance = PrismCentralCollector( sherlockDevAccount, vmmClient, clusterClient ) - plugin_instance.collect() + pcGraph = plugin_instance.collect() cloud = Cloud(id="nutanix_test") cloud_graph = Graph(root=cloud) - cloud_graph.merge(plugin_instance.graph) + cloud_graph.merge(pcGraph) # create root and add cloud graph. graph = Graph(root=GraphRoot(id="root", tags={})) graph.merge(cloud_graph) diff --git a/plugins/nutanix/test/test_config.py b/plugins/nutanix/test/test_config.py index 5ff6191635..b215d05b86 100644 --- a/plugins/nutanix/test/test_config.py +++ b/plugins/nutanix/test/test_config.py @@ -7,4 +7,4 @@ def test_config(): config = Config("dummy", "dummy") NutanixCollectorPlugin.add_config(config) Config.init_default_config() - assert Config.prism_central.credentials == [] + assert Config.nutanix.credentials == [] diff --git a/plugins/nutanix/tox.ini b/plugins/nutanix/tox.ini index 84c6673d29..9ad8c01657 100644 --- a/plugins/nutanix/tox.ini +++ b/plugins/nutanix/tox.ini @@ -26,6 +26,7 @@ setenv = passenv = NUTANIX_USER NUTANIX_PASSWORD + NUTANIX_ENDPOINT [testenv:syntax] commands = flake8