2018-02-16 00:47:52 -08:00
|
|
|
// @flow
|
2022-10-25 12:56:38 -04:00
|
|
|
/*
|
|
|
|
Removed Watchman (internal view tracking) code.
|
|
|
|
This file may eventually implement cantina
|
|
|
|
Refer to 0cc0e213a5c5bf9e2a76316df5d9da4b250a13c3 for initial integration commit
|
|
|
|
refer to ___ for removal commit.
|
|
|
|
*/
|
|
|
|
|
2018-09-23 23:44:42 -04:00
|
|
|
import { Lbryio } from 'lbryinc';
|
2020-02-14 13:58:09 -05:00
|
|
|
import * as Sentry from '@sentry/browser';
|
2020-05-29 11:11:50 -04:00
|
|
|
import MatomoTracker from '@datapunt/matomo-tracker-js';
|
2019-04-09 17:21:00 -04:00
|
|
|
import { history } from './store';
|
2019-08-14 12:28:13 -04:00
|
|
|
import Native from 'native';
|
2022-01-02 15:33:11 -05:00
|
|
|
import ElectronCookies from '@meetfranz/electron-cookies';
|
2020-02-11 22:59:30 -06:00
|
|
|
import { generateInitialUrl } from 'util/url';
|
2021-08-10 22:42:50 +02:00
|
|
|
import { MATOMO_ID, MATOMO_URL } from 'config';
|
2018-02-16 00:47:52 -08:00
|
|
|
|
2019-06-19 20:57:51 -04:00
|
|
|
const isProduction = process.env.NODE_ENV === 'production';
|
2021-07-28 11:03:16 -04:00
|
|
|
const devInternalApis = process.env.LBRY_API_URL && process.env.LBRY_API_URL.includes('dev');
|
2019-11-04 13:36:37 -05:00
|
|
|
|
2020-02-24 17:02:03 -05:00
|
|
|
export const SHARE_INTERNAL = 'shareInternal';
|
2020-02-19 01:31:40 -05:00
|
|
|
const SHARE_THIRD_PARTY = 'shareThirdParty';
|
|
|
|
|
2020-05-15 15:12:11 -04:00
|
|
|
if (isProduction) {
|
|
|
|
ElectronCookies.enable({
|
2022-04-27 18:28:09 -04:00
|
|
|
origin: 'https://lbry.tv',
|
2020-05-15 15:12:11 -04:00
|
|
|
});
|
|
|
|
}
|
2019-06-19 20:57:51 -04:00
|
|
|
|
2018-02-16 00:47:52 -08:00
|
|
|
type Analytics = {
|
2021-04-07 01:40:34 -04:00
|
|
|
error: (string) => Promise<any>,
|
2020-05-21 11:38:28 -04:00
|
|
|
sentryError: ({} | string, {}) => Promise<any>,
|
2020-11-16 14:09:00 -05:00
|
|
|
pageView: (string, ?string) => void,
|
2021-04-07 01:40:34 -04:00
|
|
|
setUser: (Object) => void,
|
2020-02-19 01:31:40 -05:00
|
|
|
toggleInternal: (boolean, ?boolean) => void,
|
2019-09-18 14:41:20 -04:00
|
|
|
apiLogView: (string, string, string, ?number, ?() => void) => Promise<any>,
|
2019-10-16 17:36:50 -04:00
|
|
|
apiLogPublish: (ChannelClaim | StreamClaim) => void,
|
2020-03-09 19:46:37 -04:00
|
|
|
apiSyncTags: ({}) => void,
|
2020-05-29 11:11:50 -04:00
|
|
|
tagFollowEvent: (string, boolean, ?string) => void,
|
2020-11-22 16:33:37 -05:00
|
|
|
playerLoadedEvent: (?boolean) => void,
|
2020-11-26 16:06:58 -05:00
|
|
|
playerStartedEvent: (?boolean) => void,
|
2021-10-06 21:59:33 +03: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 16:59:20 -04:00
|
|
|
videoBufferEvent: (
|
|
|
|
StreamClaim,
|
2020-09-09 14:54:51 -04:00
|
|
|
{
|
|
|
|
timeAtBuffer: number,
|
|
|
|
bufferDuration: number,
|
|
|
|
bitRate: number,
|
|
|
|
duration: number,
|
2021-01-05 16:29:04 -05:00
|
|
|
userId: string,
|
2020-09-09 14:54:51 -04:00
|
|
|
playerPoweredBy: string,
|
2020-12-03 14:28:37 -05:00
|
|
|
readyState: number,
|
2020-09-09 14:54:51 -04:00
|
|
|
}
|
2021-08-10 22:42:50 +02:00
|
|
|
) => Promise<any>,
|
2019-08-14 12:28:13 -04:00
|
|
|
emailProvidedEvent: () => void,
|
|
|
|
emailVerifiedEvent: () => void,
|
|
|
|
rewardEligibleEvent: () => void,
|
2019-10-02 14:20:25 -04:00
|
|
|
startupEvent: () => void,
|
2021-04-07 01:40:34 -04:00
|
|
|
purchaseEvent: (number) => void,
|
|
|
|
readyEvent: (number) => void,
|
|
|
|
openUrlEvent: (string) => void,
|
2018-02-23 17:24:00 -07:00
|
|
|
};
|
2018-02-16 00:47:52 -08:00
|
|
|
|
2019-10-16 17:36:50 -04:00
|
|
|
type LogPublishParams = {
|
2019-10-14 19:17:24 -04:00
|
|
|
uri: string,
|
|
|
|
claim_id: string,
|
|
|
|
outpoint: string,
|
2019-10-11 21:55:54 -04:00
|
|
|
channel_claim_id?: string,
|
|
|
|
};
|
|
|
|
|
2022-01-07 14:02:33 -05:00
|
|
|
let internalAnalyticsEnabled: boolean = false;
|
2020-02-19 01:31:40 -05:00
|
|
|
if (window.localStorage.getItem(SHARE_INTERNAL) === 'true') internalAnalyticsEnabled = true;
|
2020-01-02 17:30:58 -05:00
|
|
|
|
2018-02-16 00:47:52 -08:00
|
|
|
const analytics: Analytics = {
|
2021-10-06 21:59:33 +03: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) => {
|
2022-10-25 12:56:38 -04:00
|
|
|
// stub
|
2021-08-10 22:42:50 +02:00
|
|
|
},
|
|
|
|
/**
|
|
|
|
* 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) => {
|
2022-10-25 12:56:38 -04:00
|
|
|
// stub
|
2021-08-10 22:42:50 +02:00
|
|
|
},
|
2022-02-06 22:05:21 -05:00
|
|
|
videoStartEvent: (claimId, timeToStartVideo, poweredBy, passedUserId, canonicalUrl, passedPlayer, videoBitrate) => {
|
2022-01-07 14:02:33 -05:00
|
|
|
// sendPromMetric('time_to_start', duration);
|
2022-02-06 22:05:21 -05:00
|
|
|
sendMatomoEvent('Media', 'TimeToStart', claimId, timeToStartVideo);
|
2021-08-10 22:42:50 +02:00
|
|
|
},
|
2021-04-07 01:40:34 -04:00
|
|
|
error: (message) => {
|
|
|
|
return new Promise((resolve) => {
|
2020-02-19 01:31:40 -05:00
|
|
|
if (internalAnalyticsEnabled && isProduction) {
|
2020-02-14 13:58:09 -05:00
|
|
|
return Lbryio.call('event', 'desktop_error', { error_message: message }).then(() => {
|
|
|
|
resolve(true);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
resolve(false);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
sentryError: (error, errorInfo) => {
|
2021-04-07 01:40:34 -04:00
|
|
|
return new Promise((resolve) => {
|
2020-02-19 01:31:40 -05:00
|
|
|
if (internalAnalyticsEnabled && isProduction) {
|
2021-04-07 01:40:34 -04:00
|
|
|
Sentry.withScope((scope) => {
|
2020-02-14 13:58:09 -05:00
|
|
|
scope.setExtras(errorInfo);
|
|
|
|
const eventId = Sentry.captureException(error);
|
|
|
|
resolve(eventId);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
resolve(null);
|
|
|
|
}
|
|
|
|
});
|
2019-12-19 01:33:06 -05:00
|
|
|
},
|
2020-11-12 21:43:44 -05:00
|
|
|
pageView: (path, search) => {
|
2020-08-12 11:04:38 -04:00
|
|
|
if (internalAnalyticsEnabled) {
|
2020-11-16 14:09:00 -05:00
|
|
|
const params: { href: string, customDimensions?: Array<{ id: number, value: ?string }> } = { href: `${path}` };
|
2020-11-12 21:43:44 -05:00
|
|
|
const dimensions = [];
|
|
|
|
const searchParams = search && new URLSearchParams(search);
|
|
|
|
|
|
|
|
if (searchParams && searchParams.get('src')) {
|
|
|
|
dimensions.push({ id: 1, value: searchParams.get('src') });
|
|
|
|
}
|
|
|
|
if (dimensions.length) {
|
|
|
|
params['customDimensions'] = dimensions;
|
|
|
|
}
|
|
|
|
MatomoInstance.trackPageView(params);
|
2020-05-29 11:11:50 -04:00
|
|
|
}
|
2018-02-16 00:47:52 -08:00
|
|
|
},
|
2021-04-07 01:40:34 -04:00
|
|
|
setUser: (userId) => {
|
2020-06-02 16:52:34 -04:00
|
|
|
if (internalAnalyticsEnabled && userId) {
|
2020-07-06 10:18:01 -04:00
|
|
|
window._paq.push(['setUserId', String(userId)]);
|
2019-08-14 12:28:13 -04:00
|
|
|
Native.getAppVersionInfo().then(({ localVersion }) => {
|
2020-06-02 16:52:34 -04:00
|
|
|
sendMatomoEvent('Version', 'Desktop-Version', localVersion);
|
2019-07-21 22:28:49 -04:00
|
|
|
});
|
|
|
|
}
|
2018-02-16 01:16:50 -08:00
|
|
|
},
|
2020-02-19 01:31:40 -05:00
|
|
|
toggleInternal: (enabled: boolean): void => {
|
|
|
|
internalAnalyticsEnabled = enabled;
|
|
|
|
window.localStorage.setItem(SHARE_INTERNAL, enabled);
|
2018-02-23 17:24:00 -07:00
|
|
|
},
|
2020-02-19 01:31:40 -05:00
|
|
|
|
|
|
|
toggleThirdParty: (enabled: boolean): void => {
|
|
|
|
window.localStorage.setItem(SHARE_THIRD_PARTY, enabled);
|
|
|
|
},
|
|
|
|
|
2019-08-13 23:04:08 -04:00
|
|
|
apiLogView: (uri, outpoint, claimId, timeToStart) => {
|
2019-09-18 14:41:20 -04:00
|
|
|
return new Promise((resolve, reject) => {
|
2020-02-19 01:31:40 -05:00
|
|
|
if (internalAnalyticsEnabled && (isProduction || devInternalApis)) {
|
2019-09-18 14:41:20 -04:00
|
|
|
const params: {
|
|
|
|
uri: string,
|
|
|
|
outpoint: string,
|
|
|
|
claim_id: string,
|
|
|
|
time_to_start?: number,
|
|
|
|
} = {
|
|
|
|
uri,
|
|
|
|
outpoint,
|
|
|
|
claim_id: claimId,
|
|
|
|
};
|
2018-08-03 11:54:10 -04:00
|
|
|
|
2022-01-07 14:02:33 -05:00
|
|
|
if (timeToStart) {
|
2019-09-18 14:41:20 -04:00
|
|
|
params.time_to_start = timeToStart;
|
|
|
|
}
|
2018-08-03 11:54:10 -04:00
|
|
|
|
2019-09-18 14:41:20 -04:00
|
|
|
resolve(Lbryio.call('file', 'view', params));
|
|
|
|
} else {
|
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
});
|
2018-03-07 21:07:42 -08:00
|
|
|
},
|
2019-02-05 13:36:40 -05:00
|
|
|
apiLogSearch: () => {
|
2020-02-19 01:31:40 -05:00
|
|
|
if (internalAnalyticsEnabled && isProduction) {
|
2019-02-05 13:36:40 -05:00
|
|
|
Lbryio.call('event', 'search');
|
|
|
|
}
|
|
|
|
},
|
2019-10-16 17:36:50 -04:00
|
|
|
apiLogPublish: (claimResult: ChannelClaim | StreamClaim) => {
|
2020-07-23 10:22:57 -04:00
|
|
|
// Don't check if this is production so channels created on localhost are still linked to user
|
|
|
|
if (internalAnalyticsEnabled) {
|
2019-10-11 21:55:54 -04: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 17:36:50 -04:00
|
|
|
const params: LogPublishParams = { uri, claim_id: claimId, outpoint };
|
2019-10-11 21:55:54 -04:00
|
|
|
if (channelClaimId) {
|
|
|
|
params['channel_claim_id'] = channelClaimId;
|
|
|
|
}
|
2020-02-23 20:24:09 -05:00
|
|
|
|
2019-10-16 17:36:50 -04:00
|
|
|
Lbryio.call('event', 'publish', params);
|
2019-10-11 21:55:54 -04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2021-04-07 01:40:34 -04:00
|
|
|
apiSyncTags: (params) => {
|
2020-03-09 19:46:37 -04:00
|
|
|
if (internalAnalyticsEnabled && isProduction) {
|
|
|
|
Lbryio.call('content_tags', 'sync', params);
|
|
|
|
}
|
|
|
|
},
|
2021-04-12 12:43:47 -04:00
|
|
|
adsFetchedEvent: () => {
|
|
|
|
sendMatomoEvent('Media', 'AdsFetched');
|
|
|
|
},
|
|
|
|
adsReceivedEvent: (response) => {
|
|
|
|
sendMatomoEvent('Media', 'AdsReceived', JSON.stringify(response));
|
|
|
|
},
|
|
|
|
adsErrorEvent: (response) => {
|
|
|
|
sendMatomoEvent('Media', 'AdsError', JSON.stringify(response));
|
|
|
|
},
|
2021-04-07 01:40:34 -04:00
|
|
|
playerLoadedEvent: (embedded) => {
|
2020-11-22 16:33:37 -05:00
|
|
|
sendMatomoEvent('Player', 'Loaded', embedded ? 'embedded' : 'onsite');
|
|
|
|
},
|
2021-04-07 01:40:34 -04:00
|
|
|
playerStartedEvent: (embedded) => {
|
2020-11-26 16:06:58 -05:00
|
|
|
sendMatomoEvent('Player', 'Started', embedded ? 'embedded' : 'onsite');
|
|
|
|
},
|
2020-05-29 11:11:50 -04:00
|
|
|
tagFollowEvent: (tag, following) => {
|
|
|
|
sendMatomoEvent('Tag', following ? 'Tag-Follow' : 'Tag-Unfollow', tag);
|
2019-07-21 19:35:59 -04:00
|
|
|
},
|
2019-07-08 16:54:58 -04:00
|
|
|
channelBlockEvent: (uri, blocked, location) => {
|
2020-05-29 11:11:50 -04:00
|
|
|
sendMatomoEvent(blocked ? 'Channel-Hidden' : 'Channel-Unhidden', uri);
|
2019-07-08 16:54:58 -04:00
|
|
|
},
|
2019-08-14 12:28:13 -04:00
|
|
|
emailProvidedEvent: () => {
|
2020-05-29 11:11:50 -04:00
|
|
|
sendMatomoEvent('Engagement', 'Email-Provided');
|
2019-08-14 12:28:13 -04:00
|
|
|
},
|
|
|
|
emailVerifiedEvent: () => {
|
2020-05-29 11:11:50 -04:00
|
|
|
sendMatomoEvent('Engagement', 'Email-Verified');
|
2019-08-14 12:28:13 -04:00
|
|
|
},
|
|
|
|
rewardEligibleEvent: () => {
|
2020-05-29 11:11:50 -04:00
|
|
|
sendMatomoEvent('Engagement', 'Reward-Eligible');
|
2019-08-14 12:28:13 -04:00
|
|
|
},
|
2019-11-04 10:55:02 -05:00
|
|
|
openUrlEvent: (url: string) => {
|
2020-05-29 11:11:50 -04:00
|
|
|
sendMatomoEvent('Engagement', 'Open-Url', url);
|
2019-11-04 10:55:02 -05:00
|
|
|
},
|
2020-03-12 12:36:45 -04:00
|
|
|
trendingAlgorithmEvent: (trendingAlgorithm: string) => {
|
2020-06-02 16:52:34 -04:00
|
|
|
sendMatomoEvent('Engagement', 'Trending-Algorithm', trendingAlgorithm);
|
2020-03-12 12:36:45 -04:00
|
|
|
},
|
2019-10-02 14:20:25 -04:00
|
|
|
startupEvent: () => {
|
2020-05-29 11:11:50 -04:00
|
|
|
sendMatomoEvent('Startup', 'Startup');
|
2019-10-02 14:20:25 -04:00
|
|
|
},
|
|
|
|
readyEvent: (timeToReady: number) => {
|
2020-05-29 11:11:50 -04:00
|
|
|
sendMatomoEvent('Startup', 'App-Ready', 'Time', timeToReady);
|
2019-10-02 14:20:25 -04:00
|
|
|
},
|
2020-05-21 11:38:28 -04:00
|
|
|
purchaseEvent: (purchaseInt: number) => {
|
2020-05-29 11:11:50 -04:00
|
|
|
sendMatomoEvent('Purchase', 'Purchase-Complete', 'someLabel', purchaseInt);
|
2020-05-21 11:38:28 -04:00
|
|
|
},
|
2018-02-23 17:24:00 -07:00
|
|
|
};
|
2018-02-16 00:47:52 -08:00
|
|
|
|
2020-05-29 11:11:50 -04:00
|
|
|
function sendMatomoEvent(category, action, name, value) {
|
2020-08-12 11:04:38 -04:00
|
|
|
if (internalAnalyticsEnabled) {
|
2020-05-29 11:11:50 -04:00
|
|
|
const event = { category, action, name, value };
|
|
|
|
MatomoInstance.trackEvent(event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const MatomoInstance = new MatomoTracker({
|
|
|
|
urlBase: MATOMO_URL,
|
|
|
|
siteId: MATOMO_ID, // optional, default value: `1`
|
|
|
|
});
|
|
|
|
|
2022-01-04 15:40:42 -05:00
|
|
|
analytics.pageView(generateInitialUrl(window.location.hash));
|
2019-05-14 16:35:49 -04:00
|
|
|
|
2019-04-01 10:30:19 -04:00
|
|
|
// Listen for url changes and report
|
2019-04-09 17:21:00 -04:00
|
|
|
// This will include search queries
|
2021-04-07 01:40:34 -04:00
|
|
|
history.listen((location) => {
|
2019-04-14 23:49:50 -04:00
|
|
|
const { pathname, search } = location;
|
2019-04-14 01:48:11 -04:00
|
|
|
|
2019-04-14 23:49:50 -04:00
|
|
|
const page = `${pathname}${search}`;
|
2020-11-12 21:43:44 -05:00
|
|
|
analytics.pageView(page, search);
|
2019-04-09 17:21:00 -04:00
|
|
|
});
|
2019-04-01 10:30:19 -04:00
|
|
|
|
2018-02-16 00:47:52 -08:00
|
|
|
export default analytics;
|