mirror of
				https://github.com/zulip/zulip.git
				synced 2025-11-03 21:43:21 +00:00 
			
		
		
		
	The idea behind doing this is that we would rather let the code error out rather than add to the logs. It's webhook code usually never uses the logging module so this section of legacy code needed to be changed or removed. Assists PR #15942. Signed-off-by: Hemanth V. Alluri <hdrive1999@gmail.com>
		
			
				
	
	
		
			154 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			154 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
"""Webhooks for external integrations."""
 | 
						|
from typing import Any, Dict, List
 | 
						|
 | 
						|
from django.http import HttpRequest, HttpResponse
 | 
						|
 | 
						|
from zerver.decorator import authenticated_rest_api_view
 | 
						|
from zerver.lib.email_notifications import convert_html_to_markdown
 | 
						|
from zerver.lib.request import REQ, has_request_variables
 | 
						|
from zerver.lib.response import json_success
 | 
						|
from zerver.lib.webhooks.common import check_send_webhook_message
 | 
						|
from zerver.models import UserProfile
 | 
						|
 | 
						|
NOTE_TEMPLATE = "{name} <{email}> added a {note_type} note to [ticket #{ticket_id}]({ticket_url})."
 | 
						|
PROPERTY_CHANGE_TEMPLATE = """
 | 
						|
{name} <{email}> updated [ticket #{ticket_id}]({ticket_url}):
 | 
						|
 | 
						|
* **{property_name}**: {old} -> {new}
 | 
						|
""".strip()
 | 
						|
TICKET_CREATION_TEMPLATE = """
 | 
						|
{name} <{email}> created [ticket #{ticket_id}]({ticket_url}):
 | 
						|
 | 
						|
``` quote
 | 
						|
{description}
 | 
						|
```
 | 
						|
 | 
						|
* **Type**: {type}
 | 
						|
* **Priority**: {priority}
 | 
						|
* **Status**: {status}
 | 
						|
""".strip()
 | 
						|
 | 
						|
class TicketDict(Dict[str, Any]):
 | 
						|
    """
 | 
						|
    A helper class to turn a dictionary with ticket information into
 | 
						|
    an object where each of the keys is an attribute for easy access.
 | 
						|
    """
 | 
						|
 | 
						|
    def __getattr__(self, field: str) -> Any:
 | 
						|
        if "_" in field:
 | 
						|
            return self.get(field)
 | 
						|
        else:
 | 
						|
            return self.get("ticket_" + field)
 | 
						|
 | 
						|
 | 
						|
def property_name(property: str, index: int) -> str:
 | 
						|
    """The Freshdesk API is currently pretty broken: statuses are customizable
 | 
						|
    but the API will only tell you the number associated with the status, not
 | 
						|
    the name. While we engage the Freshdesk developers about exposing this
 | 
						|
    information through the API, since only FlightCar uses this integration,
 | 
						|
    hardcode their statuses.
 | 
						|
    """
 | 
						|
    statuses = ["", "", "Open", "Pending", "Resolved", "Closed",
 | 
						|
                "Waiting on Customer", "Job Application", "Monthly"]
 | 
						|
    priorities = ["", "Low", "Medium", "High", "Urgent"]
 | 
						|
 | 
						|
    name = ""
 | 
						|
    if property == "status":
 | 
						|
        name = statuses[index] if index < len(statuses) else str(index)
 | 
						|
    elif property == "priority":
 | 
						|
        name = priorities[index] if index < len(priorities) else str(index)
 | 
						|
 | 
						|
    return name
 | 
						|
 | 
						|
 | 
						|
def parse_freshdesk_event(event_string: str) -> List[str]:
 | 
						|
    """These are always of the form "{ticket_action:created}" or
 | 
						|
    "{status:{from:4,to:6}}". Note the lack of string quoting: this isn't
 | 
						|
    valid JSON so we have to parse it ourselves.
 | 
						|
    """
 | 
						|
    data = event_string.replace("{", "").replace("}", "").replace(",", ":").split(":")
 | 
						|
 | 
						|
    if len(data) == 2:
 | 
						|
        # This is a simple ticket action event, like
 | 
						|
        # {ticket_action:created}.
 | 
						|
        return data
 | 
						|
    else:
 | 
						|
        # This is a property change event, like {status:{from:4,to:6}}. Pull out
 | 
						|
        # the property, from, and to states.
 | 
						|
        property, _, from_state, _, to_state = data
 | 
						|
        return [property, property_name(property, int(from_state)),
 | 
						|
                property_name(property, int(to_state))]
 | 
						|
 | 
						|
 | 
						|
