* chore: bug fix * dev: changes in api endpoints for invitations and inbox * chore: improvements * dev: update webhook send * dev: webhook validation and fix webhook flow for app * dev: error messages for deactivation * chore: api fixes * dev: update webhook and workspace leave * chore: issue comment * dev: default values for environment variables * dev: make the user active if he was already part of project member * chore: webhook cycle and module event * dev: disable ssl for emails * dev: webhooks restructuring * dev: updated webhook configuration * dev: webhooks * dev: state get object * dev: update workspace slug validation * dev: remove deactivation flag if max retries exceeded --------- Co-authored-by: NarayanBavisetti <narayan3119@gmail.com>
121 lines
4 KiB
Python
121 lines
4 KiB
Python
# Python imports
|
|
import os
|
|
|
|
# Django imports
|
|
from django.core.mail import EmailMultiAlternatives, get_connection
|
|
from django.template.loader import render_to_string
|
|
from django.utils.html import strip_tags
|
|
from django.conf import settings
|
|
|
|
# Third party imports
|
|
from celery import shared_task
|
|
from sentry_sdk import capture_exception
|
|
from slack_sdk import WebClient
|
|
from slack_sdk.errors import SlackApiError
|
|
|
|
# Module imports
|
|
from plane.db.models import Workspace, WorkspaceMemberInvite, User
|
|
from plane.license.models import InstanceConfiguration
|
|
from plane.license.utils.instance_value import get_configuration_value
|
|
|
|
|
|
@shared_task
|
|
def workspace_invitation(email, workspace_id, token, current_site, invitor):
|
|
try:
|
|
user = User.objects.get(email=invitor)
|
|
|
|
workspace = Workspace.objects.get(pk=workspace_id)
|
|
workspace_member_invite = WorkspaceMemberInvite.objects.get(
|
|
token=token, email=email
|
|
)
|
|
|
|
# Relative link
|
|
relative_link = f"/workspace-invitations/?invitation_id={workspace_member_invite.id}&email={email}&slug={workspace.slug}"
|
|
|
|
# The complete url including the domain
|
|
abs_url = current_site + relative_link
|
|
|
|
# Subject of the email
|
|
subject = f"{user.first_name or user.display_name or user.email} invited you to join {workspace.name} on Plane"
|
|
|
|
context = {
|
|
"email": email,
|
|
"first_name": invitor,
|
|
"workspace_name": workspace.name,
|
|
"invitation_url": abs_url,
|
|
}
|
|
|
|
html_content = render_to_string(
|
|
"emails/invitations/workspace_invitation.html", context
|
|
)
|
|
|
|
text_content = strip_tags(html_content)
|
|
|
|
workspace_member_invite.message = text_content
|
|
workspace_member_invite.save()
|
|
|
|
instance_configuration = InstanceConfiguration.objects.filter(
|
|
key__startswith="EMAIL_"
|
|
).values("key", "value")
|
|
connection = get_connection(
|
|
host=get_configuration_value(
|
|
instance_configuration, "EMAIL_HOST", os.environ.get("EMAIL_HOST")
|
|
),
|
|
port=int(
|
|
get_configuration_value(
|
|
instance_configuration, "EMAIL_PORT", os.environ.get("EMAIL_PORT")
|
|
)
|
|
),
|
|
username=get_configuration_value(
|
|
instance_configuration,
|
|
"EMAIL_HOST_USER",
|
|
os.environ.get("EMAIL_HOST_USER"),
|
|
),
|
|
password=get_configuration_value(
|
|
instance_configuration,
|
|
"EMAIL_HOST_PASSWORD",
|
|
os.environ.get("EMAIL_HOST_PASSWORD"),
|
|
),
|
|
use_tls=bool(
|
|
get_configuration_value(
|
|
instance_configuration,
|
|
"EMAIL_USE_TLS",
|
|
os.environ.get("EMAIL_USE_TLS", "1"),
|
|
)
|
|
),
|
|
)
|
|
|
|
msg = EmailMultiAlternatives(
|
|
subject=subject,
|
|
body=text_content,
|
|
from_email=get_configuration_value(
|
|
instance_configuration,
|
|
"EMAIL_FROM",
|
|
os.environ.get("EMAIL_FROM", "Team Plane <team@mailer.plane.so>"),
|
|
),
|
|
to=[email],
|
|
connection=connection,
|
|
)
|
|
msg.attach_alternative(html_content, "text/html")
|
|
msg.send()
|
|
|
|
# Send message on slack as well
|
|
if settings.SLACK_BOT_TOKEN:
|
|
client = WebClient(token=settings.SLACK_BOT_TOKEN)
|
|
try:
|
|
_ = client.chat_postMessage(
|
|
channel="#trackers",
|
|
text=f"{workspace_member_invite.email} has been invited to {workspace.name} as a {workspace_member_invite.role}",
|
|
)
|
|
except SlackApiError as e:
|
|
print(f"Got an error: {e.response['error']}")
|
|
|
|
return
|
|
except (Workspace.DoesNotExist, WorkspaceMemberInvite.DoesNotExist) as e:
|
|
return
|
|
except Exception as e:
|
|
# Print logs if in DEBUG mode
|
|
if settings.DEBUG:
|
|
print(e)
|
|
capture_exception(e)
|
|
return
|