Commits vergleichen
28 Commits
Autor | SHA1 | Datum | |
---|---|---|---|
3bb2a93dab | |||
749ae48441 | |||
d82a902043 | |||
328e58c439 | |||
9045e51c23 | |||
74ded41d30 | |||
59400d158f | |||
144ffc72a5 | |||
c8d7d4e286 | |||
9d18ea533e | |||
f351bb6fb6 | |||
c5e6129682 | |||
1ea397da14 | |||
84eb37ffa7 | |||
85880d2867 | |||
380bc582e3 | |||
8190ee94d3 | |||
2f36e69a73 | |||
1da65334e2 | |||
47c8d3319e | |||
cd1d5aa84e | |||
0ba608406e | |||
d03b45d678 | |||
e583ad0f96 | |||
8dbb67faf8 | |||
e089dbc82a | |||
768713aadc | |||
5b9dd8df0b |
37 geänderte Dateien mit 1064 neuen und 539 gelöschten Zeilen
1
.gitignore
gevendort
1
.gitignore
gevendort
|
@ -164,3 +164,4 @@ cython_debug/
|
|||
# ---> Ansible
|
||||
*.retry
|
||||
|
||||
src/ansible_module/module_utils
|
||||
|
|
66
CHANGELOG.rst
Normale Datei
66
CHANGELOG.rst
Normale Datei
|
@ -0,0 +1,66 @@
|
|||
============================
|
||||
Sebastian.Base Release Notes
|
||||
============================
|
||||
|
||||
.. contents:: Topics
|
||||
|
||||
v0.4.4
|
||||
======
|
||||
|
||||
Minor Changes
|
||||
-------------
|
||||
|
||||
- removed the empty options dict
|
||||
|
||||
v0.4.3
|
||||
======
|
||||
|
||||
v0.4.2
|
||||
======
|
||||
|
||||
Minor Changes
|
||||
-------------
|
||||
|
||||
- removed the empty options dict
|
||||
|
||||
v0.4.1
|
||||
======
|
||||
|
||||
Minor Changes
|
||||
-------------
|
||||
|
||||
- added an default Display to the module
|
||||
- fixed the docification of dictionaries
|
||||
|
||||
v0.4.0
|
||||
======
|
||||
|
||||
Release Summary
|
||||
---------------
|
||||
|
||||
to prevent empty sections, the install and header methods return None if the method would just the scetion
|
||||
header
|
||||
|
||||
v0.3.1
|
||||
======
|
||||
|
||||
Release Summary
|
||||
---------------
|
||||
|
||||
removed forgotten print calls
|
||||
|
||||
v0.3.0
|
||||
======
|
||||
|
||||
Release Summary
|
||||
---------------
|
||||
|
||||
rewrote the Types helper
|
||||
|
||||
v0.2.0
|
||||
======
|
||||
|
||||
Release Summary
|
||||
---------------
|
||||
|
||||
change the module to an ansible module
|
26
Makefile
Normale Datei
26
Makefile
Normale Datei
|
@ -0,0 +1,26 @@
|
|||
VERSION := $(shell hatch version)
|
||||
|
||||
format:
|
||||
black .
|
||||
isort .
|
||||
|
||||
version:
|
||||
@yq --yaml-output-grammar-version 1.2 -i -y -s --arg 'version' "${VERSION}" '.[0] | (.version = $$version)' galaxy.yml
|
||||
|
||||
changelog: version
|
||||
antsibull-changelog generate
|
||||
|
||||
docs: format
|
||||
update-doc
|
||||
|
||||
clean-dist:
|
||||
rm -rf dist
|
||||
|
||||
hatch-release: clean-dist version
|
||||
hatch build
|
||||
|
||||
galaxy-release: clean-dist changelog version
|
||||
ansible-galaxy collection build --output-path dist/galaxy
|
||||
|
||||
upload: galaxy-release hatch-release
|
||||
./upload.sh
|
19
README.md
19
README.md
|
@ -1,18 +1,3 @@
|
|||
# Ansible module
|
||||
# Ansible Collection - sebastian.base
|
||||
|
||||
-----
|
||||
|
||||
**Table of Contents**
|
||||
|
||||
- [Installation](#installation)
|
||||
- [License](#license)
|
||||
|
||||
## Installation
|
||||
|
||||
```console
|
||||
pip install ansible-module
|
||||
```
|
||||
|
||||
## License
|
||||
|
||||
`ansible-module` is distributed under the terms of the [MIT](https://spdx.org/licenses/MIT.html) license.
|
||||
Documentation for the collection.
|
||||
|
|
19
changelogs/.plugin-cache.yaml
Normale Datei
19
changelogs/.plugin-cache.yaml
Normale Datei
|
@ -0,0 +1,19 @@
|
|||
objects:
|
||||
role: {}
|
||||
plugins:
|
||||
become: {}
|
||||
cache: {}
|
||||
callback: {}
|
||||
cliconf: {}
|
||||
connection: {}
|
||||
filter: {}
|
||||
httpapi: {}
|
||||
inventory: {}
|
||||
lookup: {}
|
||||
module: {}
|
||||
netconf: {}
|
||||
shell: {}
|
||||
strategy: {}
|
||||
test: {}
|
||||
vars: {}
|
||||
version: 0.4.2
|
59
changelogs/changelog.yaml
Normale Datei
59
changelogs/changelog.yaml
Normale Datei
|
@ -0,0 +1,59 @@
|
|||
ancestor: null
|
||||
releases:
|
||||
0.2.0:
|
||||
changes:
|
||||
release_summary: change the module to an ansible module
|
||||
fragments:
|
||||
- base_release.yml
|
||||
release_date: "2024-02-11"
|
||||
0.3.0:
|
||||
changes:
|
||||
release_summary: rewrote the Types helper
|
||||
fragments:
|
||||
- types.yml
|
||||
release_date: "2024-02-24"
|
||||
0.3.1:
|
||||
changes:
|
||||
release_summary: removed forgotten print calls
|
||||
fragments:
|
||||
- print_calls.yml
|
||||
release_date: "2024-02-24"
|
||||
0.4.0:
|
||||
changes:
|
||||
release_summary: "to prevent empty sections, the install and header methods return None if the method would just
|
||||
the scetion
|
||||
|
||||
header
|
||||
|
||||
"
|
||||
fragments:
|
||||
- sectioning.yml
|
||||
release_date: "2024-03-08"
|
||||
0.4.1:
|
||||
changes:
|
||||
minor_changes:
|
||||
- added an default Display to the module
|
||||
- fixed the docification of dictionaries
|
||||
fragments:
|
||||
- display.yml
|
||||
- options_fix.yml
|
||||
release_date: "2024-03-13"
|
||||
0.4.2:
|
||||
changes:
|
||||
minor_changes:
|
||||
- removed the empty options dict
|
||||
fragments:
|
||||
- empty_options.yml
|
||||
release_date: "2024-03-13"
|
||||
0.4.3:
|
||||
changes:
|
||||
fragments:
|
||||
- 0.4.3.yml
|
||||
release_date: "2024-03-13"
|
||||
0.4.4:
|
||||
changes:
|
||||
minor_changes:
|
||||
- removed the empty options dict
|
||||
fragments:
|
||||
- 0.4.4.yml
|
||||
release_date: "2025-03-16"
|
32
changelogs/config.yaml
Normale Datei
32
changelogs/config.yaml
Normale Datei
|
@ -0,0 +1,32 @@
|
|||
changelog_filename_template: ../CHANGELOG.rst
|
||||
changelog_filename_version_depth: 0
|
||||
changes_file: changelog.yaml
|
||||
changes_format: combined
|
||||
ignore_other_fragment_extensions: true
|
||||
keep_fragments: true
|
||||
mention_ancestor: true
|
||||
new_plugins_after_name: removed_features
|
||||
notesdir: fragments
|
||||
prelude_section_name: release_summary
|
||||
prelude_section_title: Release Summary
|
||||
sanitize_changelog: true
|
||||
sections:
|
||||
- - major_changes
|
||||
- Major Changes
|
||||
- - minor_changes
|
||||
- Minor Changes
|
||||
- - breaking_changes
|
||||
- Breaking Changes / Porting Guide
|
||||
- - deprecated_features
|
||||
- Deprecated Features
|
||||
- - removed_features
|
||||
- Removed Features (previously deprecated)
|
||||
- - security_fixes
|
||||
- Security Fixes
|
||||
- - bugfixes
|
||||
- Bugfixes
|
||||
- - known_issues
|
||||
- Known Issues
|
||||
title: Sebastian.Base
|
||||
trivial_section_name: trivial
|
||||
use_fqcn: true
|
6
changelogs/fragments/0.4.3.yml
Normale Datei
6
changelogs/fragments/0.4.3.yml
Normale Datei
|
@ -0,0 +1,6 @@
|
|||
---
|
||||
minor_changes:
|
||||
- Added an type hint to the update_doc function
|
||||
- Added Type hint to the dictionary argument of the diff method
|
||||
- added an wrapper for fail_json and exit_json
|
||||
- improved the version update, the version is now only in one place the source of truth
|
3
changelogs/fragments/0.4.4.yml
Normale Datei
3
changelogs/fragments/0.4.4.yml
Normale Datei
|
@ -0,0 +1,3 @@
|
|||
---
|
||||
major_changes:
|
||||
- the modules can now accept lists for help.
|
2
changelogs/fragments/0.4.5.yml
Normale Datei
2
changelogs/fragments/0.4.5.yml
Normale Datei
|
@ -0,0 +1,2 @@
|
|||
minor_changes:
|
||||
- Added ansible as an dependency for the python module
|
2
changelogs/fragments/base_release.yml
Normale Datei
2
changelogs/fragments/base_release.yml
Normale Datei
|
@ -0,0 +1,2 @@
|
|||
---
|
||||
release_summary: change the module to an ansible module
|
3
changelogs/fragments/dev.yml
Normale Datei
3
changelogs/fragments/dev.yml
Normale Datei
|
@ -0,0 +1,3 @@
|
|||
---
|
||||
minor_changes:
|
||||
- modspec now supports supports_check_mode and add_file_common_args
|
3
changelogs/fragments/display.yml
Normale Datei
3
changelogs/fragments/display.yml
Normale Datei
|
@ -0,0 +1,3 @@
|
|||
---
|
||||
minor_changes:
|
||||
- added an default Display to the module
|
3
changelogs/fragments/empty_options.yml
Normale Datei
3
changelogs/fragments/empty_options.yml
Normale Datei
|
@ -0,0 +1,3 @@
|
|||
---
|
||||
minor_changes:
|
||||
- removed the empty options dict
|
3
changelogs/fragments/options_fix.yml
Normale Datei
3
changelogs/fragments/options_fix.yml
Normale Datei
|
@ -0,0 +1,3 @@
|
|||
---
|
||||
minor_changes:
|
||||
- fixed the docification of dictionaries
|
2
changelogs/fragments/print_calls.yml
Normale Datei
2
changelogs/fragments/print_calls.yml
Normale Datei
|
@ -0,0 +1,2 @@
|
|||
---
|
||||
release_summary: removed forgotten print calls
|
4
changelogs/fragments/sectioning.yml
Normale Datei
4
changelogs/fragments/sectioning.yml
Normale Datei
|
@ -0,0 +1,4 @@
|
|||
---
|
||||
release_summary: |
|
||||
to prevent empty sections, the install and header methods return None if the method would just the scetion
|
||||
header
|
2
changelogs/fragments/types.yml
Normale Datei
2
changelogs/fragments/types.yml
Normale Datei
|
@ -0,0 +1,2 @@
|
|||
---
|
||||
release_summary: rewrote the Types helper
|
28
galaxy.yml
Normale Datei
28
galaxy.yml
Normale Datei
|
@ -0,0 +1,28 @@
|
|||
namespace: sebastian
|
||||
name: base
|
||||
version: 0.5.0
|
||||
readme: README.md
|
||||
authors:
|
||||
- Sebastian Tobie
|
||||
description: The base of my ansible collections. It provides the nessesary tools for
|
||||
my modules
|
||||
license_file: LICENSE.txt
|
||||
tags:
|
||||
- linux
|
||||
- systemd
|
||||
dependencies: {}
|
||||
repository: https://gitea.sebastian-tobie.de/ansible/ansible-module.git
|
||||
homepage: https://gitea.sebastian-tobie.de/ansible/ansible-module
|
||||
issues: https://gitea.sebastian-tobie.de/ansible/ansible-module/issues
|
||||
build_ignore:
|
||||
- '*.gz'
|
||||
- .*
|
||||
- Makefile
|
||||
- pyproject.toml
|
||||
- upload.sh
|
||||
- htmlcov
|
||||
- changelogs
|
||||
- docs
|
||||
- src
|
||||
- coverage.*
|
||||
- dist
|
4
meta/runtime.yml
Normale Datei
4
meta/runtime.yml
Normale Datei
|
@ -0,0 +1,4 @@
|
|||
---
|
||||
# Collections must specify a minimum required ansible version to upload
|
||||
# to galaxy
|
||||
requires_ansible: '>=2.9.10'
|
31
plugins/README.md
Normale Datei
31
plugins/README.md
Normale Datei
|
@ -0,0 +1,31 @@
|
|||
# Collections Plugins Directory
|
||||
|
||||
This directory can be used to ship various plugins inside an Ansible collection. Each plugin is placed in a folder that
|
||||
is named after the type of plugin it is in. It can also include the `module_utils` and `modules` directory that
|
||||
would contain module utils and modules respectively.
|
||||
|
||||
Here is an example directory of the majority of plugins currently supported by Ansible:
|
||||
|
||||
```
|
||||
└── plugins
|
||||
├── action
|
||||
├── become
|
||||
├── cache
|
||||
├── callback
|
||||
├── cliconf
|
||||
├── connection
|
||||
├── filter
|
||||
├── httpapi
|
||||
├── inventory
|
||||
├── lookup
|
||||
├── module_utils
|
||||
├── modules
|
||||
├── netconf
|
||||
├── shell
|
||||
├── strategy
|
||||
├── terminal
|
||||
├── test
|
||||
└── vars
|
||||
```
|
||||
|
||||
A full list of plugin types can be found at [Working With Plugins](https://docs.ansible.com/ansible-core/2.16/plugins/plugins.html).
|
311
plugins/module_utils/generic.py
Normale Datei
311
plugins/module_utils/generic.py
Normale Datei
|
@ -0,0 +1,311 @@
|
|||
import builtins
|
||||
import pathlib
|
||||
import warnings
|
||||
from typing import Any, Dict, Literal, NotRequired, Optional, Required, Sequence, Tuple, Type, TypedDict
|
||||
|
||||
__all__ = (
|
||||
"Types",
|
||||
"SYSTEMD_SERVICE_CONFIG",
|
||||
"SYSTEMD_NETWORK_CONFIG",
|
||||
"SYSTEMD_CONFIG_ROOT",
|
||||
"FILE_COMMON_ARGS",
|
||||
"systemdbool",
|
||||
"AnsibleParameter",
|
||||
"AnsibleReturnParameter",
|
||||
)
|
||||
|
||||
|
||||
SYSTEMD_CONFIG_ROOT = pathlib.Path("/etc/systemd")
|
||||
SYSTEMD_NETWORK_CONFIG = SYSTEMD_CONFIG_ROOT / "network"
|
||||
SYSTEMD_SERVICE_CONFIG = SYSTEMD_CONFIG_ROOT / "system"
|
||||
FILE_COMMON_ARGS = frozenset(("owner", "group", "mode", "seuser", "serole", "setype", "selevel", "unsafe_writes"))
|
||||
|
||||
AnsibleType = Literal[
|
||||
"str",
|
||||
"bool",
|
||||
"int",
|
||||
"float",
|
||||
"path",
|
||||
"raw",
|
||||
"jsonarg",
|
||||
"json",
|
||||
"bytes",
|
||||
"dict",
|
||||
"list",
|
||||
"bits",
|
||||
]
|
||||
ReturnOptions = Literal["always", "changed", "success"]
|
||||
|
||||
|
||||
class AnsibleParameter(TypedDict, total=False):
|
||||
description: Required[str | list[str]]
|
||||
required: NotRequired[bool]
|
||||
default: NotRequired[Any]
|
||||
type: Required[AnsibleType]
|
||||
choices: NotRequired[list[Any] | tuple[Any]]
|
||||
elements: NotRequired["AnsibleParameter" | AnsibleType]
|
||||
aliases: NotRequired[list[str]]
|
||||
version_added: NotRequired[str]
|
||||
options: NotRequired[dict[str, "AnsibleParameter"]]
|
||||
|
||||
|
||||
class AnsibleReturnParameter(TypedDict, total=False):
|
||||
description: Required[str | list[str]]
|
||||
type: Required[AnsibleType]
|
||||
returned: NotRequired[ReturnOptions]
|
||||
elements: NotRequired[AnsibleType]
|
||||
sample: NotRequired[list[Any] | Any]
|
||||
version_added: NotRequired[str]
|
||||
contains: NotRequired[dict[str, "AnsibleReturnParameter"]]
|
||||
|
||||
|
||||
def wrap_func(func, **updates):
|
||||
def wrapper(*args, **kwargs):
|
||||
return func(*args, **kwargs)
|
||||
|
||||
attrs = {'__module__', '__name__', '__qualname__', '__doc__', '__annotations__', '__type_params__'}.difference(
|
||||
updates.keys()
|
||||
)
|
||||
for attr in attrs:
|
||||
try:
|
||||
value = getattr(func, attr)
|
||||
except AttributeError:
|
||||
pass
|
||||
else:
|
||||
setattr(wrapper, attr, value)
|
||||
for attr, value in updates.items():
|
||||
setattr(wrapper, attr, value)
|
||||
wrapper.__dict__.update(func.__dict__)
|
||||
setattr(wrapper, "__wrapped__", func)
|
||||
return wrapper
|
||||
|
||||
|
||||
GENERIC_DOC = """Returns an dictionary for the Ansible {type} type."""
|
||||
|
||||
|
||||
class TypeBase(type):
|
||||
def __new__(cls, clsname, bases, attrs):
|
||||
if "_default" not in attrs:
|
||||
raise TypeError(
|
||||
f"The class {clsname} must define an wrapper function called _default that returns the default type"
|
||||
)
|
||||
default_type = attrs["_default"]
|
||||
del attrs["_default"]
|
||||
types = frozenset(
|
||||
(
|
||||
"str",
|
||||
"bool",
|
||||
"int",
|
||||
"float",
|
||||
"path",
|
||||
"raw",
|
||||
"jsonarg",
|
||||
"json",
|
||||
"bytes",
|
||||
"dict",
|
||||
"list",
|
||||
"bits",
|
||||
)
|
||||
)
|
||||
for attr in types - set(attrs.keys()):
|
||||
attrs[attr] = wrap_func(
|
||||
default_type(attr),
|
||||
__doc__=GENERIC_DOC.format(type=attr),
|
||||
__name__=attr,
|
||||
__qualname__=f"{clsname}.{attr}",
|
||||
)
|
||||
attrs["__slots__"] = ()
|
||||
return super().__new__(cls, clsname, bases, attrs)
|
||||
|
||||
|
||||
class Types(metaclass=TypeBase):
|
||||
@staticmethod
|
||||
def _default(name: AnsibleType):
|
||||
def wrapped(
|
||||
help: str | list[str],
|
||||
required: bool = False,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
) -> AnsibleParameter:
|
||||
option = AnsibleParameter(type=name, required=required, description=help)
|
||||
if choices is not None:
|
||||
option["choices"] = tuple(choices)
|
||||
if default is not None:
|
||||
option["default"] = default
|
||||
return option
|
||||
|
||||
return wrapped
|
||||
|
||||
@staticmethod
|
||||
def list( # type: ignore[misc]
|
||||
elements: Type[object] | AnsibleType | AnsibleParameter,
|
||||
help: str | list[str],
|
||||
required: bool = False,
|
||||
default: list[Any] | None = None,
|
||||
) -> AnsibleParameter:
|
||||
"""Wrapper for the Ansible list type
|
||||
|
||||
Args:
|
||||
elements: The type of the elements
|
||||
required: if the item is absolutly required
|
||||
help: an helptext for the ansible-doc
|
||||
default: an default value. The Value is not converted.
|
||||
"""
|
||||
if required and default:
|
||||
raise ValueError("required and default are not allowed")
|
||||
option: AnsibleParameter = AnsibleParameter(
|
||||
type="list",
|
||||
required=required,
|
||||
description=help,
|
||||
)
|
||||
if not isinstance(elements, (str, dict)):
|
||||
option["elements"] = elements.__name__ # type:ignore[reportGeneralTypeIssue]
|
||||
elif isinstance(elements, dict):
|
||||
option["elements"] = elements["type"]
|
||||
if elements["type"] == "dict" and "options" in elements:
|
||||
option["options"] = dict()
|
||||
for name, value in elements["options"].items():
|
||||
option["options"][name] = value
|
||||
if "description" not in option["options"][name]:
|
||||
warnings.warn( # pragma: nocover
|
||||
f"helptext of option {name} is unset."
|
||||
" Ansible requires suboptions to have an documentation"
|
||||
)
|
||||
elif "choices" in elements:
|
||||
option["choices"] = elements["choices"]
|
||||
if default is not None:
|
||||
option["default"] = default
|
||||
if help is not None and isinstance(help, str):
|
||||
option["description"] = help.split("\n")
|
||||
elif help is not None:
|
||||
option["description"] = help
|
||||
return option
|
||||
|
||||
@staticmethod
|
||||
def dict(help: str | builtins.list[str], required: bool = False, **options: AnsibleParameter) -> AnsibleParameter: # type: ignore[misc]
|
||||
"""Wrapper for the Ansible dict type
|
||||
|
||||
Args:
|
||||
required: if the item is absolutly required
|
||||
help: an helptext for the ansible-doc
|
||||
options: The individual options that this parameter has
|
||||
"""
|
||||
option: AnsibleParameter = AnsibleParameter(type="dict", description=help, required=required)
|
||||
option["options"] = options
|
||||
if help is not None and isinstance(help, str):
|
||||
option["description"] = help.split("\n")
|
||||
elif help is not None:
|
||||
option["description"] = help
|
||||
return option
|
||||
|
||||
|
||||
class ReturnTypes(metaclass=TypeBase):
|
||||
@staticmethod
|
||||
def _default(name: AnsibleType):
|
||||
def wrapped(
|
||||
help: str | list[str],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
) -> AnsibleReturnParameter:
|
||||
option = AnsibleReturnParameter(type=name, description=help)
|
||||
if returned is not None:
|
||||
option["returned"] = returned
|
||||
if sample is not None:
|
||||
option["sample"] = sample
|
||||
if version_added is not None:
|
||||
option["version_added"] = version_added
|
||||
return option
|
||||
|
||||
return wrapped
|
||||
|
||||
@staticmethod
|
||||
def list(
|
||||
help: str | list[str],
|
||||
elements: AnsibleType | AnsibleReturnParameter | Type[object],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: builtins.list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
) -> AnsibleReturnParameter:
|
||||
option = AnsibleReturnParameter(description=help, type="list")
|
||||
if returned is not None:
|
||||
option["returned"] = returned
|
||||
if isinstance(elements, str):
|
||||
option["elements"] = elements
|
||||
elif isinstance(elements, dict):
|
||||
option["elements"] = elements["type"]
|
||||
if "contains" in elements:
|
||||
option["contains"] = elements["contains"]
|
||||
else:
|
||||
option["elements"] = elements.__name__ # type:ignore[reportGeneralTypeIssue]
|
||||
if sample is not None:
|
||||
option["sample"] = sample
|
||||
if version_added is not None:
|
||||
option["version_added"] = version_added
|
||||
return option
|
||||
|
||||
@staticmethod
|
||||
def dict(
|
||||
help: str | builtins.list[str],
|
||||
contains: dict[str, AnsibleReturnParameter],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: builtins.list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
):
|
||||
option = AnsibleReturnParameter(description=help, type="dict", contains=contains)
|
||||
if returned is not None:
|
||||
option["contains"] = contains
|
||||
if sample is not None:
|
||||
option["sample"] = sample
|
||||
if version_added is not None:
|
||||
option["version_added"] = version_added
|
||||
return option
|
||||
|
||||
|
||||
def systemdbool(b: bool | str) -> str:
|
||||
"""Converts values into things systemd can parse"""
|
||||
if b is True:
|
||||
return "yes"
|
||||
elif b is False:
|
||||
return "no"
|
||||
return b
|
||||
|
||||
|
||||
def modspec(
|
||||
argument_spec: Dict[str, Dict[str, Any]],
|
||||
mutually_exclusive: Sequence[Tuple[str, ...]] = (),
|
||||
required_together: Sequence[Tuple[str, ...]] = (),
|
||||
required_one_of: Sequence[Tuple[str, ...]] = (),
|
||||
required_if: Sequence[Tuple[str, Any, Tuple[str, ...]] | Tuple[str, Any, Tuple[str, ...], bool]] = (),
|
||||
required_by: Dict[str, str | Tuple[str, ...]] = {},
|
||||
supports_check_mode: bool = False,
|
||||
add_file_common_args: bool = False,
|
||||
deprecated: bool = False,
|
||||
version_added: str | None = None,
|
||||
notes: list[str] | None = None,
|
||||
extends_documentation_fragment: list[str] | None = None,
|
||||
) -> Dict[str, Any]: # pragma: nocover
|
||||
"""Wrapper to properly Type the module specs"""
|
||||
return dict(
|
||||
argument_spec=argument_spec,
|
||||
mutually_exclusive=mutually_exclusive,
|
||||
required_together=required_together,
|
||||
required_one_of=required_one_of,
|
||||
required_if=required_if,
|
||||
required_by=required_by,
|
||||
add_file_common_args=add_file_common_args,
|
||||
supports_check_mode=supports_check_mode,
|
||||
deprecated=deprecated,
|
||||
version_added=version_added,
|
||||
notes=notes,
|
||||
extends_documentation_fragment=extends_documentation_fragment,
|
||||
)
|
||||
|
||||
|
||||
def joindict(*items: dict) -> dict:
|
||||
"""merges one or more dictionaries into one"""
|
||||
odict = dict()
|
||||
for item in items:
|
||||
for key, value in item.items():
|
||||
odict[key] = value
|
||||
return odict
|
245
plugins/module_utils/generic.pyi
Normale Datei
245
plugins/module_utils/generic.pyi
Normale Datei
|
@ -0,0 +1,245 @@
|
|||
from pathlib import PosixPath
|
||||
from typing import Any, Dict, Literal, NotRequired, Optional, Required, Sequence, Tuple, Type, TypedDict
|
||||
|
||||
__all__ = [
|
||||
"Types",
|
||||
"SYSTEMD_SERVICE_CONFIG",
|
||||
"SYSTEMD_NETWORK_CONFIG",
|
||||
"SYSTEMD_CONFIG_ROOT",
|
||||
"FILE_COMMON_ARGS",
|
||||
"systemdbool",
|
||||
"AnsibleParameter",
|
||||
"AnsibleReturnParameter",
|
||||
]
|
||||
|
||||
SYSTEMD_CONFIG_ROOT: PosixPath
|
||||
SYSTEMD_NETWORK_CONFIG: PosixPath
|
||||
SYSTEMD_SERVICE_CONFIG: PosixPath
|
||||
FILE_COMMON_ARGS: frozenset[str]
|
||||
|
||||
AnsibleType = Literal[
|
||||
"str",
|
||||
"bool",
|
||||
"int",
|
||||
"float",
|
||||
"path",
|
||||
"raw",
|
||||
"jsonarg",
|
||||
"json",
|
||||
"bytes",
|
||||
"dict",
|
||||
"list",
|
||||
"bits",
|
||||
]
|
||||
ReturnOptions = Literal["always", "changed", "success"]
|
||||
|
||||
class AnsibleParameter(TypedDict, total=False):
|
||||
description: Required[str | list[str]]
|
||||
required: NotRequired[bool]
|
||||
default: NotRequired[Any]
|
||||
type: Required[AnsibleType]
|
||||
choices: NotRequired[list[Any] | tuple[Any]]
|
||||
elements: NotRequired["AnsibleParameter" | AnsibleType]
|
||||
aliases: NotRequired[list[str]]
|
||||
version_added: NotRequired[str]
|
||||
options: NotRequired[dict[str, "AnsibleParameter"]]
|
||||
|
||||
class AnsibleReturnParameter(TypedDict, total=False):
|
||||
description: Required[str | list[str]]
|
||||
type: Required[AnsibleType]
|
||||
returned: NotRequired[ReturnOptions]
|
||||
elements: NotRequired[AnsibleType]
|
||||
sample: NotRequired[list[Any] | Any]
|
||||
version_added: NotRequired[str]
|
||||
contains: NotRequired[dict[str, "AnsibleReturnParameter"]]
|
||||
|
||||
class TypeBase(type):
|
||||
def __new__(cls, clsname, bases, attrs): ...
|
||||
|
||||
class Types(metaclass=TypeBase):
|
||||
@staticmethod
|
||||
def str(
|
||||
required: bool = False,
|
||||
help: str | list[str] | None = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def bool(
|
||||
required: bool = False,
|
||||
help: str | list[str] | None = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def int(
|
||||
required: bool = False,
|
||||
help: str | list[str] | None = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def float(
|
||||
required: bool = False,
|
||||
help: str | list[str] | None = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def path(
|
||||
required: bool = False,
|
||||
help: str | list[str] | None = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def raw(
|
||||
required: bool = False,
|
||||
help: str | list[str] | None = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def jsonarg(
|
||||
required: bool = False,
|
||||
help: str | list[str] | None = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def json(
|
||||
required: bool = False,
|
||||
help: str | list[str] | None = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def bytes(
|
||||
required: bool = False,
|
||||
help: str | list[str] | None = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def bits(
|
||||
required: bool = False,
|
||||
help: str | list[str] | None = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def list(
|
||||
elements: type[object] | str | AnsibleParameter,
|
||||
required: bool = False,
|
||||
help: str | list[str] | None = None,
|
||||
default: list[Any] | None = None,
|
||||
) -> AnsibleParameter: ...
|
||||
@staticmethod
|
||||
def dict(
|
||||
required: bool = False, help: str | list[str] | None = None, **options: AnsibleParameter
|
||||
) -> AnsibleParameter: ...
|
||||
|
||||
class ReturnTypes(metaclass=TypeBase):
|
||||
@staticmethod
|
||||
def list(
|
||||
help: str | list[str],
|
||||
elements: AnsibleType | AnsibleReturnParameter | Type[object],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def dict(
|
||||
help: str | list[str],
|
||||
contains: dict[str, AnsibleReturnParameter],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def str(
|
||||
help: str | list[str],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def bool(
|
||||
help: str | list[str],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def int(
|
||||
help: str | list[str],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def float(
|
||||
help: str | list[str],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def path(
|
||||
help: str | list[str],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def raw(
|
||||
help: str | list[str],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def jsonarg(
|
||||
help: str | list[str],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def json(
|
||||
help: str | list[str],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def bytes(
|
||||
help: str | list[str],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
): ...
|
||||
@staticmethod
|
||||
def bits(
|
||||
help: str | list[str],
|
||||
returned: ReturnOptions | None = None,
|
||||
sample: list[Any] | None = None,
|
||||
version_added: str | None = None,
|
||||
): ...
|
||||
|
||||
def systemdbool(b: bool | str) -> str: ...
|
||||
def joindict(*items: dict) -> dict: ...
|
||||
def modspec(
|
||||
argument_spec: Dict[str, Dict[str, Any]],
|
||||
mutually_exclusive: Sequence[Tuple[str, ...]] = (),
|
||||
required_together: Sequence[Tuple[str, ...]] = (),
|
||||
required_one_of: Sequence[Tuple[str, ...]] = (),
|
||||
required_if: Sequence[Tuple[str, Any, Tuple[str, ...]] | Tuple[str, Any, Tuple[str, ...], bool]] = (),
|
||||
required_by: Dict[str, str | Tuple[str, ...]] = {},
|
||||
supports_check_mode: bool = False,
|
||||
add_file_common_args: bool = False,
|
||||
deprecated: bool = False,
|
||||
version_added: str | None = None,
|
||||
notes: list[str] | None = None,
|
||||
extends_documentation_fragment: list[str] | None = None,
|
||||
) -> Dict[str, Any]: ...
|
|
@ -1,11 +1,12 @@
|
|||
import os
|
||||
import pathlib
|
||||
import shutil
|
||||
from copy import deepcopy
|
||||
from typing import (Any, Callable, ClassVar, Dict, NoReturn, Optional, Type,
|
||||
TypeVar, Union, overload)
|
||||
from typing import Any, Callable, ClassVar, Dict, Generic, NoReturn, Optional, Type, TypedDict, TypeVar, overload
|
||||
|
||||
import ansible.module_utils.basic as basic
|
||||
|
||||
from ansible_module.generic import AnsibleParameter, Types, systemdbool
|
||||
from .generic import AnsibleParameter, AnsibleReturnParameter, Types, modspec, systemdbool
|
||||
|
||||
__all__ = (
|
||||
"AnsibleModule",
|
||||
|
@ -18,28 +19,35 @@ __all__ = (
|
|||
T = TypeVar("T")
|
||||
|
||||
|
||||
def docify(input: Union[dict, AnsibleParameter]) -> dict:
|
||||
class TypedDiff(Generic[T], TypedDict, total=False):
|
||||
before: T
|
||||
after: T
|
||||
before_header: str
|
||||
after_header: str
|
||||
|
||||
|
||||
def docify(input: AnsibleParameter) -> dict:
|
||||
options = dict()
|
||||
for name, help in input.items():
|
||||
options[name] = dict(type=help["type"])
|
||||
if "description" in help:
|
||||
if isinstance(help["description"], str):
|
||||
help["description"] = help["description"].split("\n")
|
||||
options[name]["description"] = help["description"]
|
||||
if "required" in help and help["required"]:
|
||||
options[name] = dict(type=help["type"]) # type: ignore[reportIndexIssue]
|
||||
if "description" in help: # type: ignore[reportOperatorIssue]
|
||||
if isinstance(help["description"], str): # type: ignore[reportIndexIssue]
|
||||
help["description"] = help["description"].split("\n") # type: ignore[reportIndexIssue]
|
||||
options[name]["description"] = help["description"] # type: ignore[reportIndexIssue]
|
||||
if "required" in help and help["required"]: # type: ignore[reportOperatorIssue]
|
||||
options[name]["required"] = True
|
||||
else:
|
||||
options[name]["required"] = False
|
||||
if help["type"] == "list":
|
||||
options[name]["elements"] = help["elements"]
|
||||
if help["type"] == "list": # type: ignore[reportOperatorIssue]
|
||||
options[name]["elements"] = help["elements"] # type: ignore[reportIndexIssue]
|
||||
if not options[name]["required"]:
|
||||
options[name]["default"] = []
|
||||
if "default" in help:
|
||||
options[name]["default"] = help["default"]
|
||||
if "options" in help:
|
||||
options[name]["options"] = docify(help["options"])
|
||||
if "choices" in help:
|
||||
options[name]["choices"] = tuple(help["choices"])
|
||||
if "default" in help: # type: ignore[reportOperatorIssue]
|
||||
options[name]["default"] = help["default"] # type: ignore[reportIndexIssue]
|
||||
if "options" in help and help["options"] != {}: # type: ignore[reportOperatorIssue]
|
||||
options[name]["options"] = docify(help["options"]) # type: ignore[reportIndexIssue]
|
||||
if "choices" in help and len(help["choices"]) > 0: # type: ignore[reportOperatorIssue]
|
||||
options[name]["choices"] = tuple(help["choices"]) # type: ignore[reportIndexIssue]
|
||||
return options
|
||||
|
||||
|
||||
|
@ -54,6 +62,10 @@ class AnsibleModule(object):
|
|||
result: dict
|
||||
#: the specification of the arguments. Subclasses that are usable Modules must set this value.
|
||||
module_spec: ClassVar[dict]
|
||||
|
||||
#: The specification of return arguments
|
||||
return_spec: ClassVar[dict[str, AnsibleReturnParameter]]
|
||||
|
||||
#: This is set by classes that define common things for their subclasses, like behaviour of the run and check methods. This is used by `SystemdUnitModule`
|
||||
_common_args: ClassVar[dict[str, Any]] = dict()
|
||||
|
||||
|
@ -62,7 +74,11 @@ class AnsibleModule(object):
|
|||
"""params is an wrapper for the module.params"""
|
||||
return self.module.params # type: ignore
|
||||
|
||||
def __init__(self):
|
||||
def __init__(self, documentation=False):
|
||||
"""
|
||||
Args:
|
||||
documentation: Only true if the module is initialized for documentation. The Ansible Module is None is this case
|
||||
"""
|
||||
self.result = dict(changed=False)
|
||||
specs = dict()
|
||||
specs.update(deepcopy(self._common_args))
|
||||
|
@ -71,15 +87,20 @@ class AnsibleModule(object):
|
|||
specs["argument_spec"].update(modspec["argument_spec"])
|
||||
del modspec["argument_spec"]
|
||||
specs.update(modspec)
|
||||
self.module = basic.AnsibleModule(**specs)
|
||||
self.tmpdir = pathlib.Path(self.module.tmpdir)
|
||||
self.modspec = specs.copy()
|
||||
specs.pop("deprecated", None)
|
||||
specs.pop("notes", None)
|
||||
specs.pop("version_added", None)
|
||||
if not documentation:
|
||||
self.module = basic.AnsibleModule(**specs)
|
||||
self.tmpdir = pathlib.Path(self.module.tmpdir)
|
||||
|
||||
def set(self, key: str, value):
|
||||
"""sets an value for the result"""
|
||||
self.result[key] = value
|
||||
|
||||
@overload
|
||||
def diff(self, diff: Dict[str, str]): # pragma: nocover
|
||||
def diff(self, diff: TypedDiff): # pragma: nocover
|
||||
pass
|
||||
|
||||
@overload
|
||||
|
@ -107,7 +128,7 @@ class AnsibleModule(object):
|
|||
if diff is not None and not any((before is not None, after is not None)):
|
||||
pass
|
||||
elif all((before is not None, after is not None, diff is None)):
|
||||
diff = dict(
|
||||
diff = TypedDiff(
|
||||
before=before,
|
||||
after=after,
|
||||
)
|
||||
|
@ -158,12 +179,8 @@ class AnsibleModule(object):
|
|||
except Exception as exc:
|
||||
import traceback
|
||||
|
||||
self.module.fail_json(
|
||||
"".join(traceback.format_exception(type(exc), exc, exc.__traceback__)),
|
||||
**self.result,
|
||||
)
|
||||
self.module.exit_json(**self.result)
|
||||
raise Exception("exit_json failed")
|
||||
self.fail("".join(traceback.format_exception(type(exc), exc, exc.__traceback__)))
|
||||
self.exit()
|
||||
|
||||
@classmethod
|
||||
def doc(cls) -> str:
|
||||
|
@ -175,26 +192,81 @@ class AnsibleModule(object):
|
|||
doc = cls.__doc__
|
||||
if doc is None:
|
||||
doc = ""
|
||||
specs = dict()
|
||||
if "argument_spec" in cls._common_args: # pragma: nocover
|
||||
specs.update(cls._common_args["argument_spec"])
|
||||
if "argument_spec" in cls.module_spec: # pragma: nocover
|
||||
specs.update(cls.module_spec["argument_spec"])
|
||||
options = docify(specs)
|
||||
mod = cls(documentation=True)
|
||||
options = docify(mod.modspec["argument_spec"])
|
||||
docu = doc.split("\n")
|
||||
documentation: dict[str, Any] = dict(
|
||||
module=cls.name,
|
||||
short_description=docu[0],
|
||||
description=docu,
|
||||
options=options,
|
||||
)
|
||||
if mod.modspec.get("extends_documentation_fragment", None) is not None or mod.modspec.get(
|
||||
"add_file_common_args", False
|
||||
):
|
||||
documentation["extends_documentation_fragment"] = []
|
||||
if mod.modspec.get("extends_documentation_fragment", None) is not None:
|
||||
documentation["extends_documentation_fragment"].extend(mod.modspec["extends_documentation_fragment"])
|
||||
if mod.modspec.get("add_file_common_args", False):
|
||||
documentation["extends_documentation_fragment"].append("ansible.builtin.files")
|
||||
if mod.modspec.get("deprecated", False):
|
||||
documentation["deprecated"] = True
|
||||
if mod.modspec.get("notes", None) is not None:
|
||||
documentation["notes"] = mod.modspec["notes"]
|
||||
if mod.modspec.get("version_added", None) is not None:
|
||||
documentation["version_added"] = mod.modspec["version_added"]
|
||||
|
||||
return str(
|
||||
yaml.safe_dump(
|
||||
dict(
|
||||
module=cls.name,
|
||||
short_description=docu[0],
|
||||
description=docu,
|
||||
options=options,
|
||||
),
|
||||
documentation,
|
||||
stream=None,
|
||||
explicit_start=True,
|
||||
)
|
||||
)
|
||||
|
||||
@classmethod
|
||||
def returns(cls) -> str:
|
||||
"""This returns the string for the RETURNS String of the module documentation"""
|
||||
try:
|
||||
import yaml
|
||||
except ImportError: # pragma: nocover
|
||||
return "---\n"
|
||||
if not hasattr(cls, "return_spec"):
|
||||
return "--\n"
|
||||
return str(
|
||||
yaml.safe_dump(
|
||||
cls.returns,
|
||||
stream=None,
|
||||
explicit_start=True,
|
||||
)
|
||||
)
|
||||
|
||||
def fail(self, message: str) -> NoReturn: # type: ignore[reportReturnType]
|
||||
"""Wrapper for AnsibleModule.fail_json"""
|
||||
self.module.fail_json(message, **self.result)
|
||||
|
||||
def exit(self) -> NoReturn: # type: ignore[reportReturnType]
|
||||
"""Wrapper for AnsibleModule.exit_json"""
|
||||
self.module.exit_json(**self.result)
|
||||
|
||||
def move_file(self, path: pathlib.Path, dest: pathlib.Path, backup: bool = False, unsafe_writes: bool = False):
|
||||
"""Moves an Temporary file to an destination it uses the args from add_file_common_args when used
|
||||
|
||||
Args:
|
||||
path: The Path that the file currently is
|
||||
dest: the location the file should be
|
||||
backup: should an backup be made before the file is moved
|
||||
"""
|
||||
|
||||
if backup:
|
||||
shutil.copy2(dest, dest.with_suffix(dest.suffix + ".bak"), follow_symlinks=False)
|
||||
self.module.atomic_move(path, dest, unsafe_writes=unsafe_writes, keep_dest_attrs=True)
|
||||
if "add_file_common_args" in self.modspec and self.modspec["add_file_common_args"]:
|
||||
file_args = self.module.load_file_common_arguments(self.params, path=dest)
|
||||
diff = TypedDiff()
|
||||
self.changed |= self.module.set_fs_attributes_if_different(file_args, diff=diff)
|
||||
self.diff(diff)
|
||||
|
||||
|
||||
class SystemdUnitModule(AnsibleModule):
|
||||
#: path of the unitfile managed by this module
|
||||
|
@ -229,30 +301,37 @@ class SystemdUnitModule(AnsibleModule):
|
|||
),
|
||||
)
|
||||
#: if defined it will be called after run has changed the unitfile
|
||||
post: Optional[Callable[[], None]]
|
||||
post: Optional[Callable[[], None]] = None
|
||||
|
||||
#: generates the install section if the unit is installable
|
||||
install: ClassVar[Optional[Callable[["SystemdUnitModule"], str]]]
|
||||
install: ClassVar[Optional[Callable[["SystemdUnitModule"], str | None]]] = None
|
||||
|
||||
def unit(self) -> str: # pragma: nocover
|
||||
raise NotImplementedError()
|
||||
|
||||
def header(self) -> str:
|
||||
header = "[Unit]\n"
|
||||
header += "".join(
|
||||
self.map_param(
|
||||
description="Description",
|
||||
documentation="Documentation",
|
||||
requires="Requires",
|
||||
wants="Wants",
|
||||
partof="PartOf",
|
||||
before="Before",
|
||||
after="After",
|
||||
)
|
||||
def header(self) -> Optional[str]:
|
||||
parts = self.map_param(
|
||||
description="Description",
|
||||
documentation="Documentation",
|
||||
requires="Requires",
|
||||
wants="Wants",
|
||||
partof="PartOf",
|
||||
before="Before",
|
||||
after="After",
|
||||
)
|
||||
if len(parts) == 0:
|
||||
return None
|
||||
header = "[Unit]\n" + "".join(parts)
|
||||
return header
|
||||
|
||||
def map_param(self, **parammap: str):
|
||||
def _unit(self, *parts: Optional[str]) -> str:
|
||||
opart = []
|
||||
for part in parts:
|
||||
if part is not None:
|
||||
opart.append(part)
|
||||
return "\n".join(opart)
|
||||
|
||||
def map_param(self, **parammap: str) -> list[str]:
|
||||
"""maps an dict with keys for an section with given params. The key of the dictionary is the parameter and the value is the key in the unitfile. If an parameter has multiple values it adds multiple entries"""
|
||||
output: list[str] = []
|
||||
for param, key in parammap.items():
|
||||
|
@ -264,38 +343,10 @@ class SystemdUnitModule(AnsibleModule):
|
|||
output.append(f"{key}={systemdbool(params)}\n")
|
||||
return output
|
||||
|
||||
def unitfile_gen(self): # pragma: nocover
|
||||
def unitfile_gen(self) -> None: # pragma: nocover
|
||||
path = self.tmpdir / "newunit"
|
||||
with open(path, "w") as unit:
|
||||
unit.write(self.unit())
|
||||
self.module.set_owner_if_different(path.as_posix(), "root", False)
|
||||
self.module.set_group_if_different(path.as_posix(), "root", False)
|
||||
self.module.set_mode_if_different(path.as_posix(), "0644", False)
|
||||
if self.unitfile.exists():
|
||||
diff = dict()
|
||||
self.changed = self.changed | self.module.set_owner_if_different(
|
||||
self.unitfile.as_posix(),
|
||||
"root",
|
||||
self.result["changed"],
|
||||
diff,
|
||||
)
|
||||
self.diff(diff)
|
||||
diff = dict()
|
||||
self.changed = self.changed | self.module.set_group_if_different(
|
||||
self.unitfile.as_posix(),
|
||||
"root",
|
||||
self.result["changed"],
|
||||
diff,
|
||||
)
|
||||
self.diff(diff)
|
||||
diff = dict()
|
||||
self.changed = self.changed | self.module.set_mode_if_different(
|
||||
self.unitfile.as_posix(),
|
||||
"0644",
|
||||
self.result["changed"],
|
||||
diff,
|
||||
)
|
||||
self.diff(diff)
|
||||
|
||||
def check(self): # pragma: nocover
|
||||
self.set("unitfile", self.unitfile.as_posix())
|
||||
|
@ -316,10 +367,7 @@ class SystemdUnitModule(AnsibleModule):
|
|||
self.check()
|
||||
if not self.changed:
|
||||
return
|
||||
self.module.atomic_move(
|
||||
src=(self.tmpdir / "newunit").as_posix(),
|
||||
dest=self.unitfile.as_posix(),
|
||||
)
|
||||
self.move_file(self.tmpdir / "newunit", self.unitfile)
|
||||
if hasattr(self, "post") and self.post is not None:
|
||||
self.post()
|
||||
|
||||
|
@ -343,13 +391,15 @@ def installable(_class: Type[SystemdUnitModule]): # pragma: nocover
|
|||
)
|
||||
specs["argument_spec"].update(arguments)
|
||||
|
||||
def install(self: SystemdUnitModule) -> str:
|
||||
output = "[Install]\n"
|
||||
def install(self: SystemdUnitModule) -> Optional[str]:
|
||||
parts = []
|
||||
for argument, key in _INSTALL_MAPPING.items():
|
||||
if self.get(argument, False):
|
||||
for unit in self.get(argument): # type: ignore
|
||||
output += "{}={}\n".format(key, unit)
|
||||
return output
|
||||
parts.append("{}={}\n".format(key, unit))
|
||||
if len(parts) == 0:
|
||||
return None
|
||||
return "[Install]\n" + "".join(parts)
|
||||
|
||||
_class.install = install
|
||||
_class.module_spec = specs
|
0
plugins/module_utils/py.typed
Normale Datei
0
plugins/module_utils/py.typed
Normale Datei
|
@ -7,23 +7,20 @@ name = "ansible-module"
|
|||
dynamic = ["version"]
|
||||
description = 'Helps with developing modules for ansible in an easier manner'
|
||||
readme = "README.md"
|
||||
requires-python = ">=3.7"
|
||||
requires-python = ">=3.11"
|
||||
license = "MIT"
|
||||
keywords = []
|
||||
authors = [{ name = "Sebastian Tobie", email = "sebastian@sebastian-tobie.de" }]
|
||||
classifiers = [
|
||||
"Development Status :: 4 - Beta",
|
||||
"Programming Language :: Python",
|
||||
"Programming Language :: Python :: 3.7",
|
||||
"Programming Language :: Python :: 3.8",
|
||||
"Programming Language :: Python :: 3.9",
|
||||
"Programming Language :: Python :: 3.10",
|
||||
"Programming Language :: Python :: 3.11",
|
||||
"Programming Language :: Python :: Implementation :: CPython",
|
||||
"Programming Language :: Python :: Implementation :: PyPy",
|
||||
]
|
||||
dependencies = ["ansible-core>=2.9.10"]
|
||||
|
||||
dependencies = ["ansible>=11.3.0"]
|
||||
[tool.hatch.build]
|
||||
directory = "dist/python"
|
||||
[project.urls]
|
||||
Documentation = "https://github.com/unknown/ansible-module#readme"
|
||||
Issues = "https://github.com/unknown/ansible-module/issues"
|
||||
|
@ -32,6 +29,9 @@ Source = "https://github.com/unknown/ansible-module"
|
|||
[project.scripts]
|
||||
update-doc = "ansible_module.update_doc:main"
|
||||
|
||||
[tool.hatch.build.targets.wheel.force-include]
|
||||
"plugins/module_utils" = "ansible_module/module_utils"
|
||||
|
||||
[tool.hatch.version]
|
||||
path = "src/ansible_module/__init__.py"
|
||||
|
||||
|
@ -135,26 +135,30 @@ ansible_module = ["src/ansible_module", "*/ansible-module/src/ansible_module"]
|
|||
tests = ["tests", "*/ansible-module/tests"]
|
||||
|
||||
[tool.coverage.report]
|
||||
exclude_lines = ["no cov", "if __name__ == .__main__.:", "if TYPE_CHECKING:", "pragma: nocover"]
|
||||
exclude_lines = [
|
||||
"no cov",
|
||||
"if __name__ == .__main__.:",
|
||||
"if TYPE_CHECKING:",
|
||||
"pragma: nocover",
|
||||
]
|
||||
|
||||
[tool.mypy]
|
||||
disable_error_code = "import-untyped"
|
||||
|
||||
python_version = "3.11"
|
||||
warn_return_any = true
|
||||
warn_unused_configs = true
|
||||
|
||||
[[tool.mypy.overrides]]
|
||||
module = ["ansible", "ansible.module_utils", "ansible.module_utils.basic"]
|
||||
ignore_missing_imports = true
|
||||
|
||||
[tool.pytest.ini_options]
|
||||
addopts = [
|
||||
"--cov-report", "json",
|
||||
"--cov-report", "term-missing:skip-covered",
|
||||
"--cov-report",
|
||||
"json",
|
||||
"--cov-report",
|
||||
"term-missing:skip-covered",
|
||||
"--cov",
|
||||
]
|
||||
pythonpath = ["src"]
|
||||
required_plugins = [
|
||||
"pytest-cov",
|
||||
"pytest-isort",
|
||||
"pytest-mypy"
|
||||
]
|
||||
required_plugins = ["pytest-cov", "pytest-isort", "pytest-mypy"]
|
||||
|
|
|
@ -1 +1 @@
|
|||
__version__ = "0.1.3"
|
||||
__version__ = "0.5.0"
|
||||
|
|
|
@ -1,6 +1,4 @@
|
|||
from . import update_doc
|
||||
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
update_doc.main()
|
||||
|
|
|
@ -1,135 +0,0 @@
|
|||
import pathlib
|
||||
import warnings
|
||||
from typing import Any, Callable, Dict, Optional, Sequence, Tuple, Type, Union
|
||||
|
||||
__all__ = (
|
||||
"Types",
|
||||
"SYSTEMD_SERVICE_CONFIG",
|
||||
"SYSTEMD_NETWORK_CONFIG",
|
||||
"SYSTEMD_CONFIG_ROOT",
|
||||
"systemdbool",
|
||||
"AnsibleParameter",
|
||||
)
|
||||
|
||||
|
||||
SYSTEMD_CONFIG_ROOT = pathlib.Path("/etc/systemd")
|
||||
SYSTEMD_NETWORK_CONFIG = SYSTEMD_CONFIG_ROOT / "network"
|
||||
SYSTEMD_SERVICE_CONFIG = SYSTEMD_CONFIG_ROOT / "system"
|
||||
|
||||
AnsibleParameter = Dict[str, Any]
|
||||
|
||||
|
||||
class _Type:
|
||||
def __dir__(self) -> tuple: # pragma: nocover
|
||||
return (
|
||||
"str",
|
||||
"bool",
|
||||
"int",
|
||||
"float",
|
||||
"path",
|
||||
"raw",
|
||||
"jsonarg",
|
||||
"json",
|
||||
"bytes",
|
||||
"dict",
|
||||
"list",
|
||||
"bits",
|
||||
"__doc__",
|
||||
)
|
||||
|
||||
def __repr__(self) -> str: # pragma: nocover
|
||||
return "Types()"
|
||||
|
||||
def __call__(self) -> "_Type": # pragma: nocover
|
||||
return self
|
||||
|
||||
def list(
|
||||
self,
|
||||
elements: Union[Type[object], str, AnsibleParameter],
|
||||
required: bool = False,
|
||||
help: Optional[str] = None,
|
||||
) -> AnsibleParameter:
|
||||
option: AnsibleParameter = dict(type="list", required=required)
|
||||
if not isinstance(elements, (str, dict)):
|
||||
option["elements"] = elements.__name__
|
||||
elif isinstance(elements, dict):
|
||||
option["elements"] = elements["type"]
|
||||
if elements["type"] == "dict":
|
||||
option["options"] = dict()
|
||||
for name, value in elements["option"].items():
|
||||
option["options"][name] = value
|
||||
if "description" not in option["options"][name]:
|
||||
warnings.warn( # pragma: nocover
|
||||
f"helptext of option {name} is unset."
|
||||
" Ansible requires suboptions to have an documentation"
|
||||
)
|
||||
if help is not None:
|
||||
option["description"] = help.split("\n")
|
||||
return option
|
||||
|
||||
def dict(self, required: bool = False, help: Optional[str] = None, **options: dict) -> AnsibleParameter:
|
||||
option: AnsibleParameter = dict(type="dict", required=required)
|
||||
option["option"] = options
|
||||
if help is not None:
|
||||
option["description"] = help.split("\n")
|
||||
return option
|
||||
|
||||
def __getattr__(self, name: str):
|
||||
def argument(
|
||||
required: bool = False,
|
||||
help: Optional[str] = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
) -> AnsibleParameter:
|
||||
option: AnsibleParameter = dict(type=name, required=required)
|
||||
if choices is not None:
|
||||
option["choices"] = choices
|
||||
if default is not None:
|
||||
option["default"] = default
|
||||
if help is not None:
|
||||
option["description"] = help.split("\n")
|
||||
return option
|
||||
|
||||
argument.__name__ = name
|
||||
argument.__qualname__ = f"Types.{name}"
|
||||
argument.__doc__ = f"Simple wrapper for Ansible {name} argument dict"
|
||||
return argument
|
||||
|
||||
|
||||
Types = _Type()
|
||||
|
||||
|
||||
def systemdbool(b: Union[bool, str]) -> str:
|
||||
"""Converts values into things systemd can parse"""
|
||||
if b is True:
|
||||
return "yes"
|
||||
elif b is False:
|
||||
return "no"
|
||||
return b
|
||||
|
||||
|
||||
def modspec(
|
||||
argument_spec: Dict[str, Dict[str, Any]],
|
||||
mutually_exclusive: Sequence[Tuple[str, ...]] = (),
|
||||
required_together: Sequence[Tuple[str, ...]] = (),
|
||||
required_one_of: Sequence[Tuple[str, ...]] = (),
|
||||
required_if: Sequence[Union[Tuple[str, Any, Tuple[str, ...]], Tuple[str, Any, Tuple[str, ...], bool]]] = (),
|
||||
required_by: Dict[str, Union[str, Tuple[str, ...]]] = {},
|
||||
) -> Dict[str, Any]: # pragma: nocover
|
||||
return dict(
|
||||
argument_spec=argument_spec,
|
||||
mutually_exclusive=mutually_exclusive,
|
||||
required_together=required_together,
|
||||
required_one_of=required_one_of,
|
||||
required_if=required_if,
|
||||
required_by=required_by,
|
||||
)
|
||||
|
||||
|
||||
def joindict(*items: dict) -> dict:
|
||||
"""merges one or more dictionaries into one"""
|
||||
odict = dict()
|
||||
for item in items:
|
||||
for key, value in item.items():
|
||||
odict[key] = value
|
||||
return odict
|
|
@ -1,76 +0,0 @@
|
|||
from typing import Any, Dict, Optional, Type, Union
|
||||
|
||||
from _typeshed import Incomplete
|
||||
|
||||
SYSTEMD_CONFIG_ROOT: Incomplete
|
||||
SYSTEMD_NETWORK_CONFIG: Incomplete
|
||||
SYSTEMD_SERVICE_CONFIG: Incomplete
|
||||
AnsibleParameter = Dict[str, Any]
|
||||
|
||||
class Types:
|
||||
def list(
|
||||
elements: Union[Type[object], str, AnsibleParameter], required: bool = False, help: Optional[str] = None
|
||||
) -> AnsibleParameter: ...
|
||||
def dict(required: bool = False, help: Optional[str] = None, **options: dict) -> AnsibleParameter: ...
|
||||
def str(
|
||||
required: bool = False,
|
||||
help: Optional[str] = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
) -> AnsibleParameter: ...
|
||||
def bool(
|
||||
required: bool = False,
|
||||
help: Optional[str] = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
) -> AnsibleParameter: ...
|
||||
def int(
|
||||
required: bool = False,
|
||||
help: Optional[str] = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
) -> AnsibleParameter: ...
|
||||
def float(
|
||||
required: bool = False,
|
||||
help: Optional[str] = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
) -> AnsibleParameter: ...
|
||||
def path(
|
||||
required: bool = False,
|
||||
help: Optional[str] = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
) -> AnsibleParameter: ...
|
||||
def raw(
|
||||
required: bool = False,
|
||||
help: Optional[str] = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
) -> AnsibleParameter: ...
|
||||
def jsonarg(
|
||||
required: bool = False,
|
||||
help: Optional[str] = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
) -> AnsibleParameter: ...
|
||||
def json(
|
||||
required: bool = False,
|
||||
help: Optional[str] = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
) -> AnsibleParameter: ...
|
||||
def bytes(
|
||||
required: bool = False,
|
||||
help: Optional[str] = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
) -> AnsibleParameter: ...
|
||||
def bits(
|
||||
required: bool = False,
|
||||
help: Optional[str] = None,
|
||||
choices: Optional[Sequence] = None,
|
||||
default: Optional[Any] = None,
|
||||
) -> AnsibleParameter: ...
|
||||
|
||||
def systemdbool(b: Union[bool, str]) -> str: ...
|
|
@ -1,60 +0,0 @@
|
|||
import pathlib
|
||||
from typing import (Any, Callable, ClassVar, Dict, NoReturn, Optional, Type,
|
||||
TypeVar, overload)
|
||||
|
||||
import ansible.module_utils.basic as basic
|
||||
from _typeshed import Incomplete
|
||||
|
||||
T = TypeVar('T')
|
||||
|
||||
class AnsibleModule:
|
||||
name: ClassVar[str]
|
||||
module: basic.AnsibleModule
|
||||
result: dict
|
||||
module_spec: ClassVar[dict]
|
||||
@property
|
||||
def params(self) -> Dict[str, Any]: ...
|
||||
tmpdir: Incomplete
|
||||
def __init__(self) -> None: ...
|
||||
def set(self, key: str, value): ...
|
||||
@overload
|
||||
def diff(self, diff: Dict[str, str]): ...
|
||||
@overload
|
||||
def diff(
|
||||
self,
|
||||
before: Optional[str] = ...,
|
||||
after: Optional[str] = ...,
|
||||
before_header: Optional[str] = ...,
|
||||
after_header: Optional[str] = ...,
|
||||
): ...
|
||||
def get(self, key: str, default: Optional[T] = ...) -> T: ...
|
||||
def changed_get(self): ...
|
||||
def changed_set(self, value) -> None: ...
|
||||
changed: Incomplete
|
||||
def prepare(self) -> None: ...
|
||||
def check(self) -> None: ...
|
||||
def run(self) -> None: ...
|
||||
def __call__(self) -> NoReturn: ...
|
||||
@classmethod
|
||||
def doc(cls) -> str: ...
|
||||
|
||||
class SystemdUnitModule(AnsibleModule):
|
||||
unitfile: pathlib.Path
|
||||
post: Optional[Callable[[], None]]
|
||||
install: ClassVar[Optional[Callable[[SystemdUnitModule], str]]]
|
||||
def unit(self) -> str: ...
|
||||
def header(self) -> str: ...
|
||||
def map_param(self, **parammap: str): ...
|
||||
changed: Incomplete
|
||||
def unitfile_gen(self) -> None: ...
|
||||
def check(self) -> None: ...
|
||||
def run(self) -> None: ...
|
||||
|
||||
def installable(_class: Type[SystemdUnitModule]): ...
|
||||
|
||||
class SystemdReloadMixin:
|
||||
module: basic.AnsibleModule
|
||||
unitfile: pathlib.Path
|
||||
restartable: bool
|
||||
changed: bool
|
||||
def post(self) -> None: ...
|
0
src/ansible_module/py.typed
Normale Datei
0
src/ansible_module/py.typed
Normale Datei
|
@ -4,13 +4,18 @@ import pathlib
|
|||
import re
|
||||
import sys
|
||||
|
||||
from .module_utils.module import AnsibleModule
|
||||
|
||||
sys.path.append(".")
|
||||
mindocstring = "DOCUMENTATION = ''''''"
|
||||
moduledir = pathlib.Path("plugins/modules")
|
||||
regex = re.compile("DOCUMENTATION *= *r?(?P<quote>\"{3}|'{3})(---)?.*?(?P=quote)", re.MULTILINE | re.DOTALL)
|
||||
regex = re.compile(
|
||||
"(?P<type>DOCUMENTATION|RETURNS) *= *r?(?P<quote>\"{3}|'{3})(---)?.*?(?P=quote)",
|
||||
re.MULTILINE | re.DOTALL,
|
||||
)
|
||||
|
||||
|
||||
def main():
|
||||
def main() -> None:
|
||||
try:
|
||||
modules = list(moduledir.iterdir())
|
||||
except:
|
||||
|
@ -20,6 +25,7 @@ def main():
|
|||
if modfile.name in ("__init__.py", "__pycache__", "unit.py.example"):
|
||||
continue
|
||||
mod = importlib.import_module(".".join((modfile.parts[:-1]) + (modfile.stem,)))
|
||||
module: AnsibleModule
|
||||
if hasattr(mod, "Module"):
|
||||
module = mod.Module
|
||||
elif hasattr(mod, "__module_name__"):
|
||||
|
@ -29,6 +35,7 @@ def main():
|
|||
continue
|
||||
try:
|
||||
moddoc = module.doc()
|
||||
returns = module.returns()
|
||||
except AttributeError:
|
||||
print("Broken module. skipping {}".format(modfile))
|
||||
continue
|
||||
|
@ -36,16 +43,37 @@ def main():
|
|||
print("Error in documentation of module {}: {}".format(modfile, e))
|
||||
continue
|
||||
moddata = modfile.read_text()
|
||||
match = regex.search(moddata)
|
||||
if not match:
|
||||
changed = False
|
||||
start = 0
|
||||
while True:
|
||||
match = regex.search(moddata, pos=start)
|
||||
if not match:
|
||||
break
|
||||
changed = True
|
||||
type = match.group("type")
|
||||
content = ""
|
||||
if type == "DOCUMENTATION":
|
||||
content = moddoc
|
||||
elif type == "RETURNS":
|
||||
content = returns
|
||||
else:
|
||||
Exception("Please update the script: Unknown Type")
|
||||
after = moddata[match.end() :]
|
||||
moddata = "{pre}{type} = r{quote}{doc}{quote}".format(
|
||||
pre=moddata[: match.start()],
|
||||
type=type,
|
||||
quote=match.group("quote"),
|
||||
doc=content,
|
||||
)
|
||||
start = len(moddata)
|
||||
moddata = f"{moddata}{after}"
|
||||
if changed is False:
|
||||
print(
|
||||
"no Documentation set for module {}. Please add at least \"{}\" to the file".format(
|
||||
modfile.stem, mindocstring
|
||||
)
|
||||
)
|
||||
continue
|
||||
newmod = "{pre}DOCUMENTATION = {quote}{doc}{quote}{post}".format(
|
||||
pre=moddata[: match.start()], quote=match.group("quote"), doc=moddoc, post=moddata[match.end() :]
|
||||
)
|
||||
modfile.write_text(newmod)
|
||||
|
||||
modfile.write_text(moddata)
|
||||
print("updated the documentation of module {}".format(module.name))
|
||||
|
|
|
@ -1,3 +0,0 @@
|
|||
# SPDX-FileCopyrightText: 2023-present Sebastian Tobie <sebastian@sebastian-tobie.de>
|
||||
#
|
||||
# SPDX-License-Identifier: MIT
|
|
@ -1,57 +0,0 @@
|
|||
import os
|
||||
import unittest
|
||||
|
||||
from ansible_module.generic import Types, joindict, systemdbool
|
||||
|
||||
|
||||
class TestTypes(unittest.TestCase):
|
||||
"""tests the Types class"""
|
||||
|
||||
def testsimpletype(self):
|
||||
"""this tests if an simple type is correctly build"""
|
||||
output = Types.str(required=True, help="test", choices=("a", "1"), default="1")
|
||||
self.assertIn("type", output)
|
||||
self.assertIn("required", output)
|
||||
self.assertIn("default", output)
|
||||
self.assertIn("choices", output)
|
||||
self.assertEqual(output["type"], "str")
|
||||
self.assertEqual(Types.str.__name__, "str")
|
||||
self.assertEqual(output["required"], True)
|
||||
self.assertEqual(output["default"], "1")
|
||||
self.assertTupleEqual(output["choices"], ("a", "1"))
|
||||
Types.str()
|
||||
|
||||
def testlisttype(self):
|
||||
"""this tests if the special type list is correctly build"""
|
||||
output = Types.list(str)
|
||||
Types.list("str")
|
||||
self.assertIn("type", output)
|
||||
self.assertIn("elements", output)
|
||||
self.assertIn("required", output)
|
||||
self.assertEqual(output["type"], "list")
|
||||
self.assertEqual(output["required"], False)
|
||||
self.assertEqual(output["elements"], "str")
|
||||
Types.list(Types.dict(a=Types.str(help="")), help="")
|
||||
Types.list("str")
|
||||
|
||||
def testdicttype(self):
|
||||
output = Types.dict(help="HILFE")
|
||||
self.assertIn("type", output)
|
||||
self.assertIn("required", output)
|
||||
self.assertIn("description", output)
|
||||
self.assertIn("option", output)
|
||||
|
||||
|
||||
class TestFuncs(unittest.TestCase):
|
||||
def testsystemdbool(self):
|
||||
self.assertEqual("Text", systemdbool("Text"))
|
||||
self.assertEqual("no", systemdbool(False))
|
||||
self.assertEqual("yes", systemdbool(True))
|
||||
|
||||
def testjoindict(self):
|
||||
dicts = (
|
||||
dict(a=1, b=2),
|
||||
dict(b=3, c=4),
|
||||
)
|
||||
output = dict(a=1, b=3, c=4)
|
||||
self.assertDictEqual(output, joindict(*dicts))
|
|
@ -1,71 +0,0 @@
|
|||
import sys
|
||||
import unittest
|
||||
|
||||
from ansible_module.generic import modspec
|
||||
from ansible_module.module import SystemdUnitModule, docify
|
||||
|
||||
|
||||
class ModuleMock(SystemdUnitModule):
|
||||
name = "mock"
|
||||
module_spec = modspec({})
|
||||
|
||||
|
||||
class TestSystemdUnitModule(unittest.TestCase):
|
||||
def setUp(self):
|
||||
sys.argv = [sys.argv[0], '{"ANSIBLE_MODULE_ARGS":{}}']
|
||||
self.mod = ModuleMock()
|
||||
|
||||
def test_header(self):
|
||||
self.assertEqual("[Unit]\n", self.mod.header())
|
||||
|
||||
def test_doc(self):
|
||||
ModuleMock.doc()
|
||||
|
||||
def test_params(self):
|
||||
self.mod.module.params = dict(ok="ok", none=None, true=True, false=False, list=("item1", "item2"))
|
||||
self.assertEqual("ok", self.mod.get("ok"))
|
||||
self.assertEqual("default", self.mod.get("none", "default"))
|
||||
self.assertRaises(KeyError, self.mod.get, "none", None)
|
||||
self.assertListEqual(
|
||||
['TRUE=yes\n', "FALSE=no\n", "OK=ok\n", "LIST=item1\n", "LIST=item2\n"],
|
||||
self.mod.map_param(true="TRUE", false="FALSE", ok="OK", list="LIST"),
|
||||
)
|
||||
self.assertRaises(TypeError, self.mod.diff, diff=dict(), before=True, after=False)
|
||||
self.mod.diff(before="a", after="b", before_header="a", after_header="b")
|
||||
self.mod.diff(dict(before="c", after="d"))
|
||||
|
||||
def test_result(self):
|
||||
self.assertFalse(self.mod.changed)
|
||||
self.mod.changed = 5
|
||||
self.assertTrue(self.mod.changed)
|
||||
self.mod.set("key", "value")
|
||||
self.assertDictEqual(
|
||||
dict(
|
||||
changed=True,
|
||||
key="value",
|
||||
),
|
||||
self.mod.result,
|
||||
)
|
||||
|
||||
|
||||
class TestFunctions(unittest.TestCase):
|
||||
def test_docify(self):
|
||||
input = dict(
|
||||
item1=dict(type="str", description=""),
|
||||
item2=dict(required=True, type="int", default=5),
|
||||
item3=dict(type="list", required=False, elements="int"),
|
||||
item4=dict(type="list", required=True, elements="int"),
|
||||
item5=dict(type="str", choices=["a", "b", "c"]),
|
||||
item6=dict(type="dict", options=dict(a=dict(type="str"))),
|
||||
)
|
||||
expected = dict(
|
||||
item1=dict(required=False, type="str", description=[""]),
|
||||
item2=dict(required=True, type="int", default=5),
|
||||
item3=dict(type="list", required=False, elements="int", default=[]),
|
||||
item4=dict(type="list", required=True, elements="int"),
|
||||
item5=dict(type="str", required=False, choices=("a", "b", "c")),
|
||||
item6=dict(type="dict", required=False, options=dict(a=dict(type="str", required=False))),
|
||||
)
|
||||
output = docify(input)
|
||||
for key, value in input.items():
|
||||
self.assertDictEqual(expected[key], output[key])
|
7
upload.sh
Ausführbare Datei
7
upload.sh
Ausführbare Datei
|
@ -0,0 +1,7 @@
|
|||
#!/bin/bash
|
||||
user=$(yq -r .namespace galaxy.yml)
|
||||
package=$(yq -r .name galaxy.yml)
|
||||
version=$(yq -r .version galaxy.yml)
|
||||
printf "Namespace: %s\nPackage: %s\nVersion: %s\n" $user $package $version
|
||||
ansible-galaxy collection publish dist/galaxy/*
|
||||
hatch publish -r ansible dist/python/*
|
Laden …
Tabelle hinzufügen
In neuem Issue referenzieren