formatters: add github
This commit is contained in:
		
					parent
					
						
							
								4bcdb25c80
							
						
					
				
			
			
				commit
				
					
						6b5d6e6e87
					
				
			
		
					 9 changed files with 136 additions and 33 deletions
				
			
		| 
						 | 
					@ -23,7 +23,7 @@ repos:
 | 
				
			||||||
    hooks:
 | 
					    hooks:
 | 
				
			||||||
    -   id: pydocstyle
 | 
					    -   id: pydocstyle
 | 
				
			||||||
        args:
 | 
					        args:
 | 
				
			||||||
        - --ignore=D200,D212
 | 
					        - --ignore=D200,D203,D212
 | 
				
			||||||
-   repo: https://github.com/PyCQA/flake8
 | 
					-   repo: https://github.com/PyCQA/flake8
 | 
				
			||||||
    rev: 3.9.2
 | 
					    rev: 3.9.2
 | 
				
			||||||
    hooks:
 | 
					    hooks:
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -6,10 +6,11 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
## [Unreleased]
 | 
					## [Unreleased]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
- add grafana formatter
 | 
					- add github & grafana formatters
 | 
				
			||||||
- add formatted_body to bypass markdown with direct
 | 
					- add formatted_body to bypass markdown with direct
 | 
				
			||||||
  [matrix-custom-HTML](https://matrix.org/docs/spec/client_server/r0.6.1#m-room-message-msgtypes)
 | 
					  [matrix-custom-HTML](https://matrix.org/docs/spec/client_server/r0.6.1#m-room-message-msgtypes)
 | 
				
			||||||
- allow "key" to be passed as a parameter
 | 
					- allow "key" to be passed as a parameter
 | 
				
			||||||
 | 
					- allow to use a sha256 HMAC hex digest with the key instead of the raw key
 | 
				
			||||||
- allow "room_id" to be passed as a parameter or with the data
 | 
					- allow "room_id" to be passed as a parameter or with the data
 | 
				
			||||||
- rename "text" to "body".
 | 
					- rename "text" to "body".
 | 
				
			||||||
- Publish releases also on github from github actions
 | 
					- Publish releases also on github from github actions
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -13,3 +13,20 @@ def grafana(data, headers):
 | 
				
			||||||
            text = text + "* " + match["metric"] + ": " + str(match["value"]) + "\n"
 | 
					            text = text + "* " + match["metric"] + ": " + str(match["value"]) + "\n"
 | 
				
			||||||
    data["body"] = text
 | 
					    data["body"] = text
 | 
				
			||||||
    return data
 | 
					    return data
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def github(data, headers):
 | 
				
			||||||
 | 
					    """Pretty-print a github notification."""
 | 
				
			||||||
 | 
					    # TODO: Write nice useful formatters. This is only an example.
 | 
				
			||||||
 | 
					    if headers["X-GitHub-Event"] == "push":
 | 
				
			||||||
 | 
					        pusher, ref, a, b, c = [
 | 
				
			||||||
 | 
					            data[k] for k in ["pusher", "ref", "after", "before", "compare"]
 | 
				
			||||||
 | 
					        ]
 | 
				
			||||||
 | 
					        pusher = f"[{pusher['name']}](https://github.com/{pusher['name']})"
 | 
				
			||||||
 | 
					        data["body"] = f"@{pusher} pushed on {ref}: [{b} → {a}]({c}):\n\n"
 | 
				
			||||||
 | 
					        for commit in data["commits"]:
 | 
				
			||||||
 | 
					            data["body"] += f"- [{commit['message']}]({commit['url']})\n"
 | 
				
			||||||
 | 
					    else:
 | 
				
			||||||
 | 
					        data["body"] = "notification from github"
 | 
				
			||||||
 | 
					    data["digest"] = headers["X-Hub-Signature-256"].replace("sha256=", "")
 | 
				
			||||||
 | 
					    return data
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -3,6 +3,7 @@
 | 
				
			||||||
import json
 | 
					import json
 | 
				
			||||||
import logging
 | 
					import logging
 | 
				
			||||||
from http import HTTPStatus
 | 
					from http import HTTPStatus
 | 
				
			||||||
 | 
					from hmac import HMAC
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from markdown import markdown
 | 
					from markdown import markdown
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -18,10 +19,10 @@ async def matrix_webhook(request):
 | 
				
			||||||
    This one handles a POST, checks its content, and forwards it to the matrix room.
 | 
					    This one handles a POST, checks its content, and forwards it to the matrix room.
 | 
				
			||||||
    """
 | 
					    """
 | 
				
			||||||
    LOGGER.debug(f"Handling {request=}")
 | 
					    LOGGER.debug(f"Handling {request=}")
 | 
				
			||||||
    data = await request.read()
 | 
					    data_b = await request.read()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
        data = json.loads(data.decode())
 | 
					        data = json.loads(data_b.decode())
 | 
				
			||||||
    except json.decoder.JSONDecodeError:
 | 
					    except json.decoder.JSONDecodeError:
 | 
				
			||||||
        return utils.create_json_response(HTTPStatus.BAD_REQUEST, "Invalid JSON")
 | 
					        return utils.create_json_response(HTTPStatus.BAD_REQUEST, "Invalid JSON")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -48,6 +49,16 @@ async def matrix_webhook(request):
 | 
				
			||||||
    if "room_id" not in data:
 | 
					    if "room_id" not in data:
 | 
				
			||||||
        data["room_id"] = request.path.lstrip("/")
 | 
					        data["room_id"] = request.path.lstrip("/")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # If we get a good SHA-256 HMAC digest,
 | 
				
			||||||
 | 
					    # we can consider that the sender has the right API key
 | 
				
			||||||
 | 
					    if "digest" in data:
 | 
				
			||||||
 | 
					        if data["digest"] == HMAC(conf.API_KEY.encode(), data_b, "sha256").hexdigest():
 | 
				
			||||||
 | 
					            data["key"] = conf.API_KEY
 | 
				
			||||||
 | 
					        else:  # but if there is a wrong digest, an informative error should be provided
 | 
				
			||||||
 | 
					            return utils.create_json_response(
 | 
				
			||||||
 | 
					                HTTPStatus.UNAUTHORIZED, "Invalid SHA-256 HMAC digest"
 | 
				
			||||||
 | 
					            )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    missing = []
 | 
					    missing = []
 | 
				
			||||||
    for key in ["body", "key", "room_id"]:
 | 
					    for key in ["body", "key", "room_id"]:
 | 
				
			||||||
        if key not in data or not data[key]:
 | 
					        if key not in data or not data[key]:
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										1
									
								
								tests/example_github_push.json
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								tests/example_github_push.json
									
										
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										22
									
								
								tests/example_grafana.json
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								tests/example_grafana.json
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,22 @@
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  "dashboardId":1,
 | 
				
			||||||
 | 
					  "evalMatches":[
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					      "value":1,
 | 
				
			||||||
 | 
					      "metric":"Count",
 | 
				
			||||||
 | 
					      "tags":{}
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					  ],
 | 
				
			||||||
 | 
					  "imageUrl":"https://grafana.com/assets/img/blog/mixed_styles.png",
 | 
				
			||||||
 | 
					  "message":"Notification Message",
 | 
				
			||||||
 | 
					  "orgId":1,
 | 
				
			||||||
 | 
					  "panelId":2,
 | 
				
			||||||
 | 
					  "ruleId":1,
 | 
				
			||||||
 | 
					  "ruleName":"Panel Title alert",
 | 
				
			||||||
 | 
					  "ruleUrl":"http://localhost:3000/d/hZ7BuVbWz/test-dashboard?fullscreen\u0026edit\u0026tab=alert\u0026panelId=2\u0026orgId=1",
 | 
				
			||||||
 | 
					  "state":"alerting",
 | 
				
			||||||
 | 
					  "tags":{
 | 
				
			||||||
 | 
					    "tag name":"tag value"
 | 
				
			||||||
 | 
					  },
 | 
				
			||||||
 | 
					  "title":"[Alerting] Panel Title alert"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										71
									
								
								tests/test_github.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								tests/test_github.py
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,71 @@
 | 
				
			||||||
 | 
					"""Test module for grafana formatter."""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import unittest
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import httpx
 | 
				
			||||||
 | 
					import nio
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from .start import BOT_URL, FULL_ID, MATRIX_ID, MATRIX_PW, MATRIX_URL
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					SHA256 = "fd7522672889385736be8ffc86d1f8de2e15668864f49af729b5c63e5e0698c4"
 | 
				
			||||||
 | 
					EXAMPLE_GITHUB_REQUEST_HEADERS = {
 | 
				
			||||||
 | 
					    # 'Request URL': 'https://bot.saurel.me/room?formatter=github',
 | 
				
			||||||
 | 
					    # 'Request method': 'POST',
 | 
				
			||||||
 | 
					    "Accept": "*/*",
 | 
				
			||||||
 | 
					    "content-type": "application/json",
 | 
				
			||||||
 | 
					    "User-Agent": "GitHub-Hookshot/8d33975",
 | 
				
			||||||
 | 
					    "X-GitHub-Delivery": "636b9b1c-0761-11ec-8a8a-5e435c5ac4f4",
 | 
				
			||||||
 | 
					    "X-GitHub-Event": "push",
 | 
				
			||||||
 | 
					    "X-GitHub-Hook-ID": "311845633",
 | 
				
			||||||
 | 
					    "X-GitHub-Hook-Installation-Target-ID": "171114171",
 | 
				
			||||||
 | 
					    "X-GitHub-Hook-Installation-Target-Type": "repository",
 | 
				
			||||||
 | 
					    "X-Hub-Signature": "sha1=ea68fdfcb2f328aaa8f50d176f355e5d4fc95d94",
 | 
				
			||||||
 | 
					    "X-Hub-Signature-256": f"sha256={SHA256}",
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class GithubFormatterTest(unittest.IsolatedAsyncioTestCase):
 | 
				
			||||||
 | 
					    """Github formatter test class."""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async def test_github_body(self):
 | 
				
			||||||
 | 
					        """Send a markdown message, and check the result."""
 | 
				
			||||||
 | 
					        messages = []
 | 
				
			||||||
 | 
					        client = nio.AsyncClient(MATRIX_URL, MATRIX_ID)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        await client.login(MATRIX_PW)
 | 
				
			||||||
 | 
					        room = await client.room_create()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        with open("tests/example_github_push.json", "rb") as f:
 | 
				
			||||||
 | 
					            example_github_push = f.read().strip()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        self.assertEqual(
 | 
				
			||||||
 | 
					            httpx.post(
 | 
				
			||||||
 | 
					                f"{BOT_URL}/{room.room_id}",
 | 
				
			||||||
 | 
					                params={
 | 
				
			||||||
 | 
					                    "formatter": "github",
 | 
				
			||||||
 | 
					                },
 | 
				
			||||||
 | 
					                content=example_github_push,
 | 
				
			||||||
 | 
					                headers=EXAMPLE_GITHUB_REQUEST_HEADERS,
 | 
				
			||||||
 | 
					            ).json(),
 | 
				
			||||||
 | 
					            {"status": 200, "ret": "OK"},
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        sync = await client.sync()
 | 
				
			||||||
 | 
					        messages = await client.room_messages(room.room_id, sync.next_batch)
 | 
				
			||||||
 | 
					        await client.close()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        before = "ac7d1d9647008145e9d0cf65d24744d0db4862b8"
 | 
				
			||||||
 | 
					        after = "4bcdb25c809391baaabc264d9309059f9f48ead2"
 | 
				
			||||||
 | 
					        GH = "https://github.com"
 | 
				
			||||||
 | 
					        expected = f'<p>@<a href="{GH}/nim65s">nim65s</a> pushed on refs/heads/devel: '
 | 
				
			||||||
 | 
					        expected += f'<a href="{GH}/nim65s/matrix-webhook/compare/ac7d1d964700...'
 | 
				
			||||||
 | 
					        expected += f'4bcdb25c8093">{before} → {after}</a>:</p>\n<ul>\n<li>'
 | 
				
			||||||
 | 
					        expected += f'<a href="{GH}/nim65s/matrix-webhook/commit/{after}">'
 | 
				
			||||||
 | 
					        expected += "formatters: also get headers</a></li>\n</ul>"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        message = messages.chunk[0]
 | 
				
			||||||
 | 
					        self.assertEqual(message.sender, FULL_ID)
 | 
				
			||||||
 | 
					        self.assertEqual(
 | 
				
			||||||
 | 
					            message.formatted_body,
 | 
				
			||||||
 | 
					            expected,
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
| 
						 | 
					@ -1,4 +1,8 @@
 | 
				
			||||||
"""Test module for grafana formatter."""
 | 
					"""
 | 
				
			||||||
 | 
					Test module for grafana formatter.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ref https://grafana.com/docs/grafana/latest/alerting/old-alerting/notifications/#webhook
 | 
				
			||||||
 | 
					"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import unittest
 | 
					import unittest
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -7,32 +11,6 @@ import nio
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from .start import BOT_URL, FULL_ID, KEY, MATRIX_ID, MATRIX_PW, MATRIX_URL
 | 
					from .start import BOT_URL, FULL_ID, KEY, MATRIX_ID, MATRIX_PW, MATRIX_URL
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# ref https://grafana.com/docs/grafana/latest/alerting/old-alerting/notifications/#webhook
 | 
					 | 
				
			||||||
EXAMPLE_GRAFANA_REQUEST = """
 | 
					 | 
				
			||||||
{
 | 
					 | 
				
			||||||
  "dashboardId":1,
 | 
					 | 
				
			||||||
  "evalMatches":[
 | 
					 | 
				
			||||||
    {
 | 
					 | 
				
			||||||
      "value":1,
 | 
					 | 
				
			||||||
      "metric":"Count",
 | 
					 | 
				
			||||||
      "tags":{}
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  ],
 | 
					 | 
				
			||||||
  "imageUrl":"https://grafana.com/assets/img/blog/mixed_styles.png",
 | 
					 | 
				
			||||||
  "message":"Notification Message",
 | 
					 | 
				
			||||||
  "orgId":1,
 | 
					 | 
				
			||||||
  "panelId":2,
 | 
					 | 
				
			||||||
  "ruleId":1,
 | 
					 | 
				
			||||||
  "ruleName":"Panel Title alert",
 | 
					 | 
				
			||||||
  "ruleUrl":"http://localhost:3000/d/hZ7BuVbWz/test-dashboard?fullscreen\u0026edit\u0026tab=alert\u0026panelId=2\u0026orgId=1",
 | 
					 | 
				
			||||||
  "state":"alerting",
 | 
					 | 
				
			||||||
  "tags":{
 | 
					 | 
				
			||||||
    "tag name":"tag value"
 | 
					 | 
				
			||||||
  },
 | 
					 | 
				
			||||||
  "title":"[Alerting] Panel Title alert"
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
"""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
class GrafanaFormatterTest(unittest.IsolatedAsyncioTestCase):
 | 
					class GrafanaFormatterTest(unittest.IsolatedAsyncioTestCase):
 | 
				
			||||||
    """Grafana formatter test class."""
 | 
					    """Grafana formatter test class."""
 | 
				
			||||||
| 
						 | 
					@ -45,11 +23,13 @@ class GrafanaFormatterTest(unittest.IsolatedAsyncioTestCase):
 | 
				
			||||||
        await client.login(MATRIX_PW)
 | 
					        await client.login(MATRIX_PW)
 | 
				
			||||||
        room = await client.room_create()
 | 
					        room = await client.room_create()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        with open("tests/example_grafana.json") as f:
 | 
				
			||||||
 | 
					            example_grafana_request = f.read()
 | 
				
			||||||
        self.assertEqual(
 | 
					        self.assertEqual(
 | 
				
			||||||
            httpx.post(
 | 
					            httpx.post(
 | 
				
			||||||
                f"{BOT_URL}/{room.room_id}",
 | 
					                f"{BOT_URL}/{room.room_id}",
 | 
				
			||||||
                params={"formatter": "grafana", "key": KEY},
 | 
					                params={"formatter": "grafana", "key": KEY},
 | 
				
			||||||
                content=EXAMPLE_GRAFANA_REQUEST,
 | 
					                content=example_grafana_request,
 | 
				
			||||||
            ).json(),
 | 
					            ).json(),
 | 
				
			||||||
            {"status": 200, "ret": "OK"},
 | 
					            {"status": 200, "ret": "OK"},
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -86,7 +86,7 @@ class BotTest(unittest.IsolatedAsyncioTestCase):
 | 
				
			||||||
        self.assertEqual(message.formatted_body, "<h1>Hello</h1>")
 | 
					        self.assertEqual(message.formatted_body, "<h1>Hello</h1>")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    async def test_room_id_parameter(self):
 | 
					    async def test_room_id_parameter(self):
 | 
				
			||||||
        """Send a markdown message in a room given as parameter, and check the result."""
 | 
					        """Send a markdown message in a room given as parameter."""
 | 
				
			||||||
        body = "# Hello"
 | 
					        body = "# Hello"
 | 
				
			||||||
        messages = []
 | 
					        messages = []
 | 
				
			||||||
        client = nio.AsyncClient(MATRIX_URL, MATRIX_ID)
 | 
					        client = nio.AsyncClient(MATRIX_URL, MATRIX_ID)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue