From 213e0b7ed908f1791bc376fbb1c42a38154d0b8f Mon Sep 17 00:00:00 2001 From: Rafael Araujo Lehmkuhl Date: Fri, 22 Nov 2024 13:26:42 -0300 Subject: [PATCH] event-tracking: Add event-tracking with PostHog For now, the following are tracked: - App start - Session time - Video recording start/stop - Video recording time - Vehicle arming/disarming --- src/libs/external-telemetry/event-tracking.ts | 34 +++++++++++++++++++ src/main.ts | 3 ++ src/stores/mainVehicle.ts | 4 +++ src/stores/omniscientLogger.ts | 8 +++++ src/stores/video.ts | 6 ++++ 5 files changed, 55 insertions(+) create mode 100644 src/libs/external-telemetry/event-tracking.ts diff --git a/src/libs/external-telemetry/event-tracking.ts b/src/libs/external-telemetry/event-tracking.ts new file mode 100644 index 000000000..87224bef6 --- /dev/null +++ b/src/libs/external-telemetry/event-tracking.ts @@ -0,0 +1,34 @@ +import posthog from 'posthog-js' + +/** + * PostHog client + */ +class PostHog { + static posthog: ReturnType | undefined = undefined + + /** + * Initialize the PostHog client if not already initialized + */ + constructor() { + if (!PostHog.posthog) { + PostHog.posthog = posthog.init('phc_SfqVeZcpYHmhUn9NRizThxFxiI9fKqvjRjmBDB8ToRs', { + api_host: 'https://us.i.posthog.com', + person_profiles: 'always', // Create profiles for anonymous users as well + }) + } + } + + /** + * Capture an event + * @param {string} eventName - The name of the event + * @param {Record} properties - The properties of the event + */ + capture(eventName: string, properties?: Record): void { + if (!PostHog.posthog) { + throw new Error('PostHog client not initialized.') + } + PostHog.posthog.capture(eventName, properties) + } +} + +export const eventTracker = new PostHog() diff --git a/src/main.ts b/src/main.ts index e6a638250..187cbc507 100644 --- a/src/main.ts +++ b/src/main.ts @@ -13,6 +13,7 @@ import { createApp } from 'vue' import VueVirtualScroller from 'vue-virtual-scroller' import { app_version } from '@/libs/cosmos' +import { eventTracker } from '@/libs/external-telemetry/event-tracking' import App from './App.vue' import vuetify from './plugins/vuetify' @@ -25,6 +26,8 @@ loadFonts() const app = createApp(App) +eventTracker.capture('App started') + // Initialize Sentry for error tracking // Only track usage statistics if the user has not opted out and the app is not in development mode if (window.localStorage.getItem('cockpit-enable-usage-statistics-telemetry') && import.meta.env.DEV === false) { diff --git a/src/stores/mainVehicle.ts b/src/stores/mainVehicle.ts index b8617c5df..df5aba762 100644 --- a/src/stores/mainVehicle.ts +++ b/src/stores/mainVehicle.ts @@ -18,6 +18,7 @@ import { ConnectionManager } from '@/libs/connection/connection-manager' import type { Package } from '@/libs/connection/m2r/messages/mavlink2rest' import { MavAutopilot, MAVLinkType, MavType } from '@/libs/connection/m2r/messages/mavlink2rest-enum' import type { Message } from '@/libs/connection/m2r/messages/mavlink2rest-message' +import { eventTracker } from '@/libs/external-telemetry/event-tracking' import { availableCockpitActions, registerActionCallback } from '@/libs/joystick/protocols/cockpit-actions' import { MavlinkManualControlManager } from '@/libs/joystick/protocols/mavlink-manual-control' import type { ArduPilot } from '@/libs/vehicle/ardupilot/ardupilot' @@ -387,6 +388,9 @@ export const useMainVehicleStore = defineStore('main-vehicle', () => { Object.assign(attitude, newAttitude) }) mainVehicle.value.onArm.add((armed: boolean) => { + if (isArmed.value !== armed) { + eventTracker.capture(armed ? 'Vehicle armed' : 'Vehicle disarmed') + } isArmed.value = armed }) mainVehicle.value.onTakeoff.add((inAir: boolean) => { diff --git a/src/stores/omniscientLogger.ts b/src/stores/omniscientLogger.ts index eb087cef4..8e0c89f81 100644 --- a/src/stores/omniscientLogger.ts +++ b/src/stores/omniscientLogger.ts @@ -1,5 +1,6 @@ import { WebRTCStats } from '@peermetrics/webrtc-stats' import { useDocumentVisibility } from '@vueuse/core' +import { differenceInSeconds } from 'date-fns' import { defineStore } from 'pinia' import { ref, watch } from 'vue' @@ -8,6 +9,7 @@ import { createCockpitActionVariable, setCockpitActionVariableData, } from '@/libs/actions/data-lake' +import { eventTracker } from '@/libs/external-telemetry/event-tracking' import { WebRTCStatsEvent, WebRTCVideoStat } from '@/types/video' import { useVideoStore } from './video' @@ -275,6 +277,12 @@ export const useOmniscientLoggerStore = defineStore('omniscient-logger', () => { } }) + // Routine to send a ping event to the event tracking system every 5 minutes + const initialTimestamp = new Date() + setInterval(() => { + eventTracker.capture('Ping', { runningTimeInSeconds: differenceInSeconds(new Date(), initialTimestamp) }) + }, 1000 * 60 * 5) + return { streamsFrameRateHistory, appFrameRateHistory, diff --git a/src/stores/video.ts b/src/stores/video.ts index f6bb3900f..b30a389b4 100644 --- a/src/stores/video.ts +++ b/src/stores/video.ts @@ -14,6 +14,7 @@ import { useBlueOsStorage } from '@/composables/settingsSyncer' import { useSnackbar } from '@/composables/snackbar' import { WebRTCManager } from '@/composables/webRTC' import { getIpsInformationFromVehicle } from '@/libs/blueos' +import { eventTracker } from '@/libs/external-telemetry/event-tracking' import { availableCockpitActions, registerActionCallback } from '@/libs/joystick/protocols/cockpit-actions' import { datalogger } from '@/libs/sensors-logging' import { isEqual, sleep } from '@/libs/utils' @@ -190,6 +191,10 @@ export const useVideoStore = defineStore('video', () => { const stopRecording = (streamName: string): void => { if (activeStreams.value[streamName] === undefined) activateStream(streamName) + const timeRecordingStart = activeStreams.value[streamName]?.timeRecordingStart + const durationInSeconds = timeRecordingStart ? differenceInSeconds(new Date(), timeRecordingStart) : undefined + eventTracker.capture('Video recording stop', { streamName, durationInSeconds }) + activeStreams.value[streamName]!.timeRecordingStart = undefined activeStreams.value[streamName]!.mediaRecorder!.stop() @@ -249,6 +254,7 @@ export const useVideoStore = defineStore('video', () => { * @param {string} streamName - Name of the stream */ const startRecording = async (streamName: string): Promise => { + eventTracker.capture('Video recording start', { streamName: streamName }) if (activeStreams.value[streamName] === undefined) activateStream(streamName) if (namesAvailableStreams.value.isEmpty()) {