initiele commit
This commit is contained in:
160
.gitignore
vendored
Normal file
160
.gitignore
vendored
Normal file
@ -0,0 +1,160 @@
|
||||
# Byte-compiled / optimized / DLL files
|
||||
__pycache__/
|
||||
*.py[cod]
|
||||
*$py.class
|
||||
|
||||
# C extensions
|
||||
*.so
|
||||
|
||||
# Distribution / packaging
|
||||
.Python
|
||||
build/
|
||||
develop-eggs/
|
||||
dist/
|
||||
downloads/
|
||||
eggs/
|
||||
.eggs/
|
||||
lib/
|
||||
lib64/
|
||||
parts/
|
||||
sdist/
|
||||
var/
|
||||
wheels/
|
||||
share/python-wheels/
|
||||
*.egg-info/
|
||||
.installed.cfg
|
||||
*.egg
|
||||
MANIFEST
|
||||
|
||||
# PyInstaller
|
||||
# Usually these files are written by a python script from a template
|
||||
# before PyInstaller builds the exe, so as to inject date/other infos into it.
|
||||
*.manifest
|
||||
*.spec
|
||||
|
||||
# Installer logs
|
||||
pip-log.txt
|
||||
pip-delete-this-directory.txt
|
||||
|
||||
# Unit test / coverage reports
|
||||
htmlcov/
|
||||
.tox/
|
||||
.nox/
|
||||
.coverage
|
||||
.coverage.*
|
||||
.cache
|
||||
nosetests.xml
|
||||
coverage.xml
|
||||
*.cover
|
||||
*.py,cover
|
||||
.hypothesis/
|
||||
.pytest_cache/
|
||||
cover/
|
||||
|
||||
# Translations
|
||||
*.mo
|
||||
*.pot
|
||||
|
||||
# Django stuff:
|
||||
*.log
|
||||
local_settings.py
|
||||
db.sqlite3
|
||||
db.sqlite3-journal
|
||||
|
||||
# Flask stuff:
|
||||
instance/
|
||||
.webassets-cache
|
||||
|
||||
# Scrapy stuff:
|
||||
.scrapy
|
||||
|
||||
# Sphinx documentation
|
||||
docs/_build/
|
||||
|
||||
# PyBuilder
|
||||
.pybuilder/
|
||||
target/
|
||||
|
||||
# Jupyter Notebook
|
||||
.ipynb_checkpoints
|
||||
|
||||
# IPython
|
||||
profile_default/
|
||||
ipython_config.py
|
||||
|
||||
# pyenv
|
||||
# For a library or package, you might want to ignore these files since the code is
|
||||
# intended to run in multiple environments; otherwise, check them in:
|
||||
# .python-version
|
||||
|
||||
# pipenv
|
||||
# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control.
|
||||
# However, in case of collaboration, if having platform-specific dependencies or dependencies
|
||||
# having no cross-platform support, pipenv may install dependencies that don't work, or not
|
||||
# install all needed dependencies.
|
||||
#Pipfile.lock
|
||||
|
||||
# poetry
|
||||
# Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control.
|
||||
# This is especially recommended for binary packages to ensure reproducibility, and is more
|
||||
# commonly ignored for libraries.
|
||||
# https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control
|
||||
#poetry.lock
|
||||
|
||||
# pdm
|
||||
# Similar to Pipfile.lock, it is generally recommended to include pdm.lock in version control.
|
||||
#pdm.lock
|
||||
# pdm stores project-wide configurations in .pdm.toml, but it is recommended to not include it
|
||||
# in version control.
|
||||
# https://pdm.fming.dev/#use-with-ide
|
||||
.pdm.toml
|
||||
|
||||
# PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm
|
||||
__pypackages__/
|
||||
|
||||
# Celery stuff
|
||||
celerybeat-schedule
|
||||
celerybeat.pid
|
||||
|
||||
# SageMath parsed files
|
||||
*.sage.py
|
||||
|
||||
# Environments
|
||||
.env
|
||||
.venv
|
||||
env/
|
||||
venv/
|
||||
ENV/
|
||||
env.bak/
|
||||
venv.bak/
|
||||
|
||||
# Spyder project settings
|
||||
.spyderproject
|
||||
.spyproject
|
||||
|
||||
# Rope project settings
|
||||
.ropeproject
|
||||
|
||||
# mkdocs documentation
|
||||
/site
|
||||
|
||||
# mypy
|
||||
.mypy_cache/
|
||||
.dmypy.json
|
||||
dmypy.json
|
||||
|
||||
# Pyre type checker
|
||||
.pyre/
|
||||
|
||||
# pytype static type analyzer
|
||||
.pytype/
|
||||
|
||||
# Cython debug symbols
|
||||
cython_debug/
|
||||
|
||||
# PyCharm
|
||||
# JetBrains specific template is maintained in a separate JetBrains.gitignore that can
|
||||
# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore
|
||||
# and can be added to the global gitignore or merged into this file. For a more nuclear
|
||||
# option (not recommended) you can uncomment the following to ignore the entire idea folder.
|
||||
#.idea/
|
||||
3
README.md
Normal file
3
README.md
Normal file
@ -0,0 +1,3 @@
|
||||
# Ansible Collection - kembit.topdesk
|
||||
|
||||
Documentation for the collection.
|
||||
62
galaxy.yml
Normal file
62
galaxy.yml
Normal file
@ -0,0 +1,62 @@
|
||||
### REQUIRED
|
||||
# The namespace of the collection. This can be a company/brand/organization or product namespace under which all
|
||||
# content lives. May only contain alphanumeric lowercase characters and underscores. Namespaces cannot start with
|
||||
# underscores or numbers and cannot contain consecutive underscores
|
||||
namespace: kembit
|
||||
|
||||
# The name of the collection. Has the same character restrictions as 'namespace'
|
||||
name: topdesk
|
||||
|
||||
# The version of the collection. Must be compatible with semantic versioning
|
||||
version: 1.0.0
|
||||
|
||||
# The path to the Markdown (.md) readme file. This path is relative to the root of the collection
|
||||
readme: README.md
|
||||
|
||||
# A list of the collection's content authors. Can be just the name or in the format 'Full Name <email> (url)
|
||||
# @nicks:irc/im.site#channel'
|
||||
authors:
|
||||
- Martijn Remmen <mremmen@kembit.nl>
|
||||
|
||||
|
||||
### OPTIONAL but strongly recommended
|
||||
# A short summary description of the collection
|
||||
description: Een Ansible collectie voor het integreren met TOPDesk
|
||||
|
||||
# Either a single license or a list of licenses for content inside of a collection. Ansible Galaxy currently only
|
||||
# accepts L(SPDX,https://spdx.org/licenses/) licenses. This key is mutually exclusive with 'license_file'
|
||||
license:
|
||||
- GPL-2.0-or-later
|
||||
|
||||
# The path to the license file for the collection. This path is relative to the root of the collection. This key is
|
||||
# mutually exclusive with 'license'
|
||||
license_file: ''
|
||||
|
||||
# A list of tags you want to associate with the collection for indexing/searching. A tag name has the same character
|
||||
# requirements as 'namespace' and 'name'
|
||||
tags: []
|
||||
|
||||
# Collections that this collection requires to be installed for it to be usable. The key of the dict is the
|
||||
# collection label 'namespace.name'. The value is a version range
|
||||
# L(specifiers,https://python-semanticversion.readthedocs.io/en/latest/#requirement-specification). Multiple version
|
||||
# range specifiers can be set and are separated by ','
|
||||
dependencies: {}
|
||||
|
||||
# The URL of the originating SCM repository
|
||||
repository: http://example.com/repository
|
||||
|
||||
# The URL to any online docs
|
||||
documentation: http://docs.example.com
|
||||
|
||||
# The URL to the homepage of the collection/project
|
||||
homepage: http://example.com
|
||||
|
||||
# The URL to the collection issue tracker
|
||||
issues: http://example.com/issue/tracker
|
||||
|
||||
# A list of file glob-like patterns used to filter any files or directories that should not be included in the build
|
||||
# artifact. A pattern is matched from the relative path of the file or directory of the collection directory. This
|
||||
# uses 'fnmatch' to match the files or directories. Some directories and files like 'galaxy.yml', '*.pyc', '*.retry',
|
||||
# and '.git' are always filtered
|
||||
build_ignore: []
|
||||
|
||||
210
plugins/inventory/topdesk_am.py
Normal file
210
plugins/inventory/topdesk_am.py
Normal file
@ -0,0 +1,210 @@
|
||||
from __future__ import (absolute_import, division, print_function)
|
||||
from binascii import Incomplete
|
||||
from email.policy import default
|
||||
__metaclass__ = type
|
||||
|
||||
DOCUMENTATION = r'''
|
||||
---
|
||||
name: topdesk_am
|
||||
short_description: Topdesk Asset Management Inventory
|
||||
author: Martijn Remmen <mremmen@kembit.nl>
|
||||
|
||||
description:
|
||||
- Constructs an inventory from TOPdesk Asset Management.
|
||||
|
||||
options:
|
||||
url:
|
||||
type: str
|
||||
required: true
|
||||
desciprion:
|
||||
- The TOPDesk url
|
||||
username:
|
||||
type: str
|
||||
required: true
|
||||
description:
|
||||
- A TOPDesk username for authenticating with the TOPDesk API
|
||||
application_key:
|
||||
type: str
|
||||
required: true
|
||||
description:
|
||||
- A TOPDesk application key associated with the given username.
|
||||
- For instructions on creating a key: https://developers.topdesk.com/tutorial.html
|
||||
fields:
|
||||
type: list
|
||||
required: true
|
||||
description:
|
||||
- A list of fields from Asset Management that you want to be included
|
||||
- with the hosts.
|
||||
names:
|
||||
type: list
|
||||
required: true
|
||||
description:
|
||||
- A list containing the unique names from all assets you want to include
|
||||
- in the inventory
|
||||
ansible_host:
|
||||
type: str
|
||||
required: true
|
||||
description:
|
||||
- The fieldname of the value to be used as ansible_host.
|
||||
- This should be the field with a reachable hostname or IP address
|
||||
groupby:
|
||||
type: list
|
||||
description:
|
||||
- Group hosts based on field(s)
|
||||
|
||||
'''
|
||||
|
||||
EXAMPLES = '''
|
||||
|
||||
'''
|
||||
|
||||
from ansible.plugins.inventory import BaseInventoryPlugin
|
||||
from ansible.utils.display import Display
|
||||
|
||||
import requests
|
||||
|
||||
display = Display()
|
||||
|
||||
|
||||
class InventoryModule(BaseInventoryPlugin):
|
||||
|
||||
NAME = 'kembit.topdesk.topdesk_am'
|
||||
|
||||
|
||||
def __init__(self):
|
||||
super(BaseInventoryPlugin, self).__init__()
|
||||
|
||||
|
||||
def verify_file(self, path):
|
||||
# return super().verify_file(path)
|
||||
return True
|
||||
|
||||
|
||||
def parse(self, inventory, loader, path, cache=True):
|
||||
super(InventoryModule, self).parse(inventory, loader, path)
|
||||
|
||||
self._read_config_data(path)
|
||||
|
||||
url = self.get_option('url')
|
||||
username = self.get_option('username')
|
||||
application_key = self.get_option('application_key')
|
||||
fields = self.get_option('fields')
|
||||
device_names = self.get_option('names')
|
||||
ansible_host_field = self.get_option('ansible_host')
|
||||
groupby = self.get_option('groupby')
|
||||
|
||||
td = Topdesk(url, username, application_key)
|
||||
|
||||
columns, devices = devices_lookup(td, device_names, fields)
|
||||
column_lookup_table = create_id_lookup_table(td, columns)
|
||||
replace_ids(devices, column_lookup_table)
|
||||
|
||||
# validgroupfields = [group for group in groupby if group in fields]
|
||||
|
||||
# groupvalues = dict(default=list)
|
||||
# for device in devices:
|
||||
# for groupfield in validgroupfields:
|
||||
# groupvalues[groupfield]
|
||||
# device[groupfield]
|
||||
|
||||
|
||||
# self.inventory.add_group(column_lookup_table[group].values())
|
||||
|
||||
|
||||
for device in devices:
|
||||
name = device['name']
|
||||
self.inventory.add_host(name)
|
||||
self.inventory.set_variable(name, 'ansible_host', device[ansible_host_field])
|
||||
for field, value in device.items():
|
||||
if field != 'name' and field in fields:
|
||||
self.inventory.set_variable(name, field, value)
|
||||
|
||||
display.display(f"Added {len(devices)} hosts")
|
||||
|
||||
|
||||
|
||||
class Topdesk:
|
||||
|
||||
def __init__(self, url: str, username: str, application_key: str) -> None:
|
||||
self.url = url
|
||||
self.api_url = url + '/tas/api'
|
||||
self._session = requests.Session()
|
||||
self._session.auth = (username, application_key)
|
||||
self._headers = {'accept': 'application/json'}
|
||||
|
||||
|
||||
def get(self, endpoint: str, **kwargs) -> requests.Request:
|
||||
return self._session.get(self.api_url + endpoint, **kwargs)
|
||||
|
||||
|
||||
|
||||
def get_asset(td: Topdesk, parameters: dict):
|
||||
return td.get('/assetmgmt/assets', params=parameters)
|
||||
|
||||
|
||||
def create_parameters(
|
||||
fields: list[str],
|
||||
name: str,
|
||||
includefunctionalities: bool = False,
|
||||
includesettings: bool = False,
|
||||
includetemplates: bool = False
|
||||
):
|
||||
return {
|
||||
'fields': ','.join(fields),
|
||||
'$filter': f'name eq {name}',
|
||||
'includeFunctionalities': str(includefunctionalities),
|
||||
'includeSettings': str(includesettings),
|
||||
'includeTemplates': str(includetemplates)
|
||||
}
|
||||
|
||||
|
||||
|
||||
def create_id_lookup_table(td: Topdesk, columns: dict) -> dict:
|
||||
table = {}
|
||||
|
||||
for columnname, columndata in columns.items():
|
||||
column_properties = columndata.get('properties')
|
||||
|
||||
if not column_properties:
|
||||
continue
|
||||
|
||||
column_url = column_properties.get('url')
|
||||
if column_url:
|
||||
data = td.get("/assetmgmt/" + column_url).json()['dataSet']
|
||||
table.update({columnname: {value['id']: value['text'] for value in data}})
|
||||
|
||||
return table
|
||||
|
||||
|
||||
|
||||
def devices_lookup(td: Topdesk, device_names: list[str], fields: list[str]):
|
||||
|
||||
columns = {}
|
||||
devices = []
|
||||
|
||||
for device in device_names:
|
||||
params = create_parameters(fields, device)
|
||||
data = get_asset(td, params).json()
|
||||
|
||||
columns.update({column['fieldName']: column for column in data['columns']})
|
||||
devices.extend(data['dataSet'])
|
||||
|
||||
return columns, devices
|
||||
|
||||
def replace_ids(devices, column_lookup_table):
|
||||
for device in devices:
|
||||
for fieldname, fieldvalue in device.items():
|
||||
if fieldname in column_lookup_table.keys():
|
||||
if fieldvalue: # soms is het None
|
||||
device[fieldname] = column_lookup_table[fieldname][fieldvalue]
|
||||
|
||||
|
||||
|
||||
def main():
|
||||
pass
|
||||
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
main()
|
||||
|
||||
1
requirements.txt
Normal file
1
requirements.txt
Normal file
@ -0,0 +1 @@
|
||||
requests>=2.27.1
|
||||
Reference in New Issue
Block a user