2020-03-09 09:11:07 +00:00
|
|
|
#!/usr/bin/python
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2022-08-08 14:24:58 +02:00
|
|
|
# Copyright (c) 2015, Alejandro Guirao <lekumberri@gmail.com>
|
2022-08-05 12:28:29 +02:00
|
|
|
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
|
|
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
2020-03-09 09:11:07 +00:00
|
|
|
|
|
|
|
from __future__ import absolute_import, division, print_function
|
|
|
|
__metaclass__ = type
|
|
|
|
|
|
|
|
|
|
|
|
DOCUMENTATION = '''
|
|
|
|
---
|
|
|
|
module: taiga_issue
|
|
|
|
short_description: Creates/deletes an issue in a Taiga Project Management Platform
|
|
|
|
description:
|
|
|
|
- Creates/deletes an issue in a Taiga Project Management Platform (U(https://taiga.io)).
|
|
|
|
- An issue is identified by the combination of project, issue subject and issue type.
|
|
|
|
- This module implements the creation or deletion of issues (not the update).
|
2023-02-24 09:24:50 +01:00
|
|
|
extends_documentation_fragment:
|
|
|
|
- community.general.attributes
|
|
|
|
attributes:
|
|
|
|
check_mode:
|
|
|
|
support: full
|
|
|
|
diff_mode:
|
|
|
|
support: none
|
2020-03-09 09:11:07 +00:00
|
|
|
options:
|
|
|
|
taiga_host:
|
2020-11-01 01:53:57 +13:00
|
|
|
type: str
|
2020-03-09 09:11:07 +00:00
|
|
|
description:
|
|
|
|
- The hostname of the Taiga instance.
|
|
|
|
default: https://api.taiga.io
|
|
|
|
project:
|
2020-11-01 01:53:57 +13:00
|
|
|
type: str
|
2020-03-09 09:11:07 +00:00
|
|
|
description:
|
|
|
|
- Name of the project containing the issue. Must exist previously.
|
2022-09-06 20:42:17 +02:00
|
|
|
required: true
|
2020-03-09 09:11:07 +00:00
|
|
|
subject:
|
2020-11-01 01:53:57 +13:00
|
|
|
type: str
|
2020-03-09 09:11:07 +00:00
|
|
|
description:
|
|
|
|
- The issue subject.
|
2022-09-06 20:42:17 +02:00
|
|
|
required: true
|
2020-03-09 09:11:07 +00:00
|
|
|
issue_type:
|
2020-11-01 01:53:57 +13:00
|
|
|
type: str
|
2020-03-09 09:11:07 +00:00
|
|
|
description:
|
|
|
|
- The issue type. Must exist previously.
|
2022-09-06 20:42:17 +02:00
|
|
|
required: true
|
2020-03-09 09:11:07 +00:00
|
|
|
priority:
|
2020-11-01 01:53:57 +13:00
|
|
|
type: str
|
2020-03-09 09:11:07 +00:00
|
|
|
description:
|
|
|
|
- The issue priority. Must exist previously.
|
|
|
|
default: Normal
|
|
|
|
status:
|
2020-11-01 01:53:57 +13:00
|
|
|
type: str
|
2020-03-09 09:11:07 +00:00
|
|
|
description:
|
|
|
|
- The issue status. Must exist previously.
|
|
|
|
default: New
|
|
|
|
severity:
|
2020-11-01 01:53:57 +13:00
|
|
|
type: str
|
2020-03-09 09:11:07 +00:00
|
|
|
description:
|
|
|
|
- The issue severity. Must exist previously.
|
|
|
|
default: Normal
|
|
|
|
description:
|
2020-11-01 01:53:57 +13:00
|
|
|
type: str
|
2020-03-09 09:11:07 +00:00
|
|
|
description:
|
|
|
|
- The issue description.
|
|
|
|
default: ""
|
|
|
|
attachment:
|
2020-11-01 01:53:57 +13:00
|
|
|
type: path
|
2020-03-09 09:11:07 +00:00
|
|
|
description:
|
|
|
|
- Path to a file to be attached to the issue.
|
|
|
|
attachment_description:
|
2020-11-01 01:53:57 +13:00
|
|
|
type: str
|
2020-03-09 09:11:07 +00:00
|
|
|
description:
|
|
|
|
- A string describing the file to be attached to the issue.
|
|
|
|
default: ""
|
|
|
|
tags:
|
2020-11-01 01:53:57 +13:00
|
|
|
type: list
|
|
|
|
elements: str
|
2020-03-09 09:11:07 +00:00
|
|
|
description:
|
|
|
|
- A lists of tags to be assigned to the issue.
|
|
|
|
default: []
|
|
|
|
state:
|
2020-11-01 01:53:57 +13:00
|
|
|
type: str
|
2020-03-09 09:11:07 +00:00
|
|
|
description:
|
|
|
|
- Whether the issue should be present or not.
|
|
|
|
choices: ["present", "absent"]
|
|
|
|
default: present
|
|
|
|
author: Alejandro Guirao (@lekum)
|
|
|
|
requirements: [python-taiga]
|
|
|
|
notes:
|
|
|
|
- The authentication is achieved either by the environment variable TAIGA_TOKEN or by the pair of environment variables TAIGA_USERNAME and TAIGA_PASSWORD
|
|
|
|
'''
|
|
|
|
|
|
|
|
EXAMPLES = '''
|
2020-05-15 13:27:06 +03:00
|
|
|
- name: Create an issue in the my hosted Taiga environment and attach an error log
|
2020-07-13 22:50:31 +03:00
|
|
|
community.general.taiga_issue:
|
2020-03-09 09:11:07 +00:00
|
|
|
taiga_host: https://mytaigahost.example.com
|
|
|
|
project: myproject
|
|
|
|
subject: An error has been found
|
|
|
|
issue_type: Bug
|
|
|
|
priority: High
|
|
|
|
status: New
|
|
|
|
severity: Important
|
|
|
|
description: An error has been found. Please check the attached error log for details.
|
|
|
|
attachment: /path/to/error.log
|
|
|
|
attachment_description: Error log file
|
|
|
|
tags:
|
|
|
|
- Error
|
|
|
|
- Needs manual check
|
|
|
|
state: present
|
|
|
|
|
2020-05-15 13:27:06 +03:00
|
|
|
- name: Deletes the previously created issue
|
2020-07-13 22:50:31 +03:00
|
|
|
community.general.taiga_issue:
|
2020-03-09 09:11:07 +00:00
|
|
|
taiga_host: https://mytaigahost.example.com
|
|
|
|
project: myproject
|
|
|
|
subject: An error has been found
|
|
|
|
issue_type: Bug
|
|
|
|
state: absent
|
|
|
|
'''
|
|
|
|
|
|
|
|
RETURN = '''# '''
|
|
|
|
import traceback
|
|
|
|
|
|
|
|
from os import getenv
|
|
|
|
from os.path import isfile
|
|
|
|
from ansible.module_utils.basic import AnsibleModule, missing_required_lib
|
2021-06-26 23:59:11 +02:00
|
|
|
from ansible.module_utils.common.text.converters import to_native
|
2020-03-09 09:11:07 +00:00
|
|
|
|
|
|
|
TAIGA_IMP_ERR = None
|
|
|
|
try:
|
|
|
|
from taiga import TaigaAPI
|
|
|
|
from taiga.exceptions import TaigaException
|
|
|
|
TAIGA_MODULE_IMPORTED = True
|
|
|
|
except ImportError:
|
|
|
|
TAIGA_IMP_ERR = traceback.format_exc()
|
|
|
|
TAIGA_MODULE_IMPORTED = False
|
|
|
|
|
|
|
|
|
2021-07-26 08:03:45 +12:00
|
|
|
def manage_issue(taiga_host, project_name, issue_subject, issue_priority,
|
2020-03-09 09:11:07 +00:00
|
|
|
issue_status, issue_type, issue_severity, issue_description,
|
|
|
|
issue_attachment, issue_attachment_description,
|
|
|
|
issue_tags, state, check_mode=False):
|
|
|
|
"""
|
|
|
|
Method that creates/deletes issues depending whether they exist and the state desired
|
|
|
|
|
|
|
|
The credentials should be passed via environment variables:
|
|
|
|
- TAIGA_TOKEN
|
|
|
|
- TAIGA_USERNAME and TAIGA_PASSWORD
|
|
|
|
|
|
|
|
Returns a tuple with these elements:
|
|
|
|
- A boolean representing the success of the operation
|
|
|
|
- A descriptive message
|
|
|
|
- A dict with the issue attributes, in case of issue creation, otherwise empty dict
|
|
|
|
"""
|
|
|
|
|
|
|
|
changed = False
|
|
|
|
|
|
|
|
try:
|
|
|
|
token = getenv('TAIGA_TOKEN')
|
|
|
|
if token:
|
|
|
|
api = TaigaAPI(host=taiga_host, token=token)
|
|
|
|
else:
|
|
|
|
api = TaigaAPI(host=taiga_host)
|
|
|
|
username = getenv('TAIGA_USERNAME')
|
|
|
|
password = getenv('TAIGA_PASSWORD')
|
|
|
|
if not any([username, password]):
|
2021-07-26 08:03:45 +12:00
|
|
|
return False, changed, "Missing credentials", {}
|
2020-03-09 09:11:07 +00:00
|
|
|
api.auth(username=username, password=password)
|
|
|
|
|
|
|
|
user_id = api.me().id
|
2021-07-26 08:03:45 +12:00
|
|
|
project_list = list(filter(lambda x: x.name == project_name, api.projects.list(member=user_id)))
|
2020-03-09 09:11:07 +00:00
|
|
|
if len(project_list) != 1:
|
2021-07-26 08:03:45 +12:00
|
|
|
return False, changed, "Unable to find project %s" % project_name, {}
|
2020-03-09 09:11:07 +00:00
|
|
|
project = project_list[0]
|
|
|
|
project_id = project.id
|
|
|
|
|
2021-07-26 08:03:45 +12:00
|
|
|
priority_list = list(filter(lambda x: x.name == issue_priority, api.priorities.list(project=project_id)))
|
2020-03-09 09:11:07 +00:00
|
|
|
if len(priority_list) != 1:
|
2021-07-26 08:03:45 +12:00
|
|
|
return False, changed, "Unable to find issue priority %s for project %s" % (issue_priority, project_name), {}
|
2020-03-09 09:11:07 +00:00
|
|
|
priority_id = priority_list[0].id
|
|
|
|
|
2021-07-26 08:03:45 +12:00
|
|
|
status_list = list(filter(lambda x: x.name == issue_status, api.issue_statuses.list(project=project_id)))
|
2020-03-09 09:11:07 +00:00
|
|
|
if len(status_list) != 1:
|
2021-07-26 08:03:45 +12:00
|
|
|
return False, changed, "Unable to find issue status %s for project %s" % (issue_status, project_name), {}
|
2020-03-09 09:11:07 +00:00
|
|
|
status_id = status_list[0].id
|
|
|
|
|
2021-07-26 08:03:45 +12:00
|
|
|
type_list = list(filter(lambda x: x.name == issue_type, project.list_issue_types()))
|
2020-03-09 09:11:07 +00:00
|
|
|
if len(type_list) != 1:
|
2021-07-26 08:03:45 +12:00
|
|
|
return False, changed, "Unable to find issue type %s for project %s" % (issue_type, project_name), {}
|
2020-03-09 09:11:07 +00:00
|
|
|
type_id = type_list[0].id
|
|
|
|
|
2021-07-26 08:03:45 +12:00
|
|
|
severity_list = list(filter(lambda x: x.name == issue_severity, project.list_severities()))
|
2020-03-09 09:11:07 +00:00
|
|
|
if len(severity_list) != 1:
|
2021-07-26 08:03:45 +12:00
|
|
|
return False, changed, "Unable to find severity %s for project %s" % (issue_severity, project_name), {}
|
2020-03-09 09:11:07 +00:00
|
|
|
severity_id = severity_list[0].id
|
|
|
|
|
|
|
|
issue = {
|
|
|
|
"project": project_name,
|
|
|
|
"subject": issue_subject,
|
|
|
|
"priority": issue_priority,
|
|
|
|
"status": issue_status,
|
|
|
|
"type": issue_type,
|
|
|
|
"severity": issue_severity,
|
|
|
|
"description": issue_description,
|
|
|
|
"tags": issue_tags,
|
|
|
|
}
|
|
|
|
|
|
|
|
# An issue is identified by the project_name, the issue_subject and the issue_type
|
2021-07-26 08:03:45 +12:00
|
|
|
matching_issue_list = list(filter(lambda x: x.subject == issue_subject and x.type == type_id, project.list_issues()))
|
2020-03-09 09:11:07 +00:00
|
|
|
matching_issue_list_len = len(matching_issue_list)
|
|
|
|
|
|
|
|
if matching_issue_list_len == 0:
|
|
|
|
# The issue does not exist in the project
|
|
|
|
if state == "present":
|
|
|
|
# This implies a change
|
|
|
|
changed = True
|
|
|
|
if not check_mode:
|
|
|
|
# Create the issue
|
2021-07-26 08:03:45 +12:00
|
|
|
new_issue = project.add_issue(issue_subject, priority_id, status_id, type_id, severity_id, tags=issue_tags,
|
|
|
|
description=issue_description)
|
2020-03-09 09:11:07 +00:00
|
|
|
if issue_attachment:
|
|
|
|
new_issue.attach(issue_attachment, description=issue_attachment_description)
|
|
|
|
issue["attachment"] = issue_attachment
|
|
|
|
issue["attachment_description"] = issue_attachment_description
|
2021-07-26 08:03:45 +12:00
|
|
|
return True, changed, "Issue created", issue
|
2020-03-09 09:11:07 +00:00
|
|
|
|
|
|
|
else:
|
|
|
|
# If does not exist, do nothing
|
2021-07-26 08:03:45 +12:00
|
|
|
return True, changed, "Issue does not exist", {}
|
2020-03-09 09:11:07 +00:00
|
|
|
|
|
|
|
elif matching_issue_list_len == 1:
|
|
|
|
# The issue exists in the project
|
|
|
|
if state == "absent":
|
|
|
|
# This implies a change
|
|
|
|
changed = True
|
|
|
|
if not check_mode:
|
|
|
|
# Delete the issue
|
|
|
|
matching_issue_list[0].delete()
|
2021-07-26 08:03:45 +12:00
|
|
|
return True, changed, "Issue deleted", {}
|
2020-03-09 09:11:07 +00:00
|
|
|
|
|
|
|
else:
|
|
|
|
# Do nothing
|
2021-07-26 08:03:45 +12:00
|
|
|
return True, changed, "Issue already exists", {}
|
2020-03-09 09:11:07 +00:00
|
|
|
|
|
|
|
else:
|
|
|
|
# More than 1 matching issue
|
2021-07-26 08:03:45 +12:00
|
|
|
return False, changed, "More than one issue with subject %s in project %s" % (issue_subject, project_name), {}
|
2020-03-09 09:11:07 +00:00
|
|
|
|
|
|
|
except TaigaException as exc:
|
|
|
|
msg = "An exception happened: %s" % to_native(exc)
|
2021-07-26 08:03:45 +12:00
|
|
|
return False, changed, msg, {}
|
2020-03-09 09:11:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
module = AnsibleModule(
|
|
|
|
argument_spec=dict(
|
2020-11-01 01:53:57 +13:00
|
|
|
taiga_host=dict(type='str', required=False, default="https://api.taiga.io"),
|
|
|
|
project=dict(type='str', required=True),
|
|
|
|
subject=dict(type='str', required=True),
|
|
|
|
issue_type=dict(type='str', required=True),
|
|
|
|
priority=dict(type='str', required=False, default="Normal"),
|
|
|
|
status=dict(type='str', required=False, default="New"),
|
|
|
|
severity=dict(type='str', required=False, default="Normal"),
|
|
|
|
description=dict(type='str', required=False, default=""),
|
|
|
|
attachment=dict(type='path', required=False, default=None),
|
|
|
|
attachment_description=dict(type='str', required=False, default=""),
|
|
|
|
tags=dict(required=False, default=[], type='list', elements='str'),
|
2021-07-26 08:03:45 +12:00
|
|
|
state=dict(type='str', required=False, choices=['present', 'absent'], default='present'),
|
2020-03-09 09:11:07 +00:00
|
|
|
),
|
|
|
|
supports_check_mode=True
|
|
|
|
)
|
|
|
|
|
|
|
|
if not TAIGA_MODULE_IMPORTED:
|
2021-07-26 08:03:45 +12:00
|
|
|
module.fail_json(msg=missing_required_lib("python-taiga"), exception=TAIGA_IMP_ERR)
|
2020-03-09 09:11:07 +00:00
|
|
|
|
|
|
|
taiga_host = module.params['taiga_host']
|
|
|
|
project_name = module.params['project']
|
|
|
|
issue_subject = module.params['subject']
|
|
|
|
issue_priority = module.params['priority']
|
|
|
|
issue_status = module.params['status']
|
|
|
|
issue_type = module.params['issue_type']
|
|
|
|
issue_severity = module.params['severity']
|
|
|
|
issue_description = module.params['description']
|
|
|
|
issue_attachment = module.params['attachment']
|
|
|
|
issue_attachment_description = module.params['attachment_description']
|
|
|
|
if issue_attachment:
|
|
|
|
if not isfile(issue_attachment):
|
|
|
|
msg = "%s is not a file" % issue_attachment
|
|
|
|
module.fail_json(msg=msg)
|
|
|
|
issue_tags = module.params['tags']
|
|
|
|
state = module.params['state']
|
|
|
|
|
|
|
|
return_status, changed, msg, issue_attr_dict = manage_issue(
|
|
|
|
taiga_host,
|
|
|
|
project_name,
|
|
|
|
issue_subject,
|
|
|
|
issue_priority,
|
|
|
|
issue_status,
|
|
|
|
issue_type,
|
|
|
|
issue_severity,
|
|
|
|
issue_description,
|
|
|
|
issue_attachment,
|
|
|
|
issue_attachment_description,
|
|
|
|
issue_tags,
|
|
|
|
state,
|
|
|
|
check_mode=module.check_mode
|
|
|
|
)
|
|
|
|
if return_status:
|
2021-07-26 08:03:45 +12:00
|
|
|
if issue_attr_dict:
|
2020-03-09 09:11:07 +00:00
|
|
|
module.exit_json(changed=changed, msg=msg, issue=issue_attr_dict)
|
|
|
|
else:
|
|
|
|
module.exit_json(changed=changed, msg=msg)
|
|
|
|
else:
|
|
|
|
module.fail_json(msg=msg)
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|