2018-02-16 09:47:52 +01:00
|
|
|
// @flow
|
2018-09-24 05:44:42 +02:00
|
|
|
import { Lbryio } from 'lbryinc';
|
2020-02-14 19:58:09 +01:00
|
|
|
import * as Sentry from '@sentry/browser';
|
2020-03-24 19:43:51 +01:00
|
|
|
import { SDK_API_PATH } from './index';
|
2021-10-06 20:59:33 +02:00
|
|
|
// import getConnectionSpeed from 'util/detect-user-bandwidth';
|
|
|
|
|
|
|
|
// let userDownloadBandwidthInBitsPerSecond;
|
|
|
|
// async function getUserBandwidth() {
|
|
|
|
// userDownloadBandwidthInBitsPerSecond = await getConnectionSpeed();
|
|
|
|
// }
|
|
|
|
|
|
|
|
// get user bandwidth every minute, starting after an initial one minute wait
|
|
|
|
// setInterval(getUserBandwidth, 1000 * 60);
|
2018-02-16 09:47:52 +01:00
|
|
|
|
2019-06-20 02:57:51 +02:00
|
|
|
const isProduction = process.env.NODE_ENV === 'production';
|
2021-07-28 17:03:16 +02:00
|
|
|
const devInternalApis = process.env.LBRY_API_URL && process.env.LBRY_API_URL.includes('dev');
|
2019-11-04 19:36:37 +01:00
|
|
|
|
2020-02-24 23:02:03 +01:00
|
|
|
export const SHARE_INTERNAL = 'shareInternal';
|
2020-02-19 07:31:40 +01:00
|
|
|
|
2021-08-10 22:42:50 +02:00
|
|
|
const WATCHMAN_BACKEND_ENDPOINT = 'https://watchman.na-backend.odysee.com/reports/playback';
|
|
|
|
const SEND_DATA_TO_WATCHMAN_INTERVAL = 10; // in seconds
|
|
|
|
|
2018-02-16 09:47:52 +01:00
|
|
|
type Analytics = {
|
2021-04-07 07:40:34 +02:00
|
|
|
error: (string) => Promise<any>,
|
2020-05-21 17:38:28 +02:00
|
|
|
sentryError: ({} | string, {}) => Promise<any>,
|
2021-04-07 07:40:34 +02:00
|
|
|
setUser: (Object) => void,
|
2020-02-19 07:31:40 +01:00
|
|
|
toggleInternal: (boolean, ?boolean) => void,
|
2019-09-18 20:41:20 +02:00
|
|
|
apiLogView: (string, string, string, ?number, ?() => void) => Promise<any>,
|
2019-10-16 23:36:50 +02:00
|
|
|
apiLogPublish: (ChannelClaim | StreamClaim) => void,
|
2020-03-10 00:46:37 +01:00
|
|
|
apiSyncTags: ({}) => void,
|
2020-05-29 17:11:50 +02:00
|
|
|
tagFollowEvent: (string, boolean, ?string) => void,
|
2020-11-22 22:33:37 +01:00
|
|
|
playerLoadedEvent: (?boolean) => void,
|
2020-11-26 22:06:58 +01:00
|
|
|
playerStartedEvent: (?boolean) => void,
|
2021-10-06 20:59:33 +02:00
|
|
|
videoStartEvent: (string, number, string, number, string, any, number) => void,
|
2021-08-10 22:42:50 +02:00
|
|
|
videoIsPlaying: (boolean, any) => void,
|
2020-08-07 22:59:20 +02:00
|
|
|
videoBufferEvent: (
|
|
|
|
StreamClaim,
|
2020-09-09 20:54:51 +02:00
|
|
|
{
|
|
|
|
timeAtBuffer: number,
|
|
|
|
bufferDuration: number,
|
|
|
|
bitRate: number,
|
|
|
|
duration: number,
|
2021-01-05 22:29:04 +01:00
|
|
|
userId: string,
|
2020-09-09 20:54:51 +02:00
|
|
|
playerPoweredBy: string,
|
2020-12-03 20:28:37 +01:00
|
|
|
readyState: number,
|
2020-09-09 20:54:51 +02:00
|
|
|
}
|
2021-08-10 22:42:50 +02:00
|
|
|
) => Promise<any>,
|
2021-04-12 18:43:47 +02:00
|
|
|
adsFetchedEvent: () => void,
|
|
|
|
adsReceivedEvent: (any) => void,
|
|
|
|
adsErrorEvent: (any) => void,
|
2019-08-14 18:28:13 +02:00
|
|
|
emailProvidedEvent: () => void,
|
|
|
|
emailVerifiedEvent: () => void,
|
|
|
|
rewardEligibleEvent: () => void,
|
2019-10-02 20:20:25 +02:00
|
|
|
startupEvent: () => void,
|
2021-04-07 07:40:34 +02:00
|
|
|
purchaseEvent: (number) => void,
|
|
|
|
readyEvent: (number) => void,
|
|
|
|
openUrlEvent: (string) => void,
|
2018-02-24 01:24:00 +01:00
|
|
|
};
|
2018-02-16 09:47:52 +01:00
|
|
|
|
2019-10-16 23:36:50 +02:00
|
|
|
type LogPublishParams = {
|
2019-10-15 01:17:24 +02:00
|
|
|
uri: string,
|
|
|
|
claim_id: string,
|
|
|
|
outpoint: string,
|
2019-10-12 03:55:54 +02:00
|
|
|
channel_claim_id?: string,
|
|
|
|
};
|
|
|
|
|
2020-02-19 07:31:40 +01:00
|
|
|
let internalAnalyticsEnabled: boolean = IS_WEB || false;
|
2020-06-02 22:52:34 +02:00
|
|
|
// let thirdPartyAnalyticsEnabled: boolean = IS_WEB || false;
|
2020-01-02 23:30:58 +01:00
|
|
|
|
2021-08-10 22:42:50 +02:00
|
|
|
/**
|
|
|
|
* Determine the mobile device type viewing the data
|
|
|
|
* This function returns one of 'and' (Android), 'ios', or 'web'.
|
|
|
|
*
|
|
|
|
* @returns {String}
|
|
|
|
*/
|
|
|
|
function getDeviceType() {
|
2021-08-18 17:42:56 +02:00
|
|
|
// We may not care what the device is if it's in a web browser. Commenting out for now.
|
|
|
|
// if (!IS_WEB) {
|
|
|
|
// return 'elt';
|
|
|
|
// }
|
|
|
|
// const userAgent = navigator.userAgent || navigator.vendor || window.opera;
|
|
|
|
//
|
|
|
|
// if (/android/i.test(userAgent)) {
|
|
|
|
// return 'adr';
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// // iOS detection from: http://stackoverflow.com/a/9039885/177710
|
|
|
|
// if (/iPad|iPhone|iPod/.test(userAgent) && !window.MSStream) {
|
|
|
|
// return 'ios';
|
|
|
|
// }
|
2021-08-10 22:42:50 +02:00
|
|
|
|
|
|
|
// default as web, this can be optimized
|
2021-08-19 19:18:21 +02:00
|
|
|
if (!IS_WEB) {
|
|
|
|
return 'dsk';
|
|
|
|
}
|
2021-08-10 22:42:50 +02:00
|
|
|
return 'web';
|
|
|
|
}
|
|
|
|
// variables initialized for watchman
|
|
|
|
let amountOfBufferEvents = 0;
|
|
|
|
let amountOfBufferTimeInMS = 0;
|
2021-10-06 20:59:33 +02:00
|
|
|
let videoType, userId, claimUrl, playerPoweredBy, videoPlayer, bitrateAsBitsPerSecond;
|
2021-08-10 22:42:50 +02:00
|
|
|
let lastSentTime;
|
|
|
|
|
|
|
|
// calculate data for backend, send them, and reset buffer data for next interval
|
|
|
|
async function sendAndResetWatchmanData() {
|
|
|
|
if (!userId) {
|
|
|
|
return 'Can only be used with a user id';
|
|
|
|
}
|
|
|
|
|
2021-09-03 17:29:23 +02:00
|
|
|
if (!videoPlayer) {
|
|
|
|
return 'Video player not initialized';
|
|
|
|
}
|
|
|
|
|
2021-08-10 22:42:50 +02:00
|
|
|
let timeSinceLastIntervalSend = new Date() - lastSentTime;
|
|
|
|
lastSentTime = new Date();
|
|
|
|
|
|
|
|
let protocol;
|
|
|
|
if (videoType === 'application/x-mpegURL') {
|
|
|
|
protocol = 'hls';
|
2021-10-06 20:59:33 +02:00
|
|
|
// get bandwidth if it exists from the texttrack (so it's accurate if user changes quality)
|
|
|
|
// $FlowFixMe
|
|
|
|
bitrateAsBitsPerSecond = videoPlayer.textTracks?.().tracks_[0]?.activeCues[0]?.value?.bandwidth;
|
2021-08-10 22:42:50 +02:00
|
|
|
} else {
|
|
|
|
protocol = 'stb';
|
|
|
|
}
|
|
|
|
|
|
|
|
// current position in video in MS
|
|
|
|
const positionInVideo = Math.round(videoPlayer.currentTime()) * 1000;
|
|
|
|
|
|
|
|
// get the duration marking the time in the video for relative position calculation
|
|
|
|
const totalDurationInSeconds = Math.round(videoPlayer.duration());
|
|
|
|
|
|
|
|
// build object for watchman backend
|
|
|
|
const objectToSend = {
|
|
|
|
rebuf_count: amountOfBufferEvents,
|
|
|
|
rebuf_duration: amountOfBufferTimeInMS,
|
|
|
|
url: claimUrl.replace('lbry://', ''),
|
|
|
|
device: getDeviceType(),
|
|
|
|
duration: timeSinceLastIntervalSend,
|
|
|
|
protocol,
|
|
|
|
player: playerPoweredBy,
|
|
|
|
user_id: userId.toString(),
|
|
|
|
position: Math.round(positionInVideo),
|
|
|
|
rel_position: Math.round((positionInVideo / (totalDurationInSeconds * 1000)) * 100),
|
2021-10-06 20:59:33 +02:00
|
|
|
bitrate: bitrateAsBitsPerSecond,
|
|
|
|
bandwidth: undefined,
|
|
|
|
// ...(userDownloadBandwidthInBitsPerSecond && {bandwidth: userDownloadBandwidthInBitsPerSecond}), // add bandwidth if populated
|
2021-08-10 22:42:50 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
// post to watchman
|
|
|
|
await sendWatchmanData(objectToSend);
|
|
|
|
|
|
|
|
// reset buffer data
|
|
|
|
amountOfBufferEvents = 0;
|
|
|
|
amountOfBufferTimeInMS = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
let watchmanInterval;
|
|
|
|
// clear watchman interval and mark it as null (when video paused)
|
|
|
|
function stopWatchmanInterval() {
|
|
|
|
clearInterval(watchmanInterval);
|
|
|
|
watchmanInterval = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// creates the setInterval that will run send to watchman on recurring basis
|
|
|
|
function startWatchmanIntervalIfNotRunning() {
|
|
|
|
if (!watchmanInterval) {
|
|
|
|
// instantiate the first time to calculate duration from
|
|
|
|
lastSentTime = new Date();
|
|
|
|
|
|
|
|
// only set an interval if analytics are enabled and is prod
|
2021-08-18 17:42:56 +02:00
|
|
|
if (isProduction && IS_WEB) {
|
2021-08-10 22:42:50 +02:00
|
|
|
watchmanInterval = setInterval(sendAndResetWatchmanData, 1000 * SEND_DATA_TO_WATCHMAN_INTERVAL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// post data to the backend
|
|
|
|
async function sendWatchmanData(body) {
|
|
|
|
try {
|
|
|
|
const response = await fetch(WATCHMAN_BACKEND_ENDPOINT, {
|
|
|
|
method: 'POST',
|
|
|
|
headers: {
|
|
|
|
Accept: 'application/json',
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
},
|
|
|
|
body: JSON.stringify(body),
|
|
|
|
});
|
|
|
|
|
|
|
|
return response;
|
|
|
|
} catch (err) {
|
|
|
|
console.log('ERROR FROM WATCHMAN BACKEND');
|
|
|
|
console.log(err);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-16 09:47:52 +01:00
|
|
|
const analytics: Analytics = {
|
2021-10-06 20:59:33 +02:00
|
|
|
// receive buffer events from tracking plugin and save buffer amounts and times for backend call
|
2021-08-10 22:42:50 +02:00
|
|
|
videoBufferEvent: async (claim, data) => {
|
|
|
|
amountOfBufferEvents = amountOfBufferEvents + 1;
|
|
|
|
amountOfBufferTimeInMS = amountOfBufferTimeInMS + data.bufferDuration;
|
|
|
|
},
|
|
|
|
onDispose: () => {
|
|
|
|
stopWatchmanInterval();
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Is told whether video is being started or paused, and adjusts interval accordingly
|
|
|
|
* @param {boolean} isPlaying - Whether video was started or paused
|
|
|
|
* @param {object} passedPlayer - VideoJS Player object
|
|
|
|
*/
|
|
|
|
videoIsPlaying: (isPlaying, passedPlayer) => {
|
|
|
|
let playerIsSeeking = false;
|
|
|
|
// have to use this because videojs pauses/unpauses during seek
|
|
|
|
// sometimes the seeking function isn't populated yet so check for it as well
|
|
|
|
if (passedPlayer && passedPlayer.seeking) {
|
|
|
|
playerIsSeeking = passedPlayer.seeking();
|
|
|
|
}
|
|
|
|
|
|
|
|
// if being paused, and not seeking, send existing data and stop interval
|
|
|
|
if (!isPlaying && !playerIsSeeking) {
|
|
|
|
sendAndResetWatchmanData();
|
|
|
|
stopWatchmanInterval();
|
|
|
|
// if being told to pause, and seeking, send and restart interval
|
|
|
|
} else if (!isPlaying && playerIsSeeking) {
|
|
|
|
sendAndResetWatchmanData();
|
|
|
|
stopWatchmanInterval();
|
|
|
|
startWatchmanIntervalIfNotRunning();
|
|
|
|
// is being told to play, and seeking, don't do anything,
|
|
|
|
// assume it's been started already from pause
|
|
|
|
} else if (isPlaying && playerIsSeeking) {
|
|
|
|
// start but not a seek, assuming a start from paused content
|
|
|
|
} else if (isPlaying && !playerIsSeeking) {
|
|
|
|
startWatchmanIntervalIfNotRunning();
|
|
|
|
}
|
|
|
|
},
|
2021-10-13 17:04:03 +02:00
|
|
|
videoStartEvent: (claimId, timeToStartVideo, poweredBy, passedUserId, canonicalUrl, passedPlayer, videoBitrate) => {
|
2021-08-10 22:42:50 +02:00
|
|
|
// populate values for watchman when video starts
|
|
|
|
userId = passedUserId;
|
|
|
|
claimUrl = canonicalUrl;
|
|
|
|
playerPoweredBy = poweredBy;
|
|
|
|
|
|
|
|
videoType = passedPlayer.currentSource().type;
|
|
|
|
videoPlayer = passedPlayer;
|
2021-10-06 20:59:33 +02:00
|
|
|
bitrateAsBitsPerSecond = videoBitrate;
|
2021-08-10 22:42:50 +02:00
|
|
|
|
2021-10-13 17:04:03 +02:00
|
|
|
sendPromMetric('time_to_start', timeToStartVideo);
|
2021-10-16 15:13:35 +02:00
|
|
|
sendGaEvent('video_time_to_start', { claim_id: claimId, time: timeToStartVideo });
|
2021-08-10 22:42:50 +02:00
|
|
|
},
|
2021-04-07 07:40:34 +02:00
|
|
|
error: (message) => {
|
|
|
|
return new Promise((resolve) => {
|
2020-02-19 07:31:40 +01:00
|
|
|
if (internalAnalyticsEnabled && isProduction) {
|
2020-02-14 19:58:09 +01:00
|
|
|
return Lbryio.call('event', 'desktop_error', { error_message: message }).then(() => {
|
|
|
|
resolve(true);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
resolve(false);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
sentryError: (error, errorInfo) => {
|
2021-04-07 07:40:34 +02:00
|
|
|
return new Promise((resolve) => {
|
2020-02-19 07:31:40 +01:00
|
|
|
if (internalAnalyticsEnabled && isProduction) {
|
2021-04-07 07:40:34 +02:00
|
|
|
Sentry.withScope((scope) => {
|
2020-02-14 19:58:09 +01:00
|
|
|
scope.setExtras(errorInfo);
|
|
|
|
const eventId = Sentry.captureException(error);
|
|
|
|
resolve(eventId);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
resolve(null);
|
|
|
|
}
|
|
|
|
});
|
2019-12-19 07:33:06 +01:00
|
|
|
},
|
2021-04-07 07:40:34 +02:00
|
|
|
setUser: (userId) => {
|
2021-10-16 15:13:35 +02:00
|
|
|
if (internalAnalyticsEnabled && userId && window.gtag) {
|
|
|
|
window.gtag('set', { user_id: userId });
|
2019-07-22 04:28:49 +02:00
|
|
|
}
|
2018-02-16 10:16:50 +01:00
|
|
|
},
|
2020-02-19 07:31:40 +01:00
|
|
|
toggleInternal: (enabled: boolean): void => {
|
2021-10-16 15:12:52 +02:00
|
|
|
// Always collect analytics on Odysee for now.
|
2018-02-24 01:24:00 +01:00
|
|
|
},
|
2020-02-19 07:31:40 +01:00
|
|
|
|
|
|
|
toggleThirdParty: (enabled: boolean): void => {
|
2021-10-16 15:12:52 +02:00
|
|
|
// Always collect analytics on Odysee for now.
|
2020-02-19 07:31:40 +01:00
|
|
|
},
|
|
|
|
|
2019-08-14 05:04:08 +02:00
|
|
|
apiLogView: (uri, outpoint, claimId, timeToStart) => {
|
2019-09-18 20:41:20 +02:00
|
|
|
return new Promise((resolve, reject) => {
|
2020-02-19 07:31:40 +01:00
|
|
|
if (internalAnalyticsEnabled && (isProduction || devInternalApis)) {
|
2019-09-18 20:41:20 +02:00
|
|
|
const params: {
|
|
|
|
uri: string,
|
|
|
|
outpoint: string,
|
|
|
|
claim_id: string,
|
|
|
|
time_to_start?: number,
|
|
|
|
} = {
|
|
|
|
uri,
|
|
|
|
outpoint,
|
|
|
|
claim_id: claimId,
|
|
|
|
};
|
2018-08-03 17:54:10 +02:00
|
|
|
|
2019-09-18 20:41:20 +02:00
|
|
|
// lbry.tv streams from AWS so we don't care about the time to start
|
|
|
|
if (timeToStart && !IS_WEB) {
|
|
|
|
params.time_to_start = timeToStart;
|
|
|
|
}
|
2018-08-03 17:54:10 +02:00
|
|
|
|
2019-09-18 20:41:20 +02:00
|
|
|
resolve(Lbryio.call('file', 'view', params));
|
|
|
|
} else {
|
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
});
|
2018-03-08 06:07:42 +01:00
|
|
|
},
|
2019-02-05 19:36:40 +01:00
|
|
|
apiLogSearch: () => {
|
2020-02-19 07:31:40 +01:00
|
|
|
if (internalAnalyticsEnabled && isProduction) {
|
2019-02-05 19:36:40 +01:00
|
|
|
Lbryio.call('event', 'search');
|
|
|
|
}
|
|
|
|
},
|
2019-10-16 23:36:50 +02:00
|
|
|
apiLogPublish: (claimResult: ChannelClaim | StreamClaim) => {
|
2020-07-23 16:22:57 +02:00
|
|
|
// Don't check if this is production so channels created on localhost are still linked to user
|
|
|
|
if (internalAnalyticsEnabled) {
|
2019-10-12 03:55:54 +02:00
|
|
|
const { permanent_url: uri, claim_id: claimId, txid, nout, signing_channel: signingChannel } = claimResult;
|
|
|
|
let channelClaimId;
|
|
|
|
if (signingChannel) {
|
|
|
|
channelClaimId = signingChannel.claim_id;
|
|
|
|
}
|
|
|
|
const outpoint = `${txid}:${nout}`;
|
2019-10-16 23:36:50 +02:00
|
|
|
const params: LogPublishParams = { uri, claim_id: claimId, outpoint };
|
2019-10-12 03:55:54 +02:00
|
|
|
if (channelClaimId) {
|
|
|
|
params['channel_claim_id'] = channelClaimId;
|
|
|
|
}
|
2020-02-24 02:24:09 +01:00
|
|
|
|
2019-10-16 23:36:50 +02:00
|
|
|
Lbryio.call('event', 'publish', params);
|
2019-10-12 03:55:54 +02:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2021-04-07 07:40:34 +02:00
|
|
|
apiSyncTags: (params) => {
|
2020-03-10 00:46:37 +01:00
|
|
|
if (internalAnalyticsEnabled && isProduction) {
|
|
|
|
Lbryio.call('content_tags', 'sync', params);
|
|
|
|
}
|
|
|
|
},
|
2021-04-12 18:43:47 +02:00
|
|
|
adsFetchedEvent: () => {
|
2021-10-16 15:13:35 +02:00
|
|
|
sendGaEvent('ad_fetched');
|
2021-04-12 18:43:47 +02:00
|
|
|
},
|
|
|
|
adsReceivedEvent: (response) => {
|
2021-10-16 15:13:35 +02:00
|
|
|
sendGaEvent('ad_received', { response: JSON.stringify(response) });
|
2021-04-12 18:43:47 +02:00
|
|
|
},
|
|
|
|
adsErrorEvent: (response) => {
|
2021-10-16 15:13:35 +02:00
|
|
|
sendGaEvent('ad_error', { response: JSON.stringify(response) });
|
2021-04-12 18:43:47 +02:00
|
|
|
},
|
2021-04-07 07:40:34 +02:00
|
|
|
playerLoadedEvent: (embedded) => {
|
2021-10-16 15:13:35 +02:00
|
|
|
sendGaEvent('player', { action: 'loaded', type: embedded ? 'embedded' : 'onsite' });
|
2020-11-22 22:33:37 +01:00
|
|
|
},
|
2021-04-07 07:40:34 +02:00
|
|
|
playerStartedEvent: (embedded) => {
|
2021-10-16 15:13:35 +02:00
|
|
|
sendGaEvent('player', { action: 'started', type: embedded ? 'embedded' : 'onsite' });
|
2020-11-26 22:06:58 +01:00
|
|
|
},
|
2020-05-29 17:11:50 +02:00
|
|
|
tagFollowEvent: (tag, following) => {
|
2021-10-16 15:13:35 +02:00
|
|
|
sendGaEvent(following ? 'tag_follow' : 'tag_unfollow', { tag });
|
2019-07-22 01:35:59 +02:00
|
|
|
},
|
2019-07-08 22:54:58 +02:00
|
|
|
channelBlockEvent: (uri, blocked, location) => {
|
2021-10-16 15:13:35 +02:00
|
|
|
sendGaEvent(blocked ? 'channel_hidden' : 'channel_unhidden', { uri });
|
2019-07-08 22:54:58 +02:00
|
|
|
},
|
2019-08-14 18:28:13 +02:00
|
|
|
emailProvidedEvent: () => {
|
2021-10-16 15:13:35 +02:00
|
|
|
sendGaEvent('engagement', { type: 'email_provided' });
|
2019-08-14 18:28:13 +02:00
|
|
|
},
|
|
|
|
emailVerifiedEvent: () => {
|
2021-10-16 15:13:35 +02:00
|
|
|
sendGaEvent('engagement', { type: 'email_verified' });
|
2019-08-14 18:28:13 +02:00
|
|
|
},
|
|
|
|
rewardEligibleEvent: () => {
|
2021-10-16 15:13:35 +02:00
|
|
|
sendGaEvent('engagement', { type: 'reward_eligible' });
|
2019-08-14 18:28:13 +02:00
|
|
|
},
|
2019-11-04 16:55:02 +01:00
|
|
|
openUrlEvent: (url: string) => {
|
2021-10-16 15:13:35 +02:00
|
|
|
sendGaEvent('engagement', { type: 'open_url', url });
|
2019-11-04 16:55:02 +01:00
|
|
|
},
|
2020-03-12 17:36:45 +01:00
|
|
|
trendingAlgorithmEvent: (trendingAlgorithm: string) => {
|
2021-10-16 15:13:35 +02:00
|
|
|
sendGaEvent('engagement', { type: 'trending_algorithm', trending_algorithm: trendingAlgorithm });
|
2020-03-12 17:36:45 +01:00
|
|
|
},
|
2019-10-02 20:20:25 +02:00
|
|
|
startupEvent: () => {
|
2021-10-16 15:13:35 +02:00
|
|
|
// TODO: This can be removed (use the automated 'session_start' instead).
|
|
|
|
// sendGaEvent('startup', 'startup');
|
2019-10-02 20:20:25 +02:00
|
|
|
},
|
2021-10-16 15:13:35 +02:00
|
|
|
readyEvent: (timeToReadyMs: number) => {
|
|
|
|
sendGaEvent('startup_app_ready', { time_to_ready_ms: timeToReadyMs });
|
2019-10-02 20:20:25 +02:00
|
|
|
},
|
2020-05-21 17:38:28 +02:00
|
|
|
purchaseEvent: (purchaseInt: number) => {
|
2021-10-16 15:13:35 +02:00
|
|
|
// https://developers.google.com/analytics/devguides/collection/ga4/reference/events#purchase
|
|
|
|
sendGaEvent('purchase', { value: purchaseInt });
|
2020-05-21 17:38:28 +02:00
|
|
|
},
|
2018-02-24 01:24:00 +01:00
|
|
|
};
|
2018-02-16 09:47:52 +01:00
|
|
|
|
2021-10-16 15:13:35 +02:00
|
|
|
function sendGaEvent(event: string, params?: { [string]: string | number }) {
|
|
|
|
if (internalAnalyticsEnabled && isProduction && window.gtag) {
|
|
|
|
window.gtag('event', event, params);
|
2020-05-29 17:11:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-23 22:16:28 +01:00
|
|
|
function sendPromMetric(name: string, value?: number) {
|
|
|
|
if (IS_WEB) {
|
2020-03-24 19:43:51 +01:00
|
|
|
let url = new URL(SDK_API_PATH + '/metric/ui');
|
2020-03-23 22:16:28 +01:00
|
|
|
const params = { name: name, value: value ? value.toString() : '' };
|
|
|
|
url.search = new URLSearchParams(params).toString();
|
2020-03-24 20:31:00 +01:00
|
|
|
return fetch(url, { method: 'post' });
|
2020-03-23 22:16:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-16 15:13:35 +02:00
|
|
|
// Activate
|
|
|
|
if (internalAnalyticsEnabled && isProduction && window.gtag) {
|
|
|
|
window.gtag('consent', 'update', {
|
|
|
|
ad_storage: 'granted',
|
|
|
|
analytics_storage: 'granted',
|
|
|
|
});
|
|
|
|
}
|
2019-04-01 16:30:19 +02:00
|
|
|
|
2018-02-16 09:47:52 +01:00
|
|
|
export default analytics;
|