Files
Azalea Gui ef8e8a717d
Python package / build (3.7) (push) Has been cancelled
Python package / build (3.8) (push) Has been cancelled
Python package / build (3.9) (push) Has been cancelled
[O] Add message for ping
2023-01-03 04:46:36 -05:00

197 lines
6.3 KiB
Python

"""
github-webhook-to-telegram, receive GitHub webhooks and send to Telegram
Copyright (C) 2021 Dash Eclipse
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
"""
import hmac
import logging
from hashlib import sha256
from html import escape
from json.decoder import JSONDecodeError
from typing import Optional
from typing import Union
from aiohttp.web_request import Request
from multidict import CIMultiDictProxy
from config import GH_WEBHOOKS
async def validate_github_webhook(request: Request) -> Union[str, int, bool]:
try:
headers = request.headers
if not headers.get('User-Agent').startswith('GitHub-Hookshot'):
logging.warning("User agent: not from GitHub")
return False
if headers.get('Content-Type') != 'application/json':
logging.warning("Content type: not json")
return False
payload = await request.json()
hook_target: Optional[dict] = await _get_hook_target(payload)
if not hook_target:
return False
valid_signature = await _verify_signature(
bytes(hook_target['secret'], 'UTF-8'),
headers.get('X-Hub-Signature-256').split('=')[1],
await request.read()
)
if valid_signature:
return hook_target['chat_id']
else:
return False
except (JSONDecodeError, AttributeError) as error:
logging.warning("Invalid: %s", error)
return False
async def _get_hook_target(payload: dict) -> Optional[dict]:
name = (payload.get('organization', {}).get('login')
or payload.get('repository', {}).get('full_name'))
if not name:
logging.warning("no repo or organization found")
return None
target = GH_WEBHOOKS.get(name, None)
if not target:
logging.warning("unknown repo or organization")
return target
async def _verify_signature(secret: bytes, sig: str, msg: bytes) -> bool:
mac = hmac.new(secret, msg=msg, digestmod=sha256)
valid_signature = hmac.compare_digest(mac.hexdigest(), sig)
if not valid_signature:
logging.warning("Invalid signature")
return valid_signature
async def format_github_webhook(request: Request) -> Optional[str]:
headers: CIMultiDictProxy[str] = request.headers
event: str = headers.get('X-GitHub-Event')
payload: dict = await request.json()
if event in ALL_EVENTS:
title = await _get_event_title(event, payload)
details = await globals()[f"_format_{event}"](payload)
return f"{title}\n{details}" if details else title
return None
async def _format_create(payload: dict) -> str:
return "\u2192 <a href=\"{url}\">{ref_type}: {ref}</a>".format(
url=f"{payload['repository']['html_url']}/tree/{payload['ref']}",
ref_type=payload['ref_type'],
ref=payload['ref']
)
async def _format_delete(payload: dict) -> str:
return "\u2192 {ref_type}: <code>{ref}</code>".format(
ref_type=payload['ref_type'],
ref=payload['ref']
)
async def _format_discussion(payload: dict) -> str:
discussion = payload['discussion']
escaped_title = escape(discussion['title'])
return "\u2192 <a href=\"{url}\">{title}</a>".format(
url=discussion['html_url'],
title=f"{escaped_title} \u00b7 Discussion #{discussion['number']}"
)
async def _format_fork(payload: dict) -> str:
forkee = payload['forkee']
text = ["\u2192 <a href=\"{url}\">{name}</a>".format(
url=forkee['html_url'],
name=forkee['full_name']
), await _get_repo_star_and_fork(payload['repository'])]
return "\n".join(text)
async def _format_issues(payload: dict) -> str:
issue = payload['issue']
return "\u2192 <a href=\"{url}\">{title}</a>".format(
url=issue['html_url'],
title=f"{escape(issue['title'])} \u00b7 Issue #{issue['number']}"
)
async def _format_ping(_: dict) -> str:
return "Ping!"
async def _format_public(payload: dict) -> str:
repo = payload['repository']
return "\u2192 <a href=\"{url}\">{name}</a>".format(
url=repo['html_url'],
name=repo['full_name']
)
async def _format_pull_request(payload: dict) -> str:
pr = payload['pull_request']
url = pr['html_url']
title = "{pr_title} by {user} \u00b7 Pull Request #{number}".format(
pr_title=escape(pr['title']),
user=pr['user']['login'],
number=payload['number']
)
return f"\u2192 <a href=\"{url}\">{title}</a>"
async def _format_push(payload: dict) -> str:
ref = [f"\u2192 {payload['ref']}"]
commits = payload['commits']
commits_list = [
"\u2192 "
"<code>{name}</code> {message} [<a href=\"{url}\">{cid}</a>]".format(
name=commit['author']['username'],
message=escape(commit['message']),
url=commit['url'],
cid=commit['id'][:7]
)
for commit in commits
]
return "\n".join(ref + commits_list)
async def _format_star(payload: dict) -> str:
return await _get_repo_star_and_fork(payload['repository'])
async def _get_event_title(event: str, payload: dict) -> str:
name = (payload.get('repository', {}).get('full_name')
or payload.get('organization', {}).get('login'))
summary = [payload['sender']['login']]
# if action := payload.get('action'): summary.append(action)
action = payload.get('action')
summary.append(action) if action else []
summary.append(event)
return f"<b>{name}</b> | <i>{' '.join(summary)}</i>"
async def _get_repo_star_and_fork(repo: dict) -> str:
return '\u2192 ' + ", ".join([
f"<b>{repo['stargazers_count']}</b> stargazers",
f"<b>{repo['forks_count']}</b> forks"
])
ALL_EVENTS = tuple(
i.removeprefix('_format_') for i in dir() if i.startswith('_format_')
)