mirror of
https://github.com/MODSetter/SurfSense.git
synced 2026-05-31 19:45:15 +02:00
- Added tracking for automation creation, updates, deletions, and trigger modifications, including success and failure events. - Implemented event tracking in the automation creation process, including chat approval and rejection scenarios. - Updated the instrumentation client to ensure correct typing for PostHog integration. - Refactored existing mutation atoms to include tracking calls for automation-related actions, improving analytics capabilities.
117 lines
3.2 KiB
TypeScript
117 lines
3.2 KiB
TypeScript
import posthog from "posthog-js";
|
||
|
||
/**
|
||
* PostHog initialisation for the Next.js renderer.
|
||
*
|
||
* The same bundle ships in two contexts:
|
||
* 1. A normal browser session on surfsense.com -> platform = "web"
|
||
* 2. The Electron desktop app (renders the Next app from localhost)
|
||
* -> platform = "desktop"
|
||
*
|
||
* When running inside Electron we also seed `posthog-js` with the main
|
||
* process's machine distinctId so that events fired from both the renderer
|
||
* (e.g. `chat_message_sent`, page views) and the Electron main process
|
||
* (e.g. `desktop_quick_ask_opened`) share a single PostHog person before
|
||
* login, and can be merged into the authenticated user afterwards.
|
||
*/
|
||
|
||
function isElectron(): boolean {
|
||
return typeof window !== "undefined" && !!window.electronAPI;
|
||
}
|
||
|
||
function currentPlatform(): "desktop" | "web" {
|
||
return isElectron() ? "desktop" : "web";
|
||
}
|
||
|
||
async function resolveBootstrapDistinctId(): Promise<string | undefined> {
|
||
if (!isElectron() || !window.electronAPI?.getAnalyticsContext) return undefined;
|
||
try {
|
||
const ctx = await window.electronAPI.getAnalyticsContext();
|
||
return ctx?.machineId || ctx?.distinctId || undefined;
|
||
} catch {
|
||
return undefined;
|
||
}
|
||
}
|
||
|
||
async function initPostHog() {
|
||
try {
|
||
if (!process.env.NEXT_PUBLIC_POSTHOG_KEY) return;
|
||
|
||
const platform = currentPlatform();
|
||
const bootstrapDistinctId = await resolveBootstrapDistinctId();
|
||
|
||
posthog.init(process.env.NEXT_PUBLIC_POSTHOG_KEY, {
|
||
api_host: "https://assets.surfsense.com",
|
||
ui_host: "https://us.posthog.com",
|
||
defaults: "2026-01-30",
|
||
capture_pageview: "history_change",
|
||
capture_pageleave: true,
|
||
...(bootstrapDistinctId
|
||
? {
|
||
bootstrap: {
|
||
distinctID: bootstrapDistinctId,
|
||
isIdentifiedID: false,
|
||
},
|
||
}
|
||
: {}),
|
||
before_send: (event) => {
|
||
if (event?.properties) {
|
||
event.properties.platform = platform;
|
||
if (platform === "desktop") {
|
||
event.properties.is_desktop = true;
|
||
}
|
||
|
||
const params = new URLSearchParams(window.location.search);
|
||
const ref = params.get("ref");
|
||
if (ref) {
|
||
event.properties.ref_code = ref;
|
||
event.properties.$set = {
|
||
...event.properties.$set,
|
||
initial_ref_code: ref,
|
||
};
|
||
event.properties.$set_once = {
|
||
...event.properties.$set_once,
|
||
first_ref_code: ref,
|
||
};
|
||
}
|
||
|
||
event.properties.$set = {
|
||
...event.properties.$set,
|
||
platform,
|
||
last_seen_at: new Date().toISOString(),
|
||
};
|
||
|
||
event.properties.$set_once = {
|
||
...event.properties.$set_once,
|
||
first_seen_platform: platform,
|
||
};
|
||
}
|
||
return event;
|
||
},
|
||
loaded: () => {
|
||
if (typeof window !== "undefined") {
|
||
// `loaded` hands back a `PostHogInterface`, but it's the same
|
||
// singleton as the default import (typed `PostHog`); use that to
|
||
// keep `window.posthog` correctly typed.
|
||
window.posthog = posthog;
|
||
}
|
||
},
|
||
});
|
||
} catch {
|
||
// PostHog init failed (likely ad-blocker) – app must continue to work
|
||
}
|
||
}
|
||
|
||
if (typeof window !== "undefined") {
|
||
window.posthog = posthog;
|
||
|
||
if ("requestIdleCallback" in window) {
|
||
requestIdleCallback(() => {
|
||
void initPostHog();
|
||
});
|
||
} else {
|
||
setTimeout(() => {
|
||
void initPostHog();
|
||
}, 3500);
|
||
}
|
||
}
|