From 3c2c15071ebd8432608b0c8f36b37ca08282a1c2 Mon Sep 17 00:00:00 2001 From: Shantanoo 'Shan' Desai Date: Sun, 14 Apr 2024 13:21:49 +0200 Subject: [PATCH] fix: add custom certs verification logic This fix adds the `validate_certs`, `ca_path` options to the lookup plugin. Both parameters comply with the `get_url` functionality of ansible-core and provides additional utility to perform lookup of dashboards from Grafana instances that are configured with Self-Signed Certificates. `validate_certs` option value defaults to `true` - following the pattern of `url` lookup plugin from the Core. `ca_path` option value is set explicitly when using the plugin else defaults to `None`. closes #346 Signed-off-by: Shantanoo 'Shan' Desai --- ...ds-add-custom-certs-verification-logic.yml | 2 + plugins/lookup/grafana_dashboard.py | 51 +++++++++++++++++-- 2 files changed, 49 insertions(+), 4 deletions(-) create mode 100644 changelogs/fragments/346-lookup-dashboards-add-custom-certs-verification-logic.yml diff --git a/changelogs/fragments/346-lookup-dashboards-add-custom-certs-verification-logic.yml b/changelogs/fragments/346-lookup-dashboards-add-custom-certs-verification-logic.yml new file mode 100644 index 00000000..182822d9 --- /dev/null +++ b/changelogs/fragments/346-lookup-dashboards-add-custom-certs-verification-logic.yml @@ -0,0 +1,2 @@ +minor_changes: + - lookup - grafana_dashboards - add `validate_certs` and `ca_path` options to plugin for custom certs validation diff --git a/plugins/lookup/grafana_dashboard.py b/plugins/lookup/grafana_dashboard.py index c7fa3157..a0338fed 100644 --- a/plugins/lookup/grafana_dashboard.py +++ b/plugins/lookup/grafana_dashboard.py @@ -41,6 +41,13 @@ description: optional filter for dashboard search. env: - name: GRAFANA_DASHBOARD_SEARCH + validate_certs: + description: flag to control SSL certificate validation + type: boolean + default: True + ca_path: + description: string of the file system path to CA cert bundle to use for validation + type: string """ EXAMPLES = """ @@ -51,13 +58,25 @@ - name: get all grafana dashboards set_fact: grafana_dashboards: "{{ lookup('grafana_dashboard', 'grafana_url=http://grafana.company.com grafana_api_key=' ~ grafana_api_key) }}" + +- name: get project foo grafana dashboards (validate SSL certificates of the instance with custom CA Certificate Bundle) + set_fact: + grafana_dashboards: | + {{ + lookup( + 'grafana_dashboard', + 'grafana_url=https://grafana.company.com grafana_user=admin grafana_password=admin search=foo', + validate_cert=true, + ca_path='/path/to/chain.crt' + ) + }} """ import json import os from ansible.errors import AnsibleError from ansible.plugins.lookup import LookupBase -from ansible.module_utils.urls import basic_auth_header, open_url +from ansible.module_utils.urls import basic_auth_header, open_url, SSLValidationError from ansible.module_utils._text import to_native from ansible.module_utils.six.moves.urllib.error import HTTPError from ansible.utils.display import Display @@ -96,13 +115,15 @@ class GrafanaAPIException(Exception): class GrafanaAPI: - def __init__(self, **kwargs): + def __init__(self, validate_certs, ca_path, **kwargs): self.grafana_url = kwargs.get("grafana_url", ANSIBLE_GRAFANA_URL) self.grafana_api_key = kwargs.get("grafana_api_key", ANSIBLE_GRAFANA_API_KEY) self.grafana_user = kwargs.get("grafana_user", ANSIBLE_GRAFANA_USER) self.grafana_password = kwargs.get("grafana_password", ANSIBLE_GRAFANA_PASSWORD) self.grafana_org_id = kwargs.get("grafana_org_id", ANSIBLE_GRAFANA_ORG_ID) self.search = kwargs.get("search", ANSIBLE_GRAFANA_DASHBOARD_SEARCH) + self.validate_certs = validate_certs + self.ca_path = ca_path def grafana_switch_organisation(self, headers): try: @@ -110,12 +131,19 @@ def grafana_switch_organisation(self, headers): "%s/api/user/using/%s" % (self.grafana_url, self.grafana_org_id), headers=headers, method="POST", + validate_certs=self.validate_certs, + ca_path=self.ca_path, ) except HTTPError as e: raise GrafanaAPIException( "Unable to switch to organization %s : %s" % (self.grafana_org_id, to_native(e)) ) + except SSLValidationError as e: + raise GrafanaAPIException( + "Unable to validate server's certificate with %s: %s" + % (self.ca_path, to_native(e)) + ) if r.getcode() != 200: raise GrafanaAPIException( "Unable to switch to organization %s : %s" @@ -153,13 +181,23 @@ def grafana_list_dashboards(self): "%s/api/search?query=%s" % (self.grafana_url, self.search), headers=headers, method="GET", + validate_certs=self.validate_certs, + ca_path=self.ca_path, ) else: r = open_url( - "%s/api/search/" % self.grafana_url, headers=headers, method="GET" + "%s/api/search/" % self.grafana_url, + headers=headers, + method="GET", + validate_certs=self.validate_certs, + ca_path=self.ca_path, ) except HTTPError as e: raise GrafanaAPIException("Unable to search dashboards : %s" % to_native(e)) + except SSLValidationError as e: + raise GrafanaAPIException( + "Unable to validate server's certificate with %s: %s" % (self.ca_path, to_native(e)) + ) if r.getcode() == 200: try: dashboard_list = json.loads(r.read()) @@ -175,6 +213,7 @@ def grafana_list_dashboards(self): class LookupModule(LookupBase): def run(self, terms, variables=None, **kwargs): + self.set_options(var_options=variables, direct=kwargs) grafana_args = terms[0].split(" ") grafana_dict = {} ret = [] @@ -189,7 +228,11 @@ def run(self, terms, variables=None, **kwargs): ) grafana_dict[key] = value - grafana = GrafanaAPI(**grafana_dict) + grafana = GrafanaAPI( + **grafana_dict, + validate_certs=self.get_option("validate_certs"), + ca_path=self.get_option("ca_path"), + ) ret = grafana.grafana_list_dashboards()