summaryrefslogtreecommitdiff
path: root/server/continuedev/libs/util/telemetry.py
diff options
context:
space:
mode:
Diffstat (limited to 'server/continuedev/libs/util/telemetry.py')
-rw-r--r--server/continuedev/libs/util/telemetry.py108
1 files changed, 108 insertions, 0 deletions
diff --git a/server/continuedev/libs/util/telemetry.py b/server/continuedev/libs/util/telemetry.py
new file mode 100644
index 00000000..1772fe20
--- /dev/null
+++ b/server/continuedev/libs/util/telemetry.py
@@ -0,0 +1,108 @@
+import os
+import socket
+from typing import Any, Dict, Optional
+
+from dotenv import load_dotenv
+
+from ..constants.main import CONTINUE_SERVER_VERSION_FILE
+from .commonregex import clean_pii_from_any
+from .paths import getServerFolderPath
+
+load_dotenv()
+in_codespaces = os.getenv("CODESPACES") == "true"
+POSTHOG_API_KEY = "phc_JS6XFROuNbhJtVCEdTSYk6gl5ArRrTNMpCcguAXlSPs"
+
+
+def is_connected():
+ try:
+ # connect to the host -- tells us if the host is actually reachable
+ socket.create_connection(("www.google.com", 80))
+ return True
+ except OSError:
+ pass
+ return False
+
+
+class PostHogLogger:
+ unique_id: str = "NO_UNIQUE_ID"
+ allow_anonymous_telemetry: bool = False
+ ide_info: Optional[Dict] = None
+ posthog = None
+
+ def __init__(self, api_key: str):
+ self.api_key = api_key
+
+ def setup(
+ self, unique_id: str, allow_anonymous_telemetry: bool, ide_info: Optional[Dict]
+ ):
+ self.unique_id = unique_id or "NO_UNIQUE_ID"
+ self.allow_anonymous_telemetry = allow_anonymous_telemetry or False
+ self.ide_info = ide_info
+
+ # Capture initial event
+ self.capture_event("session_start", {"os": os.name})
+
+ def capture_event(self, event_name: str, event_properties: Any):
+ """Safely capture event. Telemetry should never be the reason Continue doesn't work"""
+ try:
+ self._capture_event(event_name, event_properties)
+ except Exception as e:
+ print(f"Failed to capture event: {e}")
+ pass
+
+ _found_disconnected: bool = False
+
+ def _capture_event(self, event_name: str, event_properties: Any):
+ # logger.debug(
+ # f"Logging to PostHog: {event_name} ({self.unique_id}, {self.allow_anonymous_telemetry}): {event_properties}")
+ telemetry_path = os.path.expanduser("~/.continue/telemetry.log")
+
+ # Make sure the telemetry file exists
+ if not os.path.exists(telemetry_path):
+ os.makedirs(os.path.dirname(telemetry_path), exist_ok=True)
+ open(telemetry_path, "w").close()
+
+ with open(telemetry_path, "a") as f:
+ str_to_write = f"{event_name}: {event_properties}\n{self.unique_id}\n{self.allow_anonymous_telemetry}\n\n"
+ f.write(str_to_write)
+
+ if not self.allow_anonymous_telemetry:
+ return
+
+ # Clean PII from event properties
+ event_properties = clean_pii_from_any(event_properties)
+
+ # Add additional properties that are on every event
+ if in_codespaces:
+ event_properties["codespaces"] = True
+
+ server_version_file = os.path.join(
+ getServerFolderPath(), CONTINUE_SERVER_VERSION_FILE
+ )
+ if os.path.exists(server_version_file):
+ with open(server_version_file, "r") as f:
+ event_properties["server_version"] = f.read()
+
+ # Add operating system
+ event_properties["os"] = os.name
+ if self.ide_info:
+ event_properties["ide_name"] = self.ide_info.get("name", None)
+ event_properties["ide_version"] = self.ide_info.get("version", None)
+ event_properties["ide_remote_name"] = self.ide_info.get("remoteName", None)
+
+ # Send event to PostHog
+ if self.posthog is None:
+ from posthog import Posthog
+
+ # The personal API key is necessary only if you want to use local evaluation of feature flags.
+ self.posthog = Posthog(self.api_key, host="https://app.posthog.com")
+
+ if is_connected():
+ self.posthog.capture(self.unique_id, event_name, event_properties)
+ else:
+ if not self._found_disconnected:
+ self._found_disconnected = True
+ raise ConnectionError("No internet connection")
+
+
+posthog_logger = PostHogLogger(api_key=POSTHOG_API_KEY)