2020-03-09 13:36:01 +00:00
|
|
|
# Copyright: (c) 2019, Hetzner Cloud GmbH <info@hetzner-cloud.de>
|
|
|
|
|
|
|
|
# Simplified BSD License (see licenses/simplified_bsd.txt or https://opensource.org/licenses/BSD-2-Clause)
|
|
|
|
|
|
|
|
|
2023-07-20 12:34:48 +00:00
|
|
|
import traceback
|
2023-10-18 08:28:42 +00:00
|
|
|
from typing import Any, Dict, Optional, Union
|
2023-07-20 12:34:48 +00:00
|
|
|
|
2023-09-26 07:41:01 +00:00
|
|
|
from ansible.module_utils.basic import (
|
|
|
|
AnsibleModule as AnsibleModuleBase,
|
|
|
|
env_fallback,
|
|
|
|
missing_required_lib,
|
|
|
|
)
|
2023-07-20 12:34:48 +00:00
|
|
|
from ansible.module_utils.common.text.converters import to_native
|
2023-07-31 08:12:55 +00:00
|
|
|
|
2023-09-26 07:41:01 +00:00
|
|
|
from ..module_utils.vendor.hcloud import APIException, Client, HCloudException
|
|
|
|
from ..module_utils.vendor.hcloud.actions import ActionException
|
2023-08-09 09:34:12 +00:00
|
|
|
from .version import version
|
2023-07-11 09:15:08 +00:00
|
|
|
|
|
|
|
HAS_REQUESTS = True
|
|
|
|
HAS_DATEUTIL = True
|
2020-03-09 13:36:01 +00:00
|
|
|
|
|
|
|
try:
|
2023-07-11 09:15:08 +00:00
|
|
|
import requests # pylint: disable=unused-import
|
|
|
|
except ImportError:
|
|
|
|
HAS_REQUESTS = False
|
2020-03-09 13:36:01 +00:00
|
|
|
|
2023-07-11 09:15:08 +00:00
|
|
|
try:
|
|
|
|
import dateutil # pylint: disable=unused-import
|
2020-03-09 13:36:01 +00:00
|
|
|
except ImportError:
|
2023-07-11 09:15:08 +00:00
|
|
|
HAS_DATEUTIL = False
|
2020-03-09 13:36:01 +00:00
|
|
|
|
|
|
|
|
2023-09-26 07:41:01 +00:00
|
|
|
# Provide typing definitions to the AnsibleModule class
|
|
|
|
class AnsibleModule(AnsibleModuleBase):
|
|
|
|
params: dict
|
|
|
|
|
|
|
|
|
2023-08-04 07:24:14 +00:00
|
|
|
class AnsibleHCloud:
|
2023-09-26 07:41:01 +00:00
|
|
|
represent: str
|
|
|
|
|
|
|
|
module: AnsibleModule
|
|
|
|
|
|
|
|
def __init__(self, module: AnsibleModule):
|
|
|
|
if not self.represent:
|
|
|
|
raise NotImplementedError(f"represent property is not defined for {self.__class__.__name__}")
|
|
|
|
|
2020-03-09 13:36:01 +00:00
|
|
|
self.module = module
|
|
|
|
self.result = {"changed": False, self.represent: None}
|
2023-07-11 09:15:08 +00:00
|
|
|
if not HAS_REQUESTS:
|
|
|
|
module.fail_json(msg=missing_required_lib("requests"))
|
|
|
|
if not HAS_DATEUTIL:
|
|
|
|
module.fail_json(msg=missing_required_lib("python-dateutil"))
|
2020-03-09 13:36:01 +00:00
|
|
|
self._build_client()
|
|
|
|
|
2023-09-26 07:41:01 +00:00
|
|
|
def fail_json_hcloud(
|
|
|
|
self,
|
|
|
|
exception: HCloudException,
|
|
|
|
msg: Optional[str] = None,
|
|
|
|
params: Any = None,
|
|
|
|
**kwargs,
|
|
|
|
) -> None:
|
2023-07-20 12:34:48 +00:00
|
|
|
last_traceback = traceback.format_exc()
|
|
|
|
|
|
|
|
failure = {}
|
|
|
|
|
|
|
|
if params is not None:
|
|
|
|
failure["params"] = params
|
|
|
|
|
2023-09-26 07:41:01 +00:00
|
|
|
if isinstance(exception, APIException):
|
2023-07-20 12:34:48 +00:00
|
|
|
failure["message"] = exception.message
|
|
|
|
failure["code"] = exception.code
|
|
|
|
failure["details"] = exception.details
|
|
|
|
|
2023-09-26 07:41:01 +00:00
|
|
|
elif isinstance(exception, ActionException):
|
2023-07-20 12:34:48 +00:00
|
|
|
failure["action"] = {k: getattr(exception.action, k) for k in exception.action.__slots__}
|
|
|
|
|
|
|
|
exception_message = to_native(exception)
|
|
|
|
if msg is not None:
|
|
|
|
msg = f"{exception_message}: {msg}"
|
|
|
|
else:
|
|
|
|
msg = exception_message
|
|
|
|
|
|
|
|
self.module.fail_json(msg=msg, exception=last_traceback, failure=failure, **kwargs)
|
|
|
|
|
2023-09-26 07:41:01 +00:00
|
|
|
def _build_client(self) -> None:
|
|
|
|
self.client = Client(
|
2020-03-09 13:36:01 +00:00
|
|
|
token=self.module.params["api_token"],
|
|
|
|
api_endpoint=self.module.params["endpoint"],
|
|
|
|
application_name="ansible-module",
|
2023-08-09 09:34:12 +00:00
|
|
|
application_version=version,
|
2020-03-09 13:36:01 +00:00
|
|
|
)
|
|
|
|
|
2023-10-18 08:28:42 +00:00
|
|
|
def _client_get_by_name_or_id(self, resource: str, param: Union[str, int]):
|
|
|
|
"""
|
|
|
|
Get a resource by name, and if not found by its ID.
|
|
|
|
|
|
|
|
:param resource: Name of the resource client that implements both `get_by_name` and `get_by_id` methods
|
|
|
|
:param param: Name or ID of the resource to query
|
|
|
|
"""
|
|
|
|
resource_client = getattr(self.client, resource)
|
|
|
|
|
|
|
|
result = resource_client.get_by_name(param)
|
|
|
|
if result is not None:
|
|
|
|
return result
|
|
|
|
|
|
|
|
# If the param is not a valid ID, prevent an unnecessary call to the API.
|
|
|
|
try:
|
|
|
|
int(param)
|
|
|
|
except ValueError:
|
|
|
|
self.module.fail_json(msg=f"resource ({resource.rstrip('s')}) does not exist: {param}")
|
|
|
|
|
|
|
|
return resource_client.get_by_id(param)
|
|
|
|
|
2023-09-26 07:41:01 +00:00
|
|
|
def _mark_as_changed(self) -> None:
|
2020-03-09 13:36:01 +00:00
|
|
|
self.result["changed"] = True
|
|
|
|
|
2023-08-02 10:05:00 +00:00
|
|
|
@classmethod
|
|
|
|
def base_module_arguments(cls):
|
2020-03-09 13:36:01 +00:00
|
|
|
return {
|
|
|
|
"api_token": {
|
|
|
|
"type": "str",
|
|
|
|
"required": True,
|
|
|
|
"fallback": (env_fallback, ["HCLOUD_TOKEN"]),
|
|
|
|
"no_log": True,
|
|
|
|
},
|
2023-09-26 07:41:01 +00:00
|
|
|
"endpoint": {
|
|
|
|
"type": "str",
|
|
|
|
"default": "https://api.hetzner.cloud/v1",
|
|
|
|
},
|
2020-03-09 13:36:01 +00:00
|
|
|
}
|
|
|
|
|
2023-09-26 07:41:01 +00:00
|
|
|
def _prepare_result(self) -> Dict[str, Any]:
|
|
|
|
"""Prepare the result for every module"""
|
2020-03-09 13:36:01 +00:00
|
|
|
return {}
|
|
|
|
|
2023-09-26 07:41:01 +00:00
|
|
|
def get_result(self) -> Dict[str, Any]:
|
2020-03-09 13:36:01 +00:00
|
|
|
if getattr(self, self.represent) is not None:
|
|
|
|
self.result[self.represent] = self._prepare_result()
|
|
|
|
return self.result
|