feat(jenkins_node_info): add info module
This commit is contained in:
parent
ab4b7bafe7
commit
76dbcdabff
46
plugins/module_utils/Jenkins.py
Normal file
46
plugins/module_utils/Jenkins.py
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
import xml.etree.ElementTree as ET
|
||||||
|
from typing import Optional
|
||||||
|
from dataclasses import dataclass
|
||||||
|
|
||||||
|
import requests
|
||||||
|
from pathspec import PathSpec
|
||||||
|
|
||||||
|
|
||||||
|
class Jenkins:
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class NodeInfo:
|
||||||
|
server_url: str
|
||||||
|
name: str
|
||||||
|
secret: Optional[str]
|
||||||
|
work_dir: PathSpec
|
||||||
|
internal_dir: PathSpec
|
||||||
|
server_url: str
|
||||||
|
|
||||||
|
def __init__(self, server_url, username, api_token):
|
||||||
|
self.server_url = server_url
|
||||||
|
self.username = username
|
||||||
|
self.api_token = api_token
|
||||||
|
|
||||||
|
def _log_in(self, username: str, password: str) -> (str, str):
|
||||||
|
response = requests.get(f"{self.server_url}/crumbIssuer/api/json")
|
||||||
|
response.raise_for_status()
|
||||||
|
payload = response.json()
|
||||||
|
return payload["crumbRequestField"], payload["crumb"]
|
||||||
|
|
||||||
|
def get_node_jnlp(self, node_name) -> str:
|
||||||
|
response = requests.get(
|
||||||
|
f"{self.server_url}/manage/computer/{node_name}/slave-agent.jnlp",
|
||||||
|
auth=(self.username, self.api_token),
|
||||||
|
)
|
||||||
|
response.raise_for_status()
|
||||||
|
return response.text
|
||||||
|
|
||||||
|
def get_node_info(self, node_name: str) -> NodeInfo:
|
||||||
|
jnlp_info_raw = self.get_node_jnlp(node_name)
|
||||||
|
tree = ET.ElementTree(ET.fromstring(jnlp_info_raw))
|
||||||
|
arguments = tree.findall("./application-desc/")
|
||||||
|
(node_secret, node_name, _, work_dir, _, internal_dir, _, url) = [
|
||||||
|
arg.text for arg in arguments[:7]
|
||||||
|
]
|
||||||
|
return Jenkins.NodeInfo(url, node_name, node_secret, work_dir, internal_dir)
|
0
plugins/modules/jenkins_node.py
Normal file
0
plugins/modules/jenkins_node.py
Normal file
110
plugins/modules/jenkins_node_info.py
Normal file
110
plugins/modules/jenkins_node_info.py
Normal file
@ -0,0 +1,110 @@
|
|||||||
|
# pylint: disable=E0401
|
||||||
|
|
||||||
|
from __future__ import absolute_import, annotations, division, print_function
|
||||||
|
|
||||||
|
__metaclass__ = type # pylint: disable=C0103
|
||||||
|
|
||||||
|
from typing import TYPE_CHECKING
|
||||||
|
|
||||||
|
from ansible.module_utils.basic import AnsibleModule
|
||||||
|
|
||||||
|
from ansible_collections.finallycoffee.cicd.plugins.module_utils.Jenkins import Jenkins
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from typing import Optional, Dict, Any
|
||||||
|
|
||||||
|
DOCUMENTATION = r"""
|
||||||
|
---
|
||||||
|
module: jenkins_node
|
||||||
|
|
||||||
|
short_description: Retrieve Jenkins node information
|
||||||
|
# If this is part of a collection, you need to use semantic versioning,
|
||||||
|
# i.e. the version is of the form "2.5.0" and not "2.4".
|
||||||
|
version_added: "0.0.1"
|
||||||
|
|
||||||
|
description: This is my longer description explaining my test module.
|
||||||
|
|
||||||
|
options:
|
||||||
|
name:
|
||||||
|
description: The name of the jenkins node.
|
||||||
|
required: true
|
||||||
|
type: str
|
||||||
|
server:
|
||||||
|
description: URL of the jenkins instance
|
||||||
|
required: true
|
||||||
|
type: str
|
||||||
|
api_token:
|
||||||
|
description: Jenkins API token
|
||||||
|
required: true
|
||||||
|
type: str
|
||||||
|
author:
|
||||||
|
- transcaffeine (@transcaffeine)
|
||||||
|
"""
|
||||||
|
|
||||||
|
EXAMPLES = r"""
|
||||||
|
# Pass in a message
|
||||||
|
- name: Test with a message
|
||||||
|
my_namespace.my_collection.my_test:
|
||||||
|
name: hello world
|
||||||
|
|
||||||
|
# pass in a message and have changed true
|
||||||
|
- name: Test with a message and changed output
|
||||||
|
my_namespace.my_collection.my_test:
|
||||||
|
name: hello world
|
||||||
|
new: true
|
||||||
|
|
||||||
|
# fail the module
|
||||||
|
- name: Test failure of the module
|
||||||
|
my_namespace.my_collection.my_test:
|
||||||
|
name: fail me
|
||||||
|
"""
|
||||||
|
|
||||||
|
RETURN = r"""
|
||||||
|
# These are examples of possible return values, and in general should use other names for return values.
|
||||||
|
name:
|
||||||
|
description: The name of the jenkins node
|
||||||
|
type: str
|
||||||
|
returned: always
|
||||||
|
sample: 'jenkins-agent-jdk21-alpine'
|
||||||
|
secret:
|
||||||
|
description: The secret of the agent
|
||||||
|
type: str
|
||||||
|
returned: always
|
||||||
|
sample: 'secretverylongstringwith64chars'
|
||||||
|
work_dir:
|
||||||
|
description: The local working directory of the jenkins agent
|
||||||
|
type: str
|
||||||
|
returned: always
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
def run_module():
|
||||||
|
module_args = dict(
|
||||||
|
name=dict(type="str", required=True),
|
||||||
|
server=dict(type="str", required=True),
|
||||||
|
username=dict(type="str", required=True),
|
||||||
|
api_token=dict(type="str", required=True),
|
||||||
|
)
|
||||||
|
result = dict(changed=False, original_message="", message="")
|
||||||
|
module = AnsibleModule(argument_spec=module_args, supports_check_mode=True)
|
||||||
|
|
||||||
|
jenkins = Jenkins(
|
||||||
|
module.params["server"], module.params["username"], module.params["api_token"]
|
||||||
|
)
|
||||||
|
node = jenkins.get_node_info(module.params["name"])
|
||||||
|
|
||||||
|
result["name"] = node.name
|
||||||
|
result["secret"] = node.secret
|
||||||
|
result["work_dir"] = node.work_dir
|
||||||
|
|
||||||
|
# in the event of a successful module execution, you will want to
|
||||||
|
# simple AnsibleModule.exit_json(), passing the key/value results
|
||||||
|
module.exit_json(**result)
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
run_module()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
1
requirements-dev.txt
Normal file
1
requirements-dev.txt
Normal file
@ -0,0 +1 @@
|
|||||||
|
ansible-dev-tools==25.*
|
91
requirements.txt
Normal file
91
requirements.txt
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
alabaster==1.0.0
|
||||||
|
annotated-types==0.7.0
|
||||||
|
ansible==11.2.0
|
||||||
|
ansible-core==2.18.2
|
||||||
|
argcomplete==3.4.0
|
||||||
|
attrs==23.2.1.dev0
|
||||||
|
autocommand==2.2.2
|
||||||
|
Babel==2.15.0
|
||||||
|
Brlapi==0.8.6
|
||||||
|
btrfsutil==6.12
|
||||||
|
build==1.2.2
|
||||||
|
canonicaljson==2.0.0
|
||||||
|
certifi==2025.1.31
|
||||||
|
cffi==1.17.1
|
||||||
|
charset-normalizer==3.4.1
|
||||||
|
coverage==7.6.10
|
||||||
|
cryptography==44.0.0
|
||||||
|
dbus-python==1.3.2
|
||||||
|
diceware==1.0.1
|
||||||
|
distro==1.9.0
|
||||||
|
distro-info==1.10
|
||||||
|
dnspython==2.7.0
|
||||||
|
docutils==0.21.2
|
||||||
|
fastjsonschema==2.21.1
|
||||||
|
filelock==3.17.0
|
||||||
|
idna==3.10
|
||||||
|
imagesize==1.4.1
|
||||||
|
inflect==7.5.0
|
||||||
|
iniconfig==2.0.0
|
||||||
|
installer==0.7.0
|
||||||
|
jaraco.collections==5.1.0
|
||||||
|
jaraco.context==6.0.1
|
||||||
|
jaraco.functools==4.1.0
|
||||||
|
jaraco.text==4.0.0
|
||||||
|
Jinja2==3.1.5
|
||||||
|
lensfun==0.3.4
|
||||||
|
libfdt==1.7.2
|
||||||
|
libvirt-python==11.0.0
|
||||||
|
louis==3.32.0
|
||||||
|
lxml==5.3.0
|
||||||
|
MarkupSafe==2.1.5
|
||||||
|
more-itertools==10.5.0
|
||||||
|
netsnmp-python==1.0a1
|
||||||
|
nftables==0.1
|
||||||
|
ordered-set==4.1.0
|
||||||
|
packaging==24.2
|
||||||
|
platformdirs==4.3.6
|
||||||
|
pluggy==1.5.0
|
||||||
|
ply==3.11
|
||||||
|
pycairo==1.27.0
|
||||||
|
pycparser==2.22
|
||||||
|
pydantic==2.10.6
|
||||||
|
pydantic_core==2.27.2
|
||||||
|
Pygments==2.19.1
|
||||||
|
PyGObject==3.50.0
|
||||||
|
PyNaCl==1.5.0
|
||||||
|
pyproject_hooks==1.2.0
|
||||||
|
pytest==8.3.4
|
||||||
|
pytest-cov==6.0.0
|
||||||
|
pytz==2024.2
|
||||||
|
PyYAML==6.0.2
|
||||||
|
requests==2.32.3
|
||||||
|
resolvelib==1.0.1
|
||||||
|
setuptools==75.2.0
|
||||||
|
signedjson==1.1.4
|
||||||
|
six==1.17.0
|
||||||
|
snowballstemmer==2.2.0
|
||||||
|
Sphinx==8.1.3
|
||||||
|
sphinx_rtd_theme==2.0.0
|
||||||
|
sphinxcontrib-applehelp==2.0.0
|
||||||
|
sphinxcontrib-devhelp==2.0.0
|
||||||
|
sphinxcontrib-htmlhelp==2.1.0
|
||||||
|
sphinxcontrib-jquery==4.1
|
||||||
|
sphinxcontrib-jsmath==1.0.1
|
||||||
|
sphinxcontrib-qthelp==2.0.0
|
||||||
|
sphinxcontrib-serializinghtml==2.0.0
|
||||||
|
TBB==0.2
|
||||||
|
tomli==2.0.1
|
||||||
|
tomlkit==0.13.2
|
||||||
|
trove-classifiers==2025.1.15.22
|
||||||
|
typeguard==4.4.1
|
||||||
|
typing_extensions==4.12.2
|
||||||
|
ufw==0.36.2
|
||||||
|
unpaddedbase64==2.1.0
|
||||||
|
urllib3==2.3.0
|
||||||
|
validate-pyproject==0.23.post1.dev0+gf45606b.d20250111
|
||||||
|
wheel==0.45.0
|
||||||
|
wxPython==4.2.2
|
||||||
|
xmltodict==0.14.2
|
||||||
|
yq==3.4.3
|
||||||
|
yt-dlp==2025.1.26
|
Loading…
x
Reference in New Issue
Block a user