mirror of
https://github.com/BreizhHardware/ntfy_alerts.git
synced 2026-01-18 16:37:28 +01:00
Add Slack notifcation
✨ Support Slack incoming webhook notifications with good looking block message design
This commit is contained in:
18
ntfy.py
18
ntfy.py
@@ -20,6 +20,11 @@ from send_discord import (
|
|||||||
docker_send_to_discord,
|
docker_send_to_discord,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
from send_slack import (
|
||||||
|
github_send_to_slack,
|
||||||
|
docker_send_to_slack,
|
||||||
|
)
|
||||||
|
|
||||||
# Configuring the logger
|
# Configuring the logger
|
||||||
logging.basicConfig(
|
logging.basicConfig(
|
||||||
level=logging.INFO,
|
level=logging.INFO,
|
||||||
@@ -39,7 +44,7 @@ discord_webhook_url = os.environ.get("DISCORD_WEBHOOK_URL")
|
|||||||
|
|
||||||
|
|
||||||
def create_dockerhub_token(username, password):
|
def create_dockerhub_token(username, password):
|
||||||
url = "https://hub.docker.com//v2/users/login"
|
url = "https://hub.docker.com/v2/users/login"
|
||||||
headers = {"Content-Type": "application/json"}
|
headers = {"Content-Type": "application/json"}
|
||||||
data = json.dumps({"username": username, "password": password})
|
data = json.dumps({"username": username, "password": password})
|
||||||
|
|
||||||
@@ -177,7 +182,7 @@ def get_changelog(repo):
|
|||||||
return latest_release_list["body"]
|
return latest_release_list["body"]
|
||||||
return "Changelog not available"
|
return "Changelog not available"
|
||||||
|
|
||||||
def notify_all_services(github_latest_release, docker_latest_release, auth, ntfy_url, gotify_url, gotify_token, discord_webhook_url):
|
def notify_all_services(github_latest_release, docker_latest_release, auth, ntfy_url, gotify_url, gotify_token, discord_webhook_url, slack_webhook_url):
|
||||||
threads = []
|
threads = []
|
||||||
|
|
||||||
if ntfy_url:
|
if ntfy_url:
|
||||||
@@ -198,6 +203,12 @@ def notify_all_services(github_latest_release, docker_latest_release, auth, ntfy
|
|||||||
if docker_latest_release:
|
if docker_latest_release:
|
||||||
threads.append(threading.Thread(target=docker_send_to_discord, args=(docker_latest_release, discord_webhook_url)))
|
threads.append(threading.Thread(target=docker_send_to_discord, args=(docker_latest_release, discord_webhook_url)))
|
||||||
|
|
||||||
|
if slack_webhook_url:
|
||||||
|
if github_latest_release:
|
||||||
|
threads.append(threading.Thread(target=github_send_to_slack, args=(github_latest_release, slack_webhook_url)))
|
||||||
|
if docker_latest_release:
|
||||||
|
threads.append(threading.Thread(target=docker_send_to_slack, args=(docker_latest_release, slack_webhook_url)))
|
||||||
|
|
||||||
for thread in threads:
|
for thread in threads:
|
||||||
thread.start()
|
thread.start()
|
||||||
|
|
||||||
@@ -215,6 +226,7 @@ if __name__ == "__main__":
|
|||||||
gotify_token = os.environ.get("GOTIFY_TOKEN")
|
gotify_token = os.environ.get("GOTIFY_TOKEN")
|
||||||
discord_webhook_url = os.environ.get("DISCORD_WEBHOOK_URL")
|
discord_webhook_url = os.environ.get("DISCORD_WEBHOOK_URL")
|
||||||
timeout = float(os.environ.get("GHNTFY_TIMEOUT"))
|
timeout = float(os.environ.get("GHNTFY_TIMEOUT"))
|
||||||
|
slack_webhook_url = os.environ.get("SLACK_WEBHOOK_URL")
|
||||||
|
|
||||||
if auth and (ntfy_url or gotify_url or discord_webhook_url):
|
if auth and (ntfy_url or gotify_url or discord_webhook_url):
|
||||||
while True:
|
while True:
|
||||||
@@ -223,7 +235,7 @@ if __name__ == "__main__":
|
|||||||
docker_watched_repos_list = get_docker_watched_repos()
|
docker_watched_repos_list = get_docker_watched_repos()
|
||||||
docker_latest_release = get_latest_docker_releases(docker_watched_repos_list)
|
docker_latest_release = get_latest_docker_releases(docker_watched_repos_list)
|
||||||
|
|
||||||
notify_all_services(github_latest_release, docker_latest_release, auth, ntfy_url, gotify_url, gotify_token, discord_webhook_url)
|
notify_all_services(github_latest_release, docker_latest_release, auth, ntfy_url, gotify_url, gotify_token, discord_webhook_url, slack_webhook_url)
|
||||||
|
|
||||||
time.sleep(timeout)
|
time.sleep(timeout)
|
||||||
else:
|
else:
|
||||||
|
|||||||
131
send_slack.py
Normal file
131
send_slack.py
Normal file
@@ -0,0 +1,131 @@
|
|||||||
|
import requests
|
||||||
|
import sqlite3
|
||||||
|
import logging
|
||||||
|
|
||||||
|
logging.basicConfig(
|
||||||
|
level=logging.INFO,
|
||||||
|
format="%(asctime)s - %(name)s - %(levelname)s - %(message)s",
|
||||||
|
)
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
def get_db_connection():
|
||||||
|
return sqlite3.connect("/github-ntfy/ghntfy_versions.db", check_same_thread=False)
|
||||||
|
|
||||||
|
def github_send_to_slack(releases, webhook_url):
|
||||||
|
conn = get_db_connection()
|
||||||
|
cursor = conn.cursor()
|
||||||
|
for release in releases:
|
||||||
|
app_name = release["repo"].split("/")[-1]
|
||||||
|
version_number = release["tag_name"]
|
||||||
|
app_url = release["html_url"]
|
||||||
|
changelog = release["changelog"]
|
||||||
|
release_date = release["published_at"].replace("T", " ").replace("Z", "")
|
||||||
|
|
||||||
|
cursor.execute("SELECT version FROM versions WHERE repo=?", (app_name,))
|
||||||
|
previous_version = cursor.fetchone()
|
||||||
|
if previous_version and previous_version[0] == version_number:
|
||||||
|
logger.info(f"The version of {app_name} has not changed. No notification sent.")
|
||||||
|
continue
|
||||||
|
|
||||||
|
message = f"📌 *New version*: {version_number}\n\n📦*For*: {app_name}\n\n📅 *Published on*: {release_date}\n\n📝 *Changelog*:\n\n```{changelog}```"
|
||||||
|
if len(message) > 2000:
|
||||||
|
message = f"📌 *New version*: {version_number}\n\n📦*For*: {app_name}\n\n📅 *Published on*: {release_date}\n\n📝 *Changelog*:\n\n `check url since the changelog is huge`"
|
||||||
|
|
||||||
|
cursor.execute(
|
||||||
|
"INSERT OR REPLACE INTO versions (repo, version, changelog) VALUES (?, ?, ?)",
|
||||||
|
(app_name, version_number, changelog),
|
||||||
|
)
|
||||||
|
conn.commit()
|
||||||
|
|
||||||
|
|
||||||
|
message = {
|
||||||
|
"blocks": [
|
||||||
|
{
|
||||||
|
"type": "section",
|
||||||
|
"text": {
|
||||||
|
"type": "mrkdwn",
|
||||||
|
"text": f"{message}"
|
||||||
|
},
|
||||||
|
"accessory": {
|
||||||
|
"type": "button",
|
||||||
|
"text": {
|
||||||
|
"type": "plain_text",
|
||||||
|
"text": "🔗 Changelog url"
|
||||||
|
},
|
||||||
|
"url": f"{app_url}",
|
||||||
|
"action_id": "button-action"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "divider"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
headers = {
|
||||||
|
"Content-Type": "application/json"
|
||||||
|
}
|
||||||
|
response = requests.post(webhook_url, json=message, headers=headers)
|
||||||
|
if response.status_code == 200:
|
||||||
|
logger.info(f"Message sent to Slack for {app_name}")
|
||||||
|
else:
|
||||||
|
logger.error(f"Failed to send message to Slack. Status code: {response.status_code}")
|
||||||
|
logger.error(f"Response: {response.text}")
|
||||||
|
conn.close()
|
||||||
|
|
||||||
|
def docker_send_to_slack(releases, webhook_url):
|
||||||
|
conn = get_db_connection()
|
||||||
|
cursor = conn.cursor()
|
||||||
|
for release in releases:
|
||||||
|
app_name = release["repo"].split("/")[-1]
|
||||||
|
digest_number = release["digest"]
|
||||||
|
app_url = release["html_url"]
|
||||||
|
release_date = release["published_at"].replace("T", " ").replace("Z", "")
|
||||||
|
|
||||||
|
cursor.execute("SELECT digest FROM docker_versions WHERE repo=?", (app_name,))
|
||||||
|
previous_digest = cursor.fetchone()
|
||||||
|
if previous_digest and previous_digest[0] == digest_number:
|
||||||
|
logger.info(f"The digest of {app_name} has not changed. No notification sent.")
|
||||||
|
continue
|
||||||
|
|
||||||
|
message = f"🐳 *Docker Image Updated!*\n\n🔐 *New Digest*: `{digest_number}`\n\n📦 *App*: {app_name}\n\n*Published*: {release_date}"
|
||||||
|
|
||||||
|
cursor.execute(
|
||||||
|
"INSERT OR REPLACE INTO docker_versions (repo, digest) VALUES (?, ?)",
|
||||||
|
(app_name, digest_number),
|
||||||
|
)
|
||||||
|
conn.commit()
|
||||||
|
|
||||||
|
message = {
|
||||||
|
"blocks": [
|
||||||
|
{
|
||||||
|
"type": "section",
|
||||||
|
"text": {
|
||||||
|
"type": "mrkdwn",
|
||||||
|
"text": f"{message}"
|
||||||
|
},
|
||||||
|
"accessory": {
|
||||||
|
"type": "button",
|
||||||
|
"text": {
|
||||||
|
"type": "plain_text",
|
||||||
|
"text": "🔗 Changelog url"
|
||||||
|
},
|
||||||
|
"url": f"{app_url}",
|
||||||
|
"action_id": "button-action"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "divider"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
headers = {
|
||||||
|
"Content-Type": "application/json"
|
||||||
|
}
|
||||||
|
response = requests.post(webhook_url, json=message, headers=headers)
|
||||||
|
if 200 <= response.status_code < 300:
|
||||||
|
logger.info(f"Message sent to Slack for {app_name}")
|
||||||
|
else:
|
||||||
|
logger.error(f"Failed to send message to Slack. Status code: {response.status_code}")
|
||||||
|
logger.error(f"Response: {response.text}")
|
||||||
|
conn.close()
|
||||||
|
|
||||||
Reference in New Issue
Block a user