1
0
Fork 0
mirror of https://github.com/ansible-collections/community.general.git synced 2026-06-30 07:50:43 +00:00

slack: support file upload (#12032)

* slack: add support for file uploads and threading

* slack: add support for file uploads and threading

* docs: rename fragment to match PR #12032

* Fix validate-modules issues and update documentation for files support

* Fix tests

* Fix tests

* Fix tests

* Fix tests

* chore: fix nox sanity issues

* style: add author copyright

* style: fix examples

* build: trigger CI due to infrastructure timeout

* Update plugins/modules/slack.py

Co-authored-by: Alexei Znamensky <103110+russoz@users.noreply.github.com>

* doc: address reviewer feedback on changelog and token placeholder

* doc: address reviewer feedback on changelog and token placeholder

* fix: address maintainer feedback

* fix: pipeline status, rm continue

* fix: fix unit tests

* fix: linter fix

* fix: fix comments

* Update plugins/modules/slack.py

Co-authored-by: Felix Fontein <felix@fontein.de>

* Update plugins/modules/slack.py

Co-authored-by: Felix Fontein <felix@fontein.de>

* docs: remove outdated comment about failing logic

* Update plugins/modules/slack.py

Co-authored-by: Alexei Znamensky <103110+russoz@users.noreply.github.com>

* Update plugins/modules/slack.py

Co-authored-by: Felix Fontein <felix@fontein.de>

* fix: handle missing files via fail_on_file_error

* Apply suggestions from code review

Co-authored-by: Alexei Znamensky <103110+russoz@users.noreply.github.com>

* Apply suggestions from code review

Co-authored-by: Alexei Znamensky <103110+russoz@users.noreply.github.com>

* fix: adjust options syntax and formatting

---------

Co-authored-by: Максим Бакуревич <maksimbakurevic@MacBook-Air-Maksim.local>
Co-authored-by: Alexei Znamensky <103110+russoz@users.noreply.github.com>
Co-authored-by: Felix Fontein <felix@fontein.de>
This commit is contained in:
Maxim Bakurevych 2026-05-30 14:37:04 +03:00 committed by GitHub
parent 8faf8c3838
commit 580e8ad3f9
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
3 changed files with 246 additions and 33 deletions

View file

@ -5,7 +5,7 @@
from __future__ import annotations
import json
from unittest.mock import Mock, patch
from unittest.mock import Mock, mock_open, patch
import pytest
from ansible_collections.community.internal_test_tools.tests.unit.plugins.modules.utils import (
@ -48,7 +48,7 @@ class TestSlackModule(ModuleTestCase):
def test_successful_message(self):
"""tests sending a message. This is example 1 from the docs"""
with set_module_args({"token": "XXXX/YYYY/ZZZZ", "msg": "test"}):
with set_module_args({"token": "TXX/BYY/ZZZ", "msg": "test"}):
with patch.object(slack, "fetch_url") as fetch_url_mock:
fetch_url_mock.return_value = (None, {"status": 200})
with self.assertRaises(AnsibleExitJson):
@ -58,12 +58,12 @@ class TestSlackModule(ModuleTestCase):
call_data = json.loads(fetch_url_mock.call_args[1]["data"])
assert call_data["username"] == "Ansible"
assert call_data["text"] == "test"
assert fetch_url_mock.call_args[1]["url"] == "https://hooks.slack.com/services/XXXX/YYYY/ZZZZ"
assert fetch_url_mock.call_args[1]["url"] == "https://hooks.slack.com/services/TXX/BYY/ZZZ"
def test_failed_message(self):
"""tests failing to send a message"""
with set_module_args({"token": "XXXX/YYYY/ZZZZ", "msg": "test"}):
with set_module_args({"token": "TXX/BYY/ZZZ", "msg": "test"}):
with patch.object(slack, "fetch_url") as fetch_url_mock:
fetch_url_mock.return_value = (None, {"status": 404, "msg": "test"})
with self.assertRaises(AnsibleFailJson):
@ -71,7 +71,7 @@ class TestSlackModule(ModuleTestCase):
def test_message_with_thread(self):
"""tests sending a message with a thread"""
with set_module_args({"token": "XXXX/YYYY/ZZZZ", "msg": "test", "thread_id": "100.00"}):
with set_module_args({"token": "TXX/BYY/ZZZ", "msg": "test", "thread_id": "100.00"}):
with patch.object(slack, "fetch_url") as fetch_url_mock:
fetch_url_mock.return_value = (None, {"status": 200})
with self.assertRaises(AnsibleExitJson):
@ -82,14 +82,14 @@ class TestSlackModule(ModuleTestCase):
assert call_data["username"] == "Ansible"
assert call_data["text"] == "test"
assert call_data["thread_ts"] == "100.00"
assert fetch_url_mock.call_args[1]["url"] == "https://hooks.slack.com/services/XXXX/YYYY/ZZZZ"
assert fetch_url_mock.call_args[1]["url"] == "https://hooks.slack.com/services/TXX/BYY/ZZZ"
# https://github.com/ansible-collections/community.general/issues/1097
def test_ts_in_message_does_not_cause_edit(self):
with set_module_args({"token": "xoxa-123456789abcdef", "msg": "test with ts"}):
with patch.object(slack, "fetch_url") as fetch_url_mock:
mock_response = Mock()
mock_response.read.return_value = '{"fake":"data"}'
mock_response.read.return_value = '{"ok": true, "fake":"data"}'
fetch_url_mock.return_value = (mock_response, {"status": 200})
with self.assertRaises(AnsibleExitJson):
self.module.main()
@ -101,7 +101,7 @@ class TestSlackModule(ModuleTestCase):
with set_module_args({"token": "xoxa-123456789abcdef", "domain": "slack-gov.com", "msg": "test with ts"}):
with patch.object(slack, "fetch_url") as fetch_url_mock:
mock_response = Mock()
mock_response.read.return_value = '{"fake":"data"}'
mock_response.read.return_value = '{"ok": true, "fake":"data"}'
fetch_url_mock.return_value = (mock_response, {"status": 200})
with self.assertRaises(AnsibleExitJson):
self.module.main()
@ -113,7 +113,7 @@ class TestSlackModule(ModuleTestCase):
with set_module_args({"token": "xoxa-123456789abcdef", "msg": "test2", "message_id": "12345"}):
with patch.object(slack, "fetch_url") as fetch_url_mock:
mock_response = Mock()
mock_response.read.return_value = '{"messages":[{"ts":"12345","msg":"test1"}]}'
mock_response.read.return_value = '{"ok": true, "messages":[{"ts":"12345","msg":"test1"}]}'
fetch_url_mock.side_effect = [
(mock_response, {"status": 200}),
(mock_response, {"status": 200}),
@ -130,7 +130,7 @@ class TestSlackModule(ModuleTestCase):
"""tests sending a message with blocks"""
with set_module_args(
{
"token": "XXXX/YYYY/ZZZZ",
"token": "TXX/BYY/ZZZ",
"msg": "test",
"blocks": [
{
@ -155,13 +155,13 @@ class TestSlackModule(ModuleTestCase):
call_data = json.loads(fetch_url_mock.call_args[1]["data"])
assert call_data["username"] == "Ansible"
assert call_data["blocks"][1]["text"]["text"] == "test"
assert fetch_url_mock.call_args[1]["url"] == "https://hooks.slack.com/services/XXXX/YYYY/ZZZZ"
assert fetch_url_mock.call_args[1]["url"] == "https://hooks.slack.com/services/TXX/BYY/ZZZ"
def test_message_with_invalid_color(self):
"""tests sending invalid color value to module"""
with set_module_args(
{
"token": "XXXX/YYYY/ZZZZ",
"token": "TXX/BYY/ZZZ",
"msg": "test",
"color": "aa",
}
@ -176,6 +176,24 @@ class TestSlackModule(ModuleTestCase):
)
assert exec_info.exception.args[0]["msg"] == msg
def test_upload_files_only(self):
with set_module_args(
{"token": "xoxb-12345", "channel": "C123", "files": [{"path": "/tmp/test.txt", "name": "hello.txt"}]}
):
with patch.object(slack, "fetch_url") as fetch_url_mock:
with patch("os.path.exists", return_value=True):
with patch("os.path.getsize", return_value=100):
with patch("builtins.open", mock_open(read_data=b"data")):
mock_resp = Mock()
mock_resp.read.side_effect = [
'{"ok": true, "upload_url": "https://upload", "file_id": "F1"}',
'{"ok": true}',
]
fetch_url_mock.return_value = (mock_resp, {"status": 200})
with self.assertRaises(AnsibleExitJson) as result:
self.module.main()
self.assertTrue(result.exception.args[0]["changed"])
color_test = [
("#111111", True),