diff --git a/src/pySupersetCli/__main__.py b/src/pySupersetCli/__main__.py index d01e1c1..8f61bd6 100644 --- a/src/pySupersetCli/__main__.py +++ b/src/pySupersetCli/__main__.py @@ -39,6 +39,7 @@ from pySupersetCli.version import __version__, __author__, __email__, __repository__, __license__ from pySupersetCli.ret import Ret +from pySupersetCli.superset import Superset ################################################################################ @@ -151,7 +152,18 @@ def main() -> Ret: for arg in vars(args): LOG.info("* %s = %s", arg, vars(args)[arg]) - if Ret.OK == ret_status: + # Create Superset client. + try: + # pylint: disable=unused-variable + client = Superset(args.server, + args.user, + args.password, + Superset.Provider.DB) + + except (RuntimeError, NotImplementedError) as e: + LOG.error("Failed to create Superset client: %s", e) + ret_status = Ret.ERROR_LOGIN + else: handler = None # Find the command handler. @@ -162,7 +174,7 @@ def main() -> Ret: # Execute the command. if handler is not None: - ret_status = Ret.OK + ret_status = handler(args, client) else: LOG.error("Command '%s' not found!", args.cmd) ret_status = Ret.ERROR_INVALID_ARGUMENTS diff --git a/src/pySupersetCli/superset.py b/src/pySupersetCli/superset.py new file mode 100644 index 0000000..8283094 --- /dev/null +++ b/src/pySupersetCli/superset.py @@ -0,0 +1,202 @@ +"""Server wrapper for requests to the Superset API.""" + +# BSD 3-Clause License +# +# Copyright (c) 2024, NewTec GmbH +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are met: +# +# 1. Redistributions of source code must retain the above copyright notice, this +# list of conditions and the following disclaimer. +# +# 2. Redistributions in binary form must reproduce the above copyright notice, +# this list of conditions and the following disclaimer in the documentation +# and/or other materials provided with the distribution. +# +# 3. Neither the name of the copyright holder nor the names of its +# contributors may be used to endorse or promote products derived from +# this software without specific prior written permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICU5LAR PURPOSE ARE +# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE +# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL +# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR +# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, +# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +################################################################################ +# Imports +################################################################################ + +from dataclasses import dataclass +import logging +import requests +import urllib3 + + +################################################################################ +# Variables +################################################################################ + +LOG: logging.Logger = logging.getLogger(__name__) + +################################################################################ +# Classes +################################################################################ + + +class Superset: + """ + Wrapper of the requests module for the Superset API. + Handles the authentication and the API calls. + Implements parts of the Superset API: https://superset.apache.org/docs/api/ + """ + + @dataclass + class Provider: + """ + Enum for the supported authentication providers. + """ + DB = "db" + LDAP = "ldap" + + # pylint: disable=too-many-arguments + def __init__(self, + server_url: str, + username: str, + password: str, + provider: Provider, + verify_ssl: bool = True) -> None: + """ + Initializes the Superset object and logs in the user. + + Args: + server_url (str): The URL of the Superset server. + username (str): The username of the user. + password (str): The password of the user. + provider (Provider): The authentication provider. + verify_ssl (bool): Verify the SSL certificate of the server. + """ + self._server_url: str = f"{server_url}/api/v1" + self._access_token: str = "" + self._csrf_token: str = "" + self._timeout: int = 60 + self._verify_ssl: bool = verify_ssl + + if not self._verify_ssl: + # Disable SSL warnings if SSL verification is disabled + urllib3.disable_warnings(urllib3.exceptions.InsecureRequestWarning) + + # Login the user and retrieve the access token and the CSRF token + self._login(username, password, provider) + + def _login(self, username: str, password: str, provider: Provider) -> None: + """ + Logs in the user and retrieves the access token and the refresh token. + + Args: + username (str): The username of the user. + password (str): The password of the user. + provider (Provider): The authentication provider. + + Returns: + None + """ + login_endpoint: str = "/security/login" + crsf_token_endpoint: str = "/security/csrf_token/" + + login_body: dict = { + "password": password, + "provider": provider, + "refresh": True, + "username": username + } + + # Send the login request + ret_code, response = self.request("POST", + login_endpoint, + json=login_body) + + if ret_code != 200: + LOG.fatal("Login failed: %s", response.get("message")) + raise RuntimeError("Login failed") + + self._access_token = response.get("access_token", "") + + # Get the CSRF token + ret_code, response = self.request("GET", crsf_token_endpoint) + + if ret_code != 200: + LOG.fatal("Get CSRF token failed: %s", response.get("message")) + raise RuntimeError("Get CSRF token failed") + + self._csrf_token = response.get("result", "") + + if self._access_token == "" or self._csrf_token == "": + LOG.fatal("Tokens failed: Access token or CSRF token not received.") + raise RuntimeError("Tokens failed") + + def request(self, + method: str, + endpoint: str, + **request_kwargs) -> tuple[int, dict]: + """ + Sends a request to the Superset API. + + Args: + method (str): The HTTP method of the request. + endpoint (str): The endpoint of the request after '/api/v1'. + data (dict): The data of the request. + request_kwargs (dict): Additional keyword arguments for the request. + Can be any accepted by the Requests module. + + Returns: + dict: The response of the request. + """ + + url: str = f"{self._server_url}{endpoint}" + headers: dict = {} + + # If already logged in, add the access token to the headers + if self._access_token != "": + headers = { + 'Authorization': f'Bearer {self._access_token}', + 'referer': self._server_url, + 'X-CSRFToken': self._csrf_token + } + + # Send the request + response: requests.Response = requests.request( + method=method, + url=url, + headers=headers, + timeout=self._timeout, + verify=self._verify_ssl, + ** request_kwargs) + + reponse_data: dict = response.json() + + # Check if the token has expired + if (reponse_data.get('message') == 'Token has expired') and \ + (response.status_code == 401): + LOG.error("Refreshing token is not implemented. Aborting.") + raise NotImplementedError("Token refresh is not implemented.") + + LOG.info("Request: %s %s", method, url) + LOG.info("Response Code: %s", response.status_code) + + return (response.status_code, reponse_data) + + +################################################################################ +# Functions +################################################################################ + +################################################################################ +# Main +################################################################################