mirror of
https://github.com/PostHog/posthog.git
synced 2024-11-22 17:24:15 +01:00
008ee1f04c
* Use `statshog` over python-statsd More support for tags! * Include custom tags for every query + add annotation to query After this we can: - Figure out from query logs where queries are coming from (speeding up debugging) - Break down query speeds by user queries vs others (e.g. celery) -- better represents overall speed - Can figure out how fast queries are on average for various teams * Use tags in more queries over interpolation This way we can set up more interesting graphs \o/ * Solve mypy error * Fix a flaky test (due to ordering)
82 lines
3.2 KiB
Python
82 lines
3.2 KiB
Python
from typing import Any, Dict, Sequence, cast
|
|
|
|
import requests
|
|
from celery import Task
|
|
from django.conf import settings
|
|
from django.db.utils import DataError
|
|
from sentry_sdk import capture_exception
|
|
from statshog.defaults.django import statsd
|
|
|
|
from ee.clickhouse.models.element import chain_to_elements
|
|
from posthog.celery import app
|
|
from posthog.models import Action, Event, Team
|
|
from posthog.tasks.webhooks import determine_webhook_type, get_formatted_message
|
|
|
|
|
|
@app.task(ignore_result=True, bind=True, max_retries=3)
|
|
def post_event_to_webhook_ee(self: Task, event: Dict[str, Any], team_id: int, site_url: str) -> None:
|
|
if not site_url:
|
|
site_url = settings.SITE_URL
|
|
|
|
timer = statsd.timer("posthog_cloud_hooks_processed_for_event").start()
|
|
|
|
team = Team.objects.select_related("organization").get(pk=team_id)
|
|
|
|
elements_list = chain_to_elements(event.get("elements_chain", ""))
|
|
ephemeral_postgres_event = Event.objects.create(
|
|
event=event["event"],
|
|
distinct_id=event["distinct_id"],
|
|
properties=event["properties"],
|
|
team=team,
|
|
site_url=site_url,
|
|
**({"timestamp": event["timestamp"]} if event["timestamp"] else {}),
|
|
**({"elements": elements_list})
|
|
)
|
|
|
|
try:
|
|
is_zapier_available = team.organization.is_feature_available("zapier")
|
|
|
|
actionFilters = {"team_id": team_id, "deleted": False}
|
|
if not is_zapier_available:
|
|
if not team.slack_incoming_webhook:
|
|
return # Exit this task if neither Zapier nor webhook URL are available
|
|
else:
|
|
actionFilters["post_to_slack"] = True # We only need to fire for actions that are posted to webhook URL
|
|
|
|
for action in cast(Sequence[Action], Action.objects.filter(**actionFilters).all()):
|
|
try:
|
|
# Wrapped in len to evaluate right away
|
|
qs = len(Event.objects.filter(pk=ephemeral_postgres_event.pk).query_db_by_action(action))
|
|
except DataError as e:
|
|
# Ignore invalid regex errors, which are user mistakes
|
|
if not "invalid regular expression" in str(e):
|
|
capture_exception(e)
|
|
continue
|
|
except:
|
|
capture_exception()
|
|
continue
|
|
if not qs:
|
|
continue
|
|
# REST hooks
|
|
if is_zapier_available:
|
|
action.on_perform(ephemeral_postgres_event)
|
|
# webhooks
|
|
if team.slack_incoming_webhook and action.post_to_slack:
|
|
message_text, message_markdown = get_formatted_message(action, ephemeral_postgres_event, site_url)
|
|
if determine_webhook_type(team) == "slack":
|
|
message = {
|
|
"text": message_text,
|
|
"blocks": [{"type": "section", "text": {"type": "mrkdwn", "text": message_markdown}}],
|
|
}
|
|
else:
|
|
message = {
|
|
"text": message_markdown,
|
|
}
|
|
statsd.incr("posthog_cloud_hooks_web_fired")
|
|
requests.post(team.slack_incoming_webhook, verify=False, json=message)
|
|
except:
|
|
raise
|
|
finally:
|
|
timer.stop()
|
|
ephemeral_postgres_event.delete()
|