forked from github/plane
chore: mention notification and webhook faliure (#3573)
* fix: mention rstrip error * chore: webhook deactivation email * chore: changed template * chore: current site for external api's * chore: mention in template displayed * chore: mention tag fix * chore: comment user name displayed
This commit is contained in:
parent
4563b50fad
commit
76db394ab1
@ -1,6 +1,8 @@
|
|||||||
# Python imports
|
# Python imports
|
||||||
import zoneinfo
|
import zoneinfo
|
||||||
import json
|
import json
|
||||||
|
from urllib.parse import urlparse
|
||||||
|
|
||||||
|
|
||||||
# Django imports
|
# Django imports
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
@ -51,6 +53,11 @@ class WebhookMixin:
|
|||||||
and self.request.method in ["POST", "PATCH", "DELETE"]
|
and self.request.method in ["POST", "PATCH", "DELETE"]
|
||||||
and response.status_code in [200, 201, 204]
|
and response.status_code in [200, 201, 204]
|
||||||
):
|
):
|
||||||
|
url = request.build_absolute_uri()
|
||||||
|
parsed_url = urlparse(url)
|
||||||
|
# Extract the scheme and netloc
|
||||||
|
scheme = parsed_url.scheme
|
||||||
|
netloc = parsed_url.netloc
|
||||||
# Push the object to delay
|
# Push the object to delay
|
||||||
send_webhook.delay(
|
send_webhook.delay(
|
||||||
event=self.webhook_event,
|
event=self.webhook_event,
|
||||||
@ -59,6 +66,7 @@ class WebhookMixin:
|
|||||||
action=self.request.method,
|
action=self.request.method,
|
||||||
slug=self.workspace_slug,
|
slug=self.workspace_slug,
|
||||||
bulk=self.bulk,
|
bulk=self.bulk,
|
||||||
|
current_site=f"{scheme}://{netloc}",
|
||||||
)
|
)
|
||||||
|
|
||||||
return response
|
return response
|
||||||
|
@ -64,6 +64,7 @@ class WebhookMixin:
|
|||||||
action=self.request.method,
|
action=self.request.method,
|
||||||
slug=self.workspace_slug,
|
slug=self.workspace_slug,
|
||||||
bulk=self.bulk,
|
bulk=self.bulk,
|
||||||
|
current_site=request.META.get("HTTP_ORIGIN"),
|
||||||
)
|
)
|
||||||
|
|
||||||
return response
|
return response
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
import json
|
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
|
from bs4 import BeautifulSoup
|
||||||
|
|
||||||
|
|
||||||
# Third party imports
|
# Third party imports
|
||||||
from celery import shared_task
|
from celery import shared_task
|
||||||
@ -9,7 +10,6 @@ from django.utils import timezone
|
|||||||
from django.core.mail import EmailMultiAlternatives, get_connection
|
from django.core.mail import EmailMultiAlternatives, get_connection
|
||||||
from django.template.loader import render_to_string
|
from django.template.loader import render_to_string
|
||||||
from django.utils.html import strip_tags
|
from django.utils.html import strip_tags
|
||||||
from django.conf import settings
|
|
||||||
|
|
||||||
# Module imports
|
# Module imports
|
||||||
from plane.db.models import EmailNotificationLog, User, Issue
|
from plane.db.models import EmailNotificationLog, User, Issue
|
||||||
@ -40,7 +40,7 @@ def stack_email_notification():
|
|||||||
processed_notifications = []
|
processed_notifications = []
|
||||||
# Loop through all the issues to create the emails
|
# Loop through all the issues to create the emails
|
||||||
for receiver_id in receivers:
|
for receiver_id in receivers:
|
||||||
# Notifcation triggered for the receiver
|
# Notification triggered for the receiver
|
||||||
receiver_notifications = [
|
receiver_notifications = [
|
||||||
notification
|
notification
|
||||||
for notification in email_notifications
|
for notification in email_notifications
|
||||||
@ -124,11 +124,29 @@ def create_payload(notification_data):
|
|||||||
|
|
||||||
return data
|
return data
|
||||||
|
|
||||||
|
def process_mention(mention_component):
|
||||||
|
soup = BeautifulSoup(mention_component, 'html.parser')
|
||||||
|
mentions = soup.find_all('mention-component')
|
||||||
|
for mention in mentions:
|
||||||
|
user_id = mention['id']
|
||||||
|
user = User.objects.get(pk=user_id)
|
||||||
|
user_name = user.display_name
|
||||||
|
highlighted_name = f"@{user_name}"
|
||||||
|
mention.replace_with(highlighted_name)
|
||||||
|
return str(soup)
|
||||||
|
|
||||||
|
def process_html_content(content):
|
||||||
|
processed_content_list = []
|
||||||
|
for html_content in content:
|
||||||
|
processed_content = process_mention(html_content)
|
||||||
|
processed_content_list.append(processed_content)
|
||||||
|
return processed_content_list
|
||||||
|
|
||||||
@shared_task
|
@shared_task
|
||||||
def send_email_notification(
|
def send_email_notification(
|
||||||
issue_id, notification_data, receiver_id, email_notification_ids
|
issue_id, notification_data, receiver_id, email_notification_ids
|
||||||
):
|
):
|
||||||
|
try:
|
||||||
ri = redis_instance()
|
ri = redis_instance()
|
||||||
base_api = (ri.get(str(issue_id)).decode())
|
base_api = (ri.get(str(issue_id)).decode())
|
||||||
data = create_payload(notification_data=notification_data)
|
data = create_payload(notification_data=notification_data)
|
||||||
@ -153,6 +171,7 @@ def send_email_notification(
|
|||||||
actor = User.objects.get(pk=actor_id)
|
actor = User.objects.get(pk=actor_id)
|
||||||
total_changes = total_changes + len(changes)
|
total_changes = total_changes + len(changes)
|
||||||
comment = changes.pop("comment", False)
|
comment = changes.pop("comment", False)
|
||||||
|
mention = changes.pop("mention", False)
|
||||||
actors_involved.append(actor_id)
|
actors_involved.append(actor_id)
|
||||||
if comment:
|
if comment:
|
||||||
comments.append(
|
comments.append(
|
||||||
@ -165,6 +184,19 @@ def send_email_notification(
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
if mention:
|
||||||
|
mention["new_value"] = process_html_content(mention.get("new_value"))
|
||||||
|
mention["old_value"] = process_html_content(mention.get("old_value"))
|
||||||
|
comments.append(
|
||||||
|
{
|
||||||
|
"actor_comments": mention,
|
||||||
|
"actor_detail": {
|
||||||
|
"avatar_url": actor.avatar,
|
||||||
|
"first_name": actor.first_name,
|
||||||
|
"last_name": actor.last_name,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
activity_time = changes.pop("activity_time")
|
activity_time = changes.pop("activity_time")
|
||||||
# Parse the input string into a datetime object
|
# Parse the input string into a datetime object
|
||||||
formatted_time = datetime.strptime(activity_time, "%Y-%m-%d %H:%M:%S").strftime("%H:%M %p")
|
formatted_time = datetime.strptime(activity_time, "%Y-%m-%d %H:%M:%S").strftime("%H:%M %p")
|
||||||
@ -240,3 +272,5 @@ def send_email_notification(
|
|||||||
except Exception as e:
|
except Exception as e:
|
||||||
print(e)
|
print(e)
|
||||||
return
|
return
|
||||||
|
except Issue.DoesNotExist:
|
||||||
|
return
|
||||||
|
@ -515,7 +515,7 @@ def notifications(
|
|||||||
bulk_email_logs.append(
|
bulk_email_logs.append(
|
||||||
EmailNotificationLog(
|
EmailNotificationLog(
|
||||||
triggered_by_id=actor_id,
|
triggered_by_id=actor_id,
|
||||||
receiver_id=subscriber,
|
receiver_id=mention_id,
|
||||||
entity_identifier=issue_id,
|
entity_identifier=issue_id,
|
||||||
entity_name="issue",
|
entity_name="issue",
|
||||||
data={
|
data={
|
||||||
@ -552,6 +552,7 @@ def notifications(
|
|||||||
"old_value": str(
|
"old_value": str(
|
||||||
issue_activity.get("old_value")
|
issue_activity.get("old_value")
|
||||||
),
|
),
|
||||||
|
"activity_time": issue_activity.get("created_at"),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
@ -639,6 +640,7 @@ def notifications(
|
|||||||
"old_value": str(
|
"old_value": str(
|
||||||
last_activity.old_value
|
last_activity.old_value
|
||||||
),
|
),
|
||||||
|
"activity_time": issue_activity.get("created_at"),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
@ -695,6 +697,7 @@ def notifications(
|
|||||||
"old_value"
|
"old_value"
|
||||||
)
|
)
|
||||||
),
|
),
|
||||||
|
"activity_time": issue_activity.get("created_at"),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
@ -7,6 +7,9 @@ import hmac
|
|||||||
# Django imports
|
# Django imports
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.core.serializers.json import DjangoJSONEncoder
|
from django.core.serializers.json import DjangoJSONEncoder
|
||||||
|
from django.core.mail import EmailMultiAlternatives, get_connection
|
||||||
|
from django.template.loader import render_to_string
|
||||||
|
from django.utils.html import strip_tags
|
||||||
|
|
||||||
# Third party imports
|
# Third party imports
|
||||||
from celery import shared_task
|
from celery import shared_task
|
||||||
@ -22,10 +25,10 @@ from plane.db.models import (
|
|||||||
ModuleIssue,
|
ModuleIssue,
|
||||||
CycleIssue,
|
CycleIssue,
|
||||||
IssueComment,
|
IssueComment,
|
||||||
|
User,
|
||||||
)
|
)
|
||||||
from plane.api.serializers import (
|
from plane.api.serializers import (
|
||||||
ProjectSerializer,
|
ProjectSerializer,
|
||||||
IssueSerializer,
|
|
||||||
CycleSerializer,
|
CycleSerializer,
|
||||||
ModuleSerializer,
|
ModuleSerializer,
|
||||||
CycleIssueSerializer,
|
CycleIssueSerializer,
|
||||||
@ -34,6 +37,9 @@ from plane.api.serializers import (
|
|||||||
IssueExpandSerializer,
|
IssueExpandSerializer,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
# Module imports
|
||||||
|
from plane.license.utils.instance_value import get_email_configuration
|
||||||
|
|
||||||
SERIALIZER_MAPPER = {
|
SERIALIZER_MAPPER = {
|
||||||
"project": ProjectSerializer,
|
"project": ProjectSerializer,
|
||||||
"issue": IssueExpandSerializer,
|
"issue": IssueExpandSerializer,
|
||||||
@ -72,7 +78,7 @@ def get_model_data(event, event_id, many=False):
|
|||||||
max_retries=5,
|
max_retries=5,
|
||||||
retry_jitter=True,
|
retry_jitter=True,
|
||||||
)
|
)
|
||||||
def webhook_task(self, webhook, slug, event, event_data, action):
|
def webhook_task(self, webhook, slug, event, event_data, action, current_site):
|
||||||
try:
|
try:
|
||||||
webhook = Webhook.objects.get(id=webhook, workspace__slug=slug)
|
webhook = Webhook.objects.get(id=webhook, workspace__slug=slug)
|
||||||
|
|
||||||
@ -151,7 +157,18 @@ def webhook_task(self, webhook, slug, event, event_data, action):
|
|||||||
response_body=str(e),
|
response_body=str(e),
|
||||||
retry_count=str(self.request.retries),
|
retry_count=str(self.request.retries),
|
||||||
)
|
)
|
||||||
|
# Retry logic
|
||||||
|
if self.request.retries >= self.max_retries:
|
||||||
|
Webhook.objects.filter(pk=webhook.id).update(is_active=False)
|
||||||
|
if webhook:
|
||||||
|
# send email for the deactivation of the webhook
|
||||||
|
send_webhook_deactivation_email(
|
||||||
|
webhook_id=webhook.id,
|
||||||
|
receiver_id=webhook.created_by_id,
|
||||||
|
reason=str(e),
|
||||||
|
current_site=current_site,
|
||||||
|
)
|
||||||
|
return
|
||||||
raise requests.RequestException()
|
raise requests.RequestException()
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
@ -162,7 +179,7 @@ def webhook_task(self, webhook, slug, event, event_data, action):
|
|||||||
|
|
||||||
|
|
||||||
@shared_task()
|
@shared_task()
|
||||||
def send_webhook(event, payload, kw, action, slug, bulk):
|
def send_webhook(event, payload, kw, action, slug, bulk, current_site):
|
||||||
try:
|
try:
|
||||||
webhooks = Webhook.objects.filter(workspace__slug=slug, is_active=True)
|
webhooks = Webhook.objects.filter(workspace__slug=slug, is_active=True)
|
||||||
|
|
||||||
@ -216,6 +233,7 @@ def send_webhook(event, payload, kw, action, slug, bulk):
|
|||||||
event=event,
|
event=event,
|
||||||
event_data=data,
|
event_data=data,
|
||||||
action=action,
|
action=action,
|
||||||
|
current_site=current_site,
|
||||||
)
|
)
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
@ -223,3 +241,56 @@ def send_webhook(event, payload, kw, action, slug, bulk):
|
|||||||
print(e)
|
print(e)
|
||||||
capture_exception(e)
|
capture_exception(e)
|
||||||
return
|
return
|
||||||
|
|
||||||
|
|
||||||
|
@shared_task
|
||||||
|
def send_webhook_deactivation_email(webhook_id, receiver_id, current_site, reason):
|
||||||
|
# Get email configurations
|
||||||
|
(
|
||||||
|
EMAIL_HOST,
|
||||||
|
EMAIL_HOST_USER,
|
||||||
|
EMAIL_HOST_PASSWORD,
|
||||||
|
EMAIL_PORT,
|
||||||
|
EMAIL_USE_TLS,
|
||||||
|
EMAIL_FROM,
|
||||||
|
) = get_email_configuration()
|
||||||
|
|
||||||
|
receiver = User.objects.get(pk=receiver_id)
|
||||||
|
webhook = Webhook.objects.get(pk=webhook_id)
|
||||||
|
subject="Webhook Deactivated"
|
||||||
|
message=f"Webhook {webhook.url} has been deactivated due to failed requests."
|
||||||
|
|
||||||
|
# Send the mail
|
||||||
|
context = {
|
||||||
|
"email": receiver.email,
|
||||||
|
"message": message,
|
||||||
|
"webhook_url":f"{current_site}/{str(webhook.workspace.slug)}/settings/webhooks/{str(webhook.id)}",
|
||||||
|
}
|
||||||
|
html_content = render_to_string(
|
||||||
|
"emails/notifications/webhook-deactivate.html", context
|
||||||
|
)
|
||||||
|
text_content = strip_tags(html_content)
|
||||||
|
|
||||||
|
try:
|
||||||
|
connection = get_connection(
|
||||||
|
host=EMAIL_HOST,
|
||||||
|
port=int(EMAIL_PORT),
|
||||||
|
username=EMAIL_HOST_USER,
|
||||||
|
password=EMAIL_HOST_PASSWORD,
|
||||||
|
use_tls=EMAIL_USE_TLS == "1",
|
||||||
|
)
|
||||||
|
|
||||||
|
msg = EmailMultiAlternatives(
|
||||||
|
subject=subject,
|
||||||
|
body=text_content,
|
||||||
|
from_email=EMAIL_FROM,
|
||||||
|
to=[receiver.email],
|
||||||
|
connection=connection,
|
||||||
|
)
|
||||||
|
msg.attach_alternative(html_content, "text/html")
|
||||||
|
msg.send()
|
||||||
|
|
||||||
|
return
|
||||||
|
except Exception as e:
|
||||||
|
print(e)
|
||||||
|
return
|
||||||
|
@ -282,10 +282,8 @@ if REDIS_SSL:
|
|||||||
redis_url = os.environ.get("REDIS_URL")
|
redis_url = os.environ.get("REDIS_URL")
|
||||||
broker_url = f"{redis_url}?ssl_cert_reqs={ssl.CERT_NONE.name}&ssl_ca_certs={certifi.where()}"
|
broker_url = f"{redis_url}?ssl_cert_reqs={ssl.CERT_NONE.name}&ssl_ca_certs={certifi.where()}"
|
||||||
CELERY_BROKER_URL = broker_url
|
CELERY_BROKER_URL = broker_url
|
||||||
CELERY_RESULT_BACKEND = broker_url
|
|
||||||
else:
|
else:
|
||||||
CELERY_BROKER_URL = REDIS_URL
|
CELERY_BROKER_URL = REDIS_URL
|
||||||
CELERY_RESULT_BACKEND = REDIS_URL
|
|
||||||
|
|
||||||
CELERY_IMPORTS = (
|
CELERY_IMPORTS = (
|
||||||
"plane.bgtasks.issue_automation_task",
|
"plane.bgtasks.issue_automation_task",
|
||||||
|
@ -112,21 +112,29 @@
|
|||||||
<p style="font-size: 1rem;color: #1f2d5c; line-height: 28px">
|
<p style="font-size: 1rem;color: #1f2d5c; line-height: 28px">
|
||||||
{{summary}}
|
{{summary}}
|
||||||
<span style="font-size: 1rem; font-weight: 700; line-height: 28px">
|
<span style="font-size: 1rem; font-weight: 700; line-height: 28px">
|
||||||
|
{% if data|length > 0 %}
|
||||||
{{ data.0.actor_detail.first_name}}
|
{{ data.0.actor_detail.first_name}}
|
||||||
{{data.0.actor_detail.last_name}}
|
{{data.0.actor_detail.last_name}}
|
||||||
|
{% else %}
|
||||||
|
{{ comments.0.actor_detail.first_name}}
|
||||||
|
{{comments.0.actor_detail.last_name}}
|
||||||
|
{% endif %}
|
||||||
</span>.
|
</span>.
|
||||||
</p>
|
</p>
|
||||||
{% else %}
|
{% else %}
|
||||||
<p style="font-size: 1rem;color: #1f2d5c; line-height: 28px">
|
<p style="font-size: 1rem;color: #1f2d5c; line-height: 28px">
|
||||||
{{summary}}
|
{{summary}}
|
||||||
<span style="font-size: 1rem; font-weight: 700; line-height: 28px">
|
<span style="font-size: 1rem; font-weight: 700; line-height: 28px">
|
||||||
|
{% if data|length > 0 %}
|
||||||
{{ data.0.actor_detail.first_name}}
|
{{ data.0.actor_detail.first_name}}
|
||||||
{{data.0.actor_detail.last_name}}
|
{{data.0.actor_detail.last_name}}
|
||||||
|
{% else %}
|
||||||
|
{{ comments.0.actor_detail.first_name}}
|
||||||
|
{{comments.0.actor_detail.last_name}}
|
||||||
|
{% endif %}
|
||||||
</span>and others.
|
</span>and others.
|
||||||
</p>
|
</p>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
|
|
||||||
<!-- {% if actors_involved == 1 %}
|
<!-- {% if actors_involved == 1 %}
|
||||||
{% if data|length > 0 and comments|length == 0 %}
|
{% if data|length > 0 and comments|length == 0 %}
|
||||||
<p style="font-size: 1rem;color: #1f2d5c; line-height: 28px">
|
<p style="font-size: 1rem;color: #1f2d5c; line-height: 28px">
|
||||||
|
1544
apiserver/templates/emails/notifications/webhook-deactivate.html
Normal file
1544
apiserver/templates/emails/notifications/webhook-deactivate.html
Normal file
File diff suppressed because it is too large
Load Diff
Loading…
Reference in New Issue
Block a user