mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-24 21:14:00 -07:00 
			
		
		
		
	* first push: add discord module and test for notifications
* fix the yaml docs and edit the result output
* add link
* fix link
* fix docs and remove required=False in argument spec
* add elements specd and more info about embeds
* called str...
* elements for embeds oc.
* fix typo's in description and set checkmode to false
* edit docs and module return
* support checkmode with get method
* fix unit test
* handle exception and add new example for embeds
* quote line
* fix typos
* fix yaml
(cherry picked from commit 0912e8cc7a)
Co-authored-by: CWollinger <CWollinger@web.de>
	
	
This commit is contained in:
		
					parent
					
						
							
								4cb6f39a80
							
						
					
				
			
			
				commit
				
					
						4032dd6b08
					
				
			
		
					 3 changed files with 319 additions and 0 deletions
				
			
		
							
								
								
									
										1
									
								
								plugins/modules/discord.py
									
										
									
									
									
										Symbolic link
									
								
							
							
						
						
									
										1
									
								
								plugins/modules/discord.py
									
										
									
									
									
										Symbolic link
									
								
							|  | @ -0,0 +1 @@ | |||
| ./notification/discord.py | ||||
							
								
								
									
										215
									
								
								plugins/modules/notification/discord.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										215
									
								
								plugins/modules/notification/discord.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,215 @@ | |||
| #!/usr/bin/python | ||||
| # -*- coding: utf-8 -*- | ||||
| 
 | ||||
| # Copyright: (c) 2021, Christian Wollinger <cwollinger@web.de> | ||||
| # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) | ||||
| 
 | ||||
| from __future__ import absolute_import, division, print_function | ||||
| __metaclass__ = type | ||||
| 
 | ||||
| DOCUMENTATION = ''' | ||||
| --- | ||||
| module: discord | ||||
| short_description: Send Discord messages | ||||
| version_added: 3.1.0 | ||||
| description: | ||||
|   - Sends a message to a Discord channel using the Discord webhook API. | ||||
| author: Christian Wollinger (@cwollinger) | ||||
| seealso: | ||||
|   - name: API documentation | ||||
|     description: Documentation for Discord API | ||||
|     link: https://discord.com/developers/docs/resources/webhook#execute-webhook | ||||
| options: | ||||
|   webhook_id: | ||||
|     description: | ||||
|       - The webhook ID. | ||||
|       - "Format from Discord webhook URL: C(/webhooks/{webhook.id}/{webhook.token})." | ||||
|     required: yes | ||||
|     type: str | ||||
|   webhook_token: | ||||
|     description: | ||||
|       - The webhook token. | ||||
|       - "Format from Discord webhook URL: C(/webhooks/{webhook.id}/{webhook.token})." | ||||
|     required: yes | ||||
|     type: str | ||||
|   content: | ||||
|     description: | ||||
|       - Content of the message to the Discord channel. | ||||
|       - At least one of I(content) and I(embeds) must be specified. | ||||
|     type: str | ||||
|   username: | ||||
|     description: | ||||
|       - Overrides the default username of the webhook. | ||||
|     type: str | ||||
|   avatar_url: | ||||
|     description: | ||||
|       - Overrides the default avatar of the webhook. | ||||
|     type: str | ||||
|   tts: | ||||
|     description: | ||||
|       - Set this to C(true) if this is a TTS (Text to Speech) message. | ||||
|     type: bool | ||||
|     default: false | ||||
|   embeds: | ||||
|     description: | ||||
|       - Send messages as Embeds to the Discord channel. | ||||
|       - Embeds can have a colored border, embedded images, text fields and more. | ||||
|       - "Allowed parameters are described in the Discord Docs: U(https://discord.com/developers/docs/resources/channel#embed-object)" | ||||
|       - At least one of I(content) and I(embeds) must be specified. | ||||
|     type: list | ||||
|     elements: dict | ||||
| ''' | ||||
| 
 | ||||
