Add support for Discord webhooks
This commit is contained in:
@ -2,3 +2,4 @@ DJANGO_SECRET_KEY=
|
|||||||
DEBUG=True
|
DEBUG=True
|
||||||
EMAIL_HOST_USER=
|
EMAIL_HOST_USER=
|
||||||
EMAIL_HOST_PASSWORD=
|
EMAIL_HOST_PASSWORD=
|
||||||
|
DISCORD_WEBHOOK_URL=
|
||||||
|
@ -59,6 +59,7 @@ DEFAULT_FROM_EMAIL: str = os.getenv(
|
|||||||
default="webmaster@localhost",
|
default="webmaster@localhost",
|
||||||
)
|
)
|
||||||
SERVER_EMAIL: str = os.getenv(key="EMAIL_HOST_USER", default="webmaster@localhost")
|
SERVER_EMAIL: str = os.getenv(key="EMAIL_HOST_USER", default="webmaster@localhost")
|
||||||
|
DISCORD_WEBHOOK_URL: str = os.getenv(key="DISCORD_WEBHOOK_URL", default="")
|
||||||
|
|
||||||
INSTALLED_APPS: list[str] = [
|
INSTALLED_APPS: list[str] = [
|
||||||
"core.apps.CoreConfig",
|
"core.apps.CoreConfig",
|
||||||
|
31
core/discord.py
Normal file
31
core/discord.py
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
import logging
|
||||||
|
from typing import TYPE_CHECKING
|
||||||
|
|
||||||
|
from discord_webhook import DiscordWebhook
|
||||||
|
from django.conf import settings
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from requests import Response
|
||||||
|
|
||||||
|
logger: logging.Logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
def send(message: str) -> None:
|
||||||
|
"""Send a message to Discord.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
message: The message to send.
|
||||||
|
"""
|
||||||
|
webhook_url = str(settings.DISCORD_WEBHOOK_URL)
|
||||||
|
if not webhook_url:
|
||||||
|
logger.error("No Discord webhook URL found.")
|
||||||
|
return
|
||||||
|
|
||||||
|
webhook = DiscordWebhook(
|
||||||
|
url=webhook_url,
|
||||||
|
content=message,
|
||||||
|
username="TTVDrops",
|
||||||
|
rate_limit_retry=True,
|
||||||
|
)
|
||||||
|
response: Response = webhook.execute()
|
||||||
|
logger.debug(response)
|
@ -131,7 +131,18 @@
|
|||||||
<h2>
|
<h2>
|
||||||
<a href="https://www.twitch.tv/directory/category/{{ game.slug }}">{{ game.display_name }}</a>
|
<a href="https://www.twitch.tv/directory/category/{{ game.slug }}">{{ game.display_name }}</a>
|
||||||
</h2>
|
</h2>
|
||||||
<button>Subscribe to {{ game.display_name }}</button>
|
<form action="" method="post">
|
||||||
|
{% csrf_token %}
|
||||||
|
<input type="hidden" name="game_id" value="{{ game.id }}">
|
||||||
|
<button type="submit">Subscribe to {{ game.display_name }}</button>
|
||||||
|
</form>
|
||||||
|
<form action='{% url "core:test" %}' method="post">
|
||||||
|
{% csrf_token %}
|
||||||
|
<input type="hidden"
|
||||||
|
name="org_id"
|
||||||
|
value="{{ org_data.drop_campaigns.0.pk }}">
|
||||||
|
<button type="submit">Test</button>
|
||||||
|
</form>
|
||||||
</header>
|
</header>
|
||||||
<ul>
|
<ul>
|
||||||
{% for drop_benefit in game_data.drop_benefits %}
|
{% for drop_benefit in game_data.drop_benefits %}
|
||||||
|
@ -9,4 +9,5 @@ app_name: str = "core"
|
|||||||
|
|
||||||
urlpatterns: list[URLPattern | URLResolver] = [
|
urlpatterns: list[URLPattern | URLResolver] = [
|
||||||
path(route="", view=views.index, name="index"),
|
path(route="", view=views.index, name="index"),
|
||||||
|
path(route="test/", view=views.test_webhook, name="test"),
|
||||||
]
|
]
|
||||||
|
@ -1,8 +1,11 @@
|
|||||||
|
import logging
|
||||||
from typing import TYPE_CHECKING
|
from typing import TYPE_CHECKING
|
||||||
|
|
||||||
|
from django.db.models.manager import BaseManager
|
||||||
from django.http import HttpRequest, HttpResponse
|
from django.http import HttpRequest, HttpResponse
|
||||||
from django.template.response import TemplateResponse
|
from django.template.response import TemplateResponse
|
||||||
|
|
||||||
|
from core.discord import send
|
||||||
from twitch.models import (
|
from twitch.models import (
|
||||||
DropBenefit,
|
DropBenefit,
|
||||||
DropCampaign,
|
DropCampaign,
|
||||||
@ -13,6 +16,9 @@ from twitch.models import (
|
|||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
from django.db.models.manager import BaseManager
|
from django.db.models.manager import BaseManager
|
||||||
|
from django.views.decorators.http import require_POST
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
def index(request: HttpRequest) -> HttpResponse:
|
def index(request: HttpRequest) -> HttpResponse:
|
||||||
@ -48,7 +54,7 @@ def index(request: HttpRequest) -> HttpResponse:
|
|||||||
for benefit in drop_benefits:
|
for benefit in drop_benefits:
|
||||||
orgs_data[org]["games"][benefit.game]["drop_benefits"].append(benefit)
|
orgs_data[org]["games"][benefit.game]["drop_benefits"].append(benefit)
|
||||||
|
|
||||||
time_based_drops = TimeBasedDrop.objects.filter(
|
time_based_drops: BaseManager[TimeBasedDrop] = TimeBasedDrop.objects.filter(
|
||||||
benefits__in=drop_benefits,
|
benefits__in=drop_benefits,
|
||||||
).distinct()
|
).distinct()
|
||||||
for drop in time_based_drops:
|
for drop in time_based_drops:
|
||||||
@ -58,7 +64,9 @@ def index(request: HttpRequest) -> HttpResponse:
|
|||||||
drop,
|
drop,
|
||||||
)
|
)
|
||||||
|
|
||||||
drop_campaigns = DropCampaign.objects.filter(owner=org)
|
drop_campaigns: BaseManager[DropCampaign] = DropCampaign.objects.filter(
|
||||||
|
owner=org,
|
||||||
|
)
|
||||||
for campaign in drop_campaigns:
|
for campaign in drop_campaigns:
|
||||||
orgs_data[org]["drop_campaigns"].append(campaign)
|
orgs_data[org]["drop_campaigns"].append(campaign)
|
||||||
|
|
||||||
@ -67,3 +75,36 @@ def index(request: HttpRequest) -> HttpResponse:
|
|||||||
template="index.html",
|
template="index.html",
|
||||||
context={"orgs_data": orgs_data},
|
context={"orgs_data": orgs_data},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@require_POST
|
||||||
|
def test_webhook(
|
||||||
|
request: HttpRequest,
|
||||||
|
*args, # noqa: ANN002, ARG001
|
||||||
|
**kwargs, # noqa: ARG001, ANN003
|
||||||
|
) -> HttpResponse:
|
||||||
|
"""Test webhook.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
request: The request.
|
||||||
|
args: Additional arguments.
|
||||||
|
kwargs: Additional keyword arguments.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
HttpResponse: Returns a response.
|
||||||
|
"""
|
||||||
|
org_id: str | None = request.POST.get("org_id")
|
||||||
|
if not org_id:
|
||||||
|
return HttpResponse(status=400)
|
||||||
|
|
||||||
|
campaign: DropCampaign = DropCampaign.objects.get(id=org_id)
|
||||||
|
|
||||||
|
msg: str = f"""
|
||||||
|
Found new drop for {campaign.game.display_name}:\n
|
||||||
|
{campaign.name}\n
|
||||||
|
{campaign.description}\n
|
||||||
|
<{campaign.details_url}>
|
||||||
|
"""
|
||||||
|
send(msg.strip())
|
||||||
|
|
||||||
|
return HttpResponse(status=200)
|
||||||
|
@ -11,3 +11,4 @@ whitenoise[brotli]
|
|||||||
platformdirs
|
platformdirs
|
||||||
playwright
|
playwright
|
||||||
django-ninja
|
django-ninja
|
||||||
|
discord-webhook
|
||||||
|
@ -72,6 +72,15 @@ class DropCampaignSchema(Schema):
|
|||||||
account_link_url: str | None = None
|
account_link_url: str | None = None
|
||||||
description: str | None = None
|
description: str | None = None
|
||||||
details_url: str | None = None
|
details_url: str | None = None
|
||||||
|
end_at: datetime.datetime | None = None
|
||||||
|
image_url: str | None = None
|
||||||
|
name: str | None = None
|
||||||
|
start_at: datetime.datetime | None = None
|
||||||
|
status: str | None = None
|
||||||
|
game: GameSchema | None = None
|
||||||
|
owner: OrganizationSchema | None = None
|
||||||
|
channels: list[ChannelSchema] | None = None
|
||||||
|
time_based_drops: list[TimeBasedDropSchema] | None = None
|
||||||
added_at: datetime.datetime | None = None
|
added_at: datetime.datetime | None = None
|
||||||
modified_at: datetime.datetime | None = None
|
modified_at: datetime.datetime | None = None
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user