def format_freshdesk_note_message(ticket: TicketDict, event_info: List[str]) -> str:
 | 
						|
    """There are public (visible to customers) and private note types."""
 | 
						|
    note_type = event_info[1]
 | 
						|
    content = NOTE_TEMPLATE.format(
 | 
						|
        name=ticket.requester_name,
 | 
						|
        email=ticket.requester_email,
 | 
						|
        note_type=note_type,
 | 
						|
        ticket_id=ticket.id,
 | 
						|
        ticket_url=ticket.url,
 | 
						|
    )
 | 
						|
 | 
						|
    return content
 | 
						|
 | 
						|
 | 
						|
def format_freshdesk_property_change_message(ticket: TicketDict, event_info: List[str]) -> str:
 | 
						|
    """Freshdesk will only tell us the first event to match our webhook
 | 
						|
    configuration, so if we change multiple properties, we only get the before
 | 
						|
    and after data for the first one.
 | 
						|
    """
 | 
						|
    content = PROPERTY_CHANGE_TEMPLATE.format(
 | 
						|
        name=ticket.requester_name,
 | 
						|
        email=ticket.requester_email,
 | 
						|
        ticket_id=ticket.id,
 | 
						|
        ticket_url=ticket.url,
 | 
						|
        property_name=event_info[0].capitalize(),
 | 
						|
        old=event_info[1],
 | 
						|
        new=event_info[2],
 | 
						|
    )
 | 
						|
 | 
						|
    return content
 | 
						|
 | 
						|
 | 
						|
def format_freshdesk_ticket_creation_message(ticket: TicketDict) -> str:
 | 
						|
    """They send us the description as HTML."""
 | 
						|
    cleaned_description = convert_html_to_markdown(ticket.description)
 | 
						|
    content = TICKET_CREATION_TEMPLATE.format(
 | 
						|
        name=ticket.requester_name,
 | 
						|
        email=ticket.requester_email,
 | 
						|
        ticket_id=ticket.id,
 | 
						|
        ticket_url=ticket.url,
 | 
						|
        description=cleaned_description,
 | 
						|
        type=ticket.type,
 | 
						|
        priority=ticket.priority,
 | 
						|
        status=ticket.status,
 | 
						|
    )
 | 
						|
 | 
						|
    return content
 | 
						|
 | 
						|
@authenticated_rest_api_view(webhook_client_name="Freshdesk")
 | 
						|
@has_request_variables
 | 
						|
def api_freshdesk_webhook(request: HttpRequest, user_profile: UserProfile,
 | 
						|
                          payload: Dict[str, Any]=REQ(argument_type='body')) -> HttpResponse:
 | 
						|
    ticket_data = payload["freshdesk_webhook"]
 | 
						|
 | 
						|
    ticket = TicketDict(ticket_data)
 | 
						|
 | 
						|
    subject = f"#{ticket.id}: {ticket.subject}"
 | 
						|
    event_info = parse_freshdesk_event(ticket.triggered_event)
 | 
						|
 | 
						|
    if event_info[1] == "created":
 | 
						|
        content = format_freshdesk_ticket_creation_message(ticket)
 | 
						|
    elif event_info[0] == "note_type":
 | 
						|
        content = format_freshdesk_note_message(ticket, event_info)
 | 
						|
    elif event_info[0] in ("status", "priority"):
 | 
						|
        content = format_freshdesk_property_change_message(ticket, event_info)
 | 
						|
    else:
 | 
						|
        # Not an event we know handle; do nothing.
 | 
						|
        return json_success()
 | 
						|
 | 
						|
    check_send_webhook_message(request, user_profile, subject, content)
 | 
						|
    return json_success()
 |