| EXAMPLES = """ | ||||
| - name: Send a message to the Discord channel | ||||
|   community.general.discord: | ||||
|     webhook_id: "00000" | ||||
|     webhook_token: "XXXYYY" | ||||
|     content: "This is a message from ansible" | ||||
| 
 | ||||
| - name: Send a message to the Discord channel with specific username and avatar | ||||
|   community.general.discord: | ||||
|     webhook_id: "00000" | ||||
|     webhook_token: "XXXYYY" | ||||
|     content: "This is a message from ansible" | ||||
|     username: Ansible | ||||
|     avatar_url: "https://docs.ansible.com/ansible/latest/_static/images/logo_invert.png" | ||||
| 
 | ||||
| - name: Send a embedded message to the Discord channel | ||||
|   community.general.discord: | ||||
|     webhook_id: "00000" | ||||
|     webhook_token: "XXXYYY" | ||||
|     embeds: | ||||
|       - title: "Embedded message" | ||||
|         description: "This is an embedded message" | ||||
|         footer: | ||||
|           text: "Author: Ansible" | ||||
|         image: | ||||
|           url: "https://docs.ansible.com/ansible/latest/_static/images/logo_invert.png" | ||||
| 
 | ||||
| - name: Send two embedded messages | ||||
|   community.general.discord: | ||||
|     webhook_id: "00000" | ||||
|     webhook_token: "XXXYYY" | ||||
|     embeds: | ||||
|       - title: "First message" | ||||
|         description: "This is my first embedded message" | ||||
|         footer: | ||||
|           text: "Author: Ansible" | ||||
|         image: | ||||
|           url: "https://docs.ansible.com/ansible/latest/_static/images/logo_invert.png" | ||||
|       - title: "Second message" | ||||
|         description: "This is my first second message" | ||||
|         footer: | ||||
|           text: "Author: Ansible" | ||||
|           icon_url: "https://docs.ansible.com/ansible/latest/_static/images/logo_invert.png" | ||||
|         fields: | ||||
|           - name: "Field 1" | ||||
|             value: "Value of my first field" | ||||
|           - name: "Field 2" | ||||
|             value: "Value of my second field" | ||||
|         timestamp: "{{ ansible_date_time.iso8601 }}" | ||||
| """ | ||||
| 
 | ||||
| RETURN = """ | ||||
| http_code: | ||||
|   description: | ||||
|     - Response Code returned by Discord API. | ||||
|   returned: always | ||||
|   type: int | ||||
|   sample: 204 | ||||
| """ | ||||
| 
 | ||||
| from ansible.module_utils.urls import fetch_url | ||||
| from ansible.module_utils.basic import AnsibleModule | ||||
| 
 | ||||
| 
 | ||||
| def discord_check_mode(module): | ||||
| 
 | ||||
|     webhook_id = module.params['webhook_id'] | ||||
|     webhook_token = module.params['webhook_token'] | ||||
| 
 | ||||
|     headers = { | ||||
|         'content-type': 'application/json' | ||||
|     } | ||||
| 
 | ||||
|     url = "https://discord.com/api/webhooks/%s/%s" % ( | ||||
|         webhook_id, webhook_token) | ||||
| 
 | ||||
|     response, info = fetch_url(module, url, method='GET', headers=headers) | ||||
|     return response, info | ||||
| 
 | ||||
| 
 | ||||
| def discord_text_msg(module): | ||||
| 
 | ||||
|     webhook_id = module.params['webhook_id'] | ||||
|     webhook_token = module.params['webhook_token'] | ||||
|     content = module.params['content'] | ||||
|     user = module.params['username'] | ||||
|     avatar_url = module.params['avatar_url'] | ||||
|     tts = module.params['tts'] | ||||
|     embeds = module.params['embeds'] | ||||
| 
 | ||||
|     headers = { | ||||
|         'content-type': 'application/json' | ||||
|     } | ||||
| 
 | ||||
