PagerMaid-Pyro/pagermaid/modules/mixpanel.py

123 lines
3.6 KiB
Python
Raw Normal View History

2022-09-07 08:16:37 +00:00
import contextlib
import datetime
import json
import time
import uuid
from pagermaid import Config, logs
2022-08-01 16:04:45 +00:00
from pagermaid.enums import Client, Message
2022-09-07 08:16:37 +00:00
from pagermaid.services import client as request
2022-08-01 16:04:45 +00:00
from pagermaid.hook import Hook
2022-09-07 08:16:37 +00:00
class DatetimeSerializer(json.JSONEncoder):
def default(self, obj):
if isinstance(obj, datetime.datetime):
2023-03-12 03:56:01 +00:00
fmt = "%Y-%m-%dT%H:%M:%S"
2022-09-07 08:16:37 +00:00
return obj.strftime(fmt)
return json.JSONEncoder.default(self, obj)
class Mixpanel:
def __init__(self, token: str):
self._token = token
self._serializer = DatetimeSerializer
self._request = request
self.api_host = "api.mixpanel.com"
@staticmethod
def _now():
return time.time()
@staticmethod
def _make_insert_id():
return uuid.uuid4().hex
@staticmethod
def json_dumps(data, cls=None):
# Separators are specified to eliminate whitespace.
2023-03-12 03:56:01 +00:00
return json.dumps(data, separators=(",", ":"), cls=cls)
2022-09-07 08:16:37 +00:00
async def api_call(self, endpoint, json_message):
_endpoints = {
2023-03-12 03:56:01 +00:00
"events": f"https://{self.api_host}/track",
"people": f"https://{self.api_host}/engage",
2022-09-07 08:16:37 +00:00
}
request_url = _endpoints.get(endpoint)
if request_url is None:
return
params = {
2023-03-12 03:56:01 +00:00
"data": json_message,
"verbose": 1,
"ip": 0,
2022-09-07 08:16:37 +00:00
}
start = self._now()
with contextlib.suppress(Exception):
2023-03-12 03:56:01 +00:00
await self._request.post(request_url, data=params, timeout=10.0)
2022-09-07 08:16:37 +00:00
logs.debug(f"Mixpanel request took {self._now() - start} seconds")
async def people_set(self, distinct_id: str, properties: dict):
message = {
2023-03-12 03:56:01 +00:00
"$distinct_id": distinct_id,
"$set": properties,
2022-09-07 08:16:37 +00:00
}
2023-03-12 03:56:01 +00:00
record = {"$token": self._token, "$time": self._now()}
2022-09-15 08:25:33 +00:00
# sourcery skip: dict-assign-update-to-union
record.update(message)
2023-03-12 03:56:01 +00:00
return await self.api_call(
"people", self.json_dumps(record, cls=self._serializer)
)
2022-09-07 08:16:37 +00:00
async def track(self, distinct_id: str, event_name: str, properties: dict):
all_properties = {
2023-03-12 03:56:01 +00:00
"token": self._token,
"distinct_id": distinct_id,
"time": self._now(),
"$insert_id": self._make_insert_id(),
"mp_lib": "python",
"$lib_version": "4.10.0",
2022-09-07 08:16:37 +00:00
}
if properties:
2022-09-15 08:25:33 +00:00
# sourcery skip: dict-assign-update-to-union
all_properties.update(properties)
2022-09-07 08:16:37 +00:00
event = {
2023-03-12 03:56:01 +00:00
"event": event_name,
"properties": all_properties,
2022-09-07 08:16:37 +00:00
}
2023-03-12 03:56:01 +00:00
return await self.api_call(
"events", self.json_dumps(event, cls=self._serializer)
)
2022-09-07 08:16:37 +00:00
mp = Mixpanel(Config.MIXPANEL_API)
2022-08-01 16:04:45 +00:00
@Hook.on_startup()
async def mixpanel_init_id(bot: Client):
2022-09-07 08:16:37 +00:00
if not bot.me:
bot.me = await bot.get_me()
2023-03-12 03:56:01 +00:00
data = {"$first_name": bot.me.first_name}
2022-09-07 08:16:37 +00:00
if bot.me.username:
data["username"] = bot.me.username
bot.loop.create_task(mp.people_set(str(bot.me.id), data))
2022-08-01 16:04:45 +00:00
@Hook.command_postprocessor()
async def mixpanel_report(bot: Client, message: Message, command):
if not Config.ALLOW_ANALYTIC:
return
2022-09-07 08:16:37 +00:00
if not bot.me:
bot.me = await bot.get_me()
2022-08-01 16:04:45 +00:00
sender_id = message.from_user.id if message.from_user else ""
sender_id = message.sender_chat.id if message.sender_chat else sender_id
if sender_id < 0 and message.outgoing:
2022-09-07 08:16:37 +00:00
sender_id = bot.me.id
2023-03-12 03:56:01 +00:00
bot.loop.create_task(
mp.track(
str(sender_id),
f"Function {command}",
{"command": command, "bot_id": bot.me.id},
)
)