|     url = "https://discord.com/api/webhooks/%s/%s" % ( | ||||
|         webhook_id, webhook_token) | ||||
| 
 | ||||
|     payload = { | ||||
|         'content': content, | ||||
|         'username': user, | ||||
|         'avatar_url': avatar_url, | ||||
|         'tts': tts, | ||||
|         'embeds': embeds, | ||||
|     } | ||||
| 
 | ||||
|     payload = module.jsonify(payload) | ||||
| 
 | ||||
|     response, info = fetch_url(module, url, data=payload, headers=headers, method='POST') | ||||
|     return response, info | ||||
| 
 | ||||
| 
 | ||||
| def main(): | ||||
|     module = AnsibleModule( | ||||
|         argument_spec=dict( | ||||
|             webhook_id=dict(type='str', required=True), | ||||
|             webhook_token=dict(type='str', required=True, no_log=True), | ||||
|             content=dict(type='str'), | ||||
|             username=dict(type='str'), | ||||
|             avatar_url=dict(type='str'), | ||||
|             tts=dict(type='bool', default=False), | ||||
|             embeds=dict(type='list', elements='dict'), | ||||
|         ), | ||||
|         required_one_of=[['content', 'embeds']], | ||||
|         supports_check_mode=True | ||||
|     ) | ||||
| 
 | ||||
|     result = dict( | ||||
|         changed=False, | ||||
|         http_code='', | ||||
|     ) | ||||
| 
 | ||||
|     if module.check_mode: | ||||
|         response, info = discord_check_mode(module) | ||||
|         if info['status'] != 200: | ||||
|             try: | ||||
|                 module.fail_json(http_code=info['status'], msg=info['msg'], response=module.from_json(info['body']), info=info) | ||||
|             except Exception: | ||||
|                 module.fail_json(http_code=info['status'], msg=info['msg'], info=info) | ||||
|         else: | ||||
|             module.exit_json(msg=info['msg'], changed=False, http_code=info['status'], response=module.from_json(response.read())) | ||||
|     else: | ||||
|         response, info = discord_text_msg(module) | ||||
|         if info['status'] != 204: | ||||
|             try: | ||||
|                 module.fail_json(http_code=info['status'], msg=info['msg'], response=module.from_json(info['body']), info=info) | ||||
|             except Exception: | ||||
|                 module.fail_json(http_code=info['status'], msg=info['msg'], info=info) | ||||
|         else: | ||||
|             module.exit_json(msg=info['msg'], changed=True, http_code=info['status']) | ||||
| 
 | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										103
									
								
								tests/unit/plugins/modules/notification/test_discord.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										103
									
								
								tests/unit/plugins/modules/notification/test_discord.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,103 @@ | |||
| # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) | ||||
| 
 | ||||
| from __future__ import (absolute_import, division, print_function) | ||||
| __metaclass__ = type | ||||
| 
 | ||||
| import json | ||||
| import pytest | ||||
| from ansible_collections.community.general.tests.unit.compat.mock import Mock, patch | ||||
| from ansible_collections.community.general.plugins.modules.notification import discord | ||||
| from ansible_collections.community.general.tests.unit.plugins.modules.utils import AnsibleExitJson, AnsibleFailJson, ModuleTestCase, set_module_args | ||||
| 
 | ||||
| 
 | ||||
| class TestDiscordModule(ModuleTestCase): | ||||
| 
 | ||||
|     def setUp(self): | ||||
|         super(TestDiscordModule, self).setUp() | ||||
|         self.module = discord | ||||
| 
 | ||||
|     def tearDown(self): | ||||
|         super(TestDiscordModule, self).tearDown() | ||||
| 
 | ||||
|     @pytest.fixture | ||||
|     def fetch_url_mock(self, mocker): | ||||
|         return mocker.patch('ansible.module_utils.notification.discord.fetch_url') | ||||
| 
 | ||||
|     def test_without_parameters(self): | ||||
|         """Failure if no parameters set""" | ||||
|         with self.assertRaises(AnsibleFailJson): | ||||
|             set_module_args({}) | ||||
|             self.module.main() | ||||
| 
 | ||||
|     def test_without_content(self): | ||||
|         """Failure if content and embeds both are missing""" | ||||
|         set_module_args({ | ||||
|             'webhook_id': 'xxx', | ||||
|             'webhook_token': 'xxx' | ||||
|         }) | ||||
|         with self.assertRaises(AnsibleFailJson): | ||||
|             self.module.main() | ||||
| 
 | ||||
|     def test_successful_message(self): | ||||
|         """Test a basic message successfully.""" | ||||
|         set_module_args({ | ||||
|             'webhook_id': 'xxx', | ||||
|             'webhook_token': 'xxx', | ||||
|             'content': 'test' | ||||
|         }) | ||||
| 
 | ||||
|         with patch.object(discord, "fetch_url") as fetch_url_mock: | ||||
|             fetch_url_mock.return_value = (None, {"status": 204, 'msg': 'OK (0 bytes)'}) | ||||
|             with self.assertRaises(AnsibleExitJson): | ||||
|                 self.module.main() | ||||
| 
 | ||||
|             self.assertTrue(fetch_url_mock.call_count, 1) | ||||
|             call_data = json.loads(fetch_url_mock.call_args[1]['data']) | ||||
|             assert call_data['content'] == "test" | ||||
| 
 | ||||
|     def test_message_with_username(self): | ||||
|         """Test a message with username set successfully.""" | ||||
|         set_module_args({ | ||||
|             'webhook_id': 'xxx', | ||||
|             'webhook_token': 'xxx', | ||||
|             'content': 'test', | ||||
|             'username': 'Ansible Bot' | ||||
|         }) | ||||
| 
 | ||||
|         with patch.object(discord, "fetch_url") as fetch_url_mock: | ||||
|             fetch_url_mock.return_value = (None, {"status": 204, 'msg': 'OK (0 bytes)'}) | ||||
|             with self.assertRaises(AnsibleExitJson): | ||||
|                 self.module.main() | ||||
| 
 | ||||
|             self.assertTrue(fetch_url_mock.call_count, 1) | ||||
|             call_data = json.loads(fetch_url_mock.call_args[1]['data']) | ||||
|             assert call_data['username'] == "Ansible Bot" | ||||
|             assert call_data['content'] == "test" | ||||
| 
 | ||||
|     def test_failed_message(self): | ||||
|         """Test failure because webhook id is wrong.""" | ||||
| 
 | ||||
|         set_module_args({ | ||||
|             'webhook_id': 'wrong', | ||||
|             'webhook_token': 'xxx', | ||||
|             'content': 'test' | ||||
|         }) | ||||
| 
 | ||||
|         with patch.object(discord, "fetch_url") as fetch_url_mock: | ||||
|             fetch_url_mock.return_value = (None, {"status": 404, 'msg': 'HTTP Error 404: Not Found', 'body': '{"message": "Unknown Webhook", "code": 10015}'}) | ||||
|             with self.assertRaises(AnsibleFailJson): | ||||
|                 self.module.main() | ||||
| 
 | ||||
|     def test_failed_message_without_body(self): | ||||
|         """Test failure with empty response body.""" | ||||
| 
 | ||||
|         set_module_args({ | ||||
|             'webhook_id': 'wrong', | ||||
|             'webhook_token': 'xxx', | ||||
|             'content': 'test' | ||||
|         }) | ||||
| 
 | ||||
|         with patch.object(discord, "fetch_url") as fetch_url_mock: | ||||
|             fetch_url_mock.return_value = (None, {"status": 404, 'msg': 'HTTP Error 404: Not Found'}) | ||||
|             with self.assertRaises(AnsibleFailJson): | ||||
|                 self.module.main() | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue