2022-04-19 11:29:36 +05:30
|
|
|
import { Fyo } from 'fyo';
|
|
|
|
import { ConfigKeys } from 'fyo/core/types';
|
2022-03-09 16:24:42 +05:30
|
|
|
import { cloneDeep } from 'lodash';
|
2022-03-18 15:39:17 +05:30
|
|
|
import {
|
|
|
|
getCountry,
|
|
|
|
getCounts,
|
|
|
|
getDeviceId,
|
|
|
|
getInstanceId,
|
|
|
|
getLanguage,
|
|
|
|
} from './helpers';
|
2022-04-18 16:59:20 +05:30
|
|
|
import {
|
|
|
|
Noun,
|
|
|
|
NounEnum,
|
|
|
|
Platform,
|
|
|
|
Telemetry,
|
|
|
|
TelemetrySetting,
|
|
|
|
Verb,
|
|
|
|
} from './types';
|
2022-03-18 15:39:17 +05:30
|
|
|
|
|
|
|
/**
|
|
|
|
* # Telemetry
|
|
|
|
*
|
|
|
|
* ## `start`
|
|
|
|
* Used to initialize state. It should be called before interaction.
|
|
|
|
* It is called on three events:
|
|
|
|
* 1. On db initialization which happens everytime a db is loaded or changed.
|
|
|
|
* 2. On visibility change if not started, eg: when user minimizeds Books and
|
|
|
|
* then comes back later.
|
|
|
|
* 3. When `log` is called if not initialized.
|
|
|
|
*
|
|
|
|
* ## `log`
|
|
|
|
* Used to make entries in the `timeline` which happens only if telmetry
|
|
|
|
* is set to 'Allow Telemetry`
|
|
|
|
*
|
|
|
|
* ## `error`
|
|
|
|
* Called in errorHandling.ts and maintains a count of errors that were
|
|
|
|
* thrown during usage.
|
|
|
|
*
|
|
|
|
* ## `stop`
|
|
|
|
* This is to be called when a session is being stopped. It's called on two events
|
|
|
|
* 1. When the db is being changed.
|
|
|
|
* 2. When the visiblity has changed which happens when either the app is being shut or
|
|
|
|
* the app is hidden.
|
|
|
|
*
|
|
|
|
* This function can't be async as it's called when visibility changes to 'hidden'
|
|
|
|
* at which point async doesn't seem to work and hence count is captured on `start()`
|
|
|
|
*
|
|
|
|
* ## `finalLogAndStop`
|
|
|
|
* Called when telemetry is set to "Don't Log Anything" so as to indicate cessation of
|
|
|
|
* telemetry and not app usage.
|
|
|
|
*/
|
2022-03-09 15:43:17 +05:30
|
|
|
|
2022-04-18 16:59:20 +05:30
|
|
|
export class TelemetryManager {
|
2022-03-14 18:26:26 +05:30
|
|
|
#url: string = '';
|
|
|
|
#token: string = '';
|
2022-03-09 15:43:17 +05:30
|
|
|
#started = false;
|
|
|
|
#telemetryObject: Partial<Telemetry> = {};
|
2022-04-18 16:59:20 +05:30
|
|
|
#interestingDocs: string[] = [];
|
2022-04-19 11:29:36 +05:30
|
|
|
fyo: Fyo;
|
2022-04-18 16:59:20 +05:30
|
|
|
|
2022-04-19 11:29:36 +05:30
|
|
|
constructor(fyo: Fyo) {
|
|
|
|
this.fyo = fyo;
|
2022-04-18 16:59:20 +05:30
|
|
|
}
|
2022-03-09 15:43:17 +05:30
|
|
|
|
2022-03-18 15:39:17 +05:30
|
|
|
set platform(value: Platform) {
|
|
|
|
this.#telemetryObject.platform ||= value;
|
|
|
|
}
|
|
|
|
|
2022-04-18 16:59:20 +05:30
|
|
|
set interestingDocs(schemaNames: string[]) {
|
|
|
|
this.#interestingDocs = schemaNames;
|
|
|
|
}
|
|
|
|
|
2022-03-18 15:39:17 +05:30
|
|
|
get hasCreds() {
|
|
|
|
return !!this.#url && !!this.#token;
|
|
|
|
}
|
|
|
|
|
|
|
|
get started() {
|
|
|
|
return this.#started;
|
|
|
|
}
|
|
|
|
|
|
|
|
get telemetryObject(): Readonly<Partial<Telemetry>> {
|
|
|
|
return cloneDeep(this.#telemetryObject);
|
|
|
|
}
|
|
|
|
|
|
|
|
async start() {
|
2022-04-19 11:29:36 +05:30
|
|
|
this.#telemetryObject.country ||= getCountry(this.fyo);
|
|
|
|
this.#telemetryObject.language ??= getLanguage(this.fyo);
|
|
|
|
this.#telemetryObject.deviceId ||= getDeviceId(this.fyo);
|
|
|
|
this.#telemetryObject.instanceId ||= getInstanceId(this.fyo);
|
2022-03-09 16:24:42 +05:30
|
|
|
this.#telemetryObject.openTime ||= new Date().valueOf();
|
|
|
|
this.#telemetryObject.timeline ??= [];
|
2022-03-09 16:50:48 +05:30
|
|
|
this.#telemetryObject.errors ??= {};
|
2022-03-18 15:39:17 +05:30
|
|
|
this.#telemetryObject.counts ??= {};
|
2022-03-09 15:43:17 +05:30
|
|
|
this.#started = true;
|
|
|
|
|
2022-03-18 15:39:17 +05:30
|
|
|
await this.#postStart();
|
2022-03-14 18:26:26 +05:30
|
|
|
}
|
|
|
|
|
2022-03-18 15:39:17 +05:30
|
|
|
async log(verb: Verb, noun: Noun, more?: Record<string, unknown>) {
|
2022-03-09 15:43:17 +05:30
|
|
|
if (!this.#started) {
|
2022-03-18 15:39:17 +05:30
|
|
|
await this.start();
|
2022-03-09 15:43:17 +05:30
|
|
|
}
|
|
|
|
|
2022-03-18 15:39:17 +05:30
|
|
|
if (!this.#getCanLog()) {
|
2022-03-10 13:21:29 +05:30
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-03-09 15:43:17 +05:30
|
|
|
const time = new Date().valueOf();
|
|
|
|
if (this.#telemetryObject.timeline === undefined) {
|
|
|
|
this.#telemetryObject.timeline = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
this.#telemetryObject.timeline.push({ time, verb, noun, more });
|
|
|
|
}
|
2022-03-09 16:24:42 +05:30
|
|
|
|
2022-03-09 16:50:48 +05:30
|
|
|
error(name: string) {
|
|
|
|
if (this.#telemetryObject.errors === undefined) {
|
|
|
|
this.#telemetryObject.errors = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
this.#telemetryObject.errors[name] ??= 0;
|
|
|
|
this.#telemetryObject.errors[name] += 1;
|
|
|
|
}
|
|
|
|
|
2022-03-10 17:36:28 +05:30
|
|
|
stop() {
|
2022-03-18 15:39:17 +05:30
|
|
|
this.#started = false;
|
2022-03-09 16:24:42 +05:30
|
|
|
|
2022-04-19 11:29:36 +05:30
|
|
|
this.#telemetryObject.version = this.fyo.store.appVersion ?? '';
|
2022-03-09 15:43:17 +05:30
|
|
|
this.#telemetryObject.closeTime = new Date().valueOf();
|
2022-03-10 14:35:47 +05:30
|
|
|
|
2022-03-18 15:39:17 +05:30
|
|
|
const data = JSON.stringify({
|
|
|
|
token: this.#token,
|
|
|
|
telemetryData: this.#telemetryObject,
|
|
|
|
});
|
2022-03-10 17:36:28 +05:30
|
|
|
|
2022-03-18 15:39:17 +05:30
|
|
|
this.#clear();
|
2022-03-10 17:36:28 +05:30
|
|
|
|
2022-04-18 16:59:20 +05:30
|
|
|
if (
|
2022-04-19 11:29:36 +05:30
|
|
|
this.fyo.config.get(ConfigKeys.Telemetry) ===
|
2022-04-18 16:59:20 +05:30
|
|
|
TelemetrySetting.dontLogAnything
|
|
|
|
) {
|
2022-03-15 16:28:43 +05:30
|
|
|
return;
|
2022-03-10 14:35:47 +05:30
|
|
|
}
|
2022-03-15 16:28:43 +05:30
|
|
|
navigator.sendBeacon(this.#url, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
finalLogAndStop() {
|
|
|
|
this.log(Verb.Stopped, NounEnum.Telemetry);
|
|
|
|
this.stop();
|
2022-03-09 15:43:17 +05:30
|
|
|
}
|
2022-03-09 16:24:42 +05:30
|
|
|
|
2022-03-18 15:39:17 +05:30
|
|
|
async #postStart() {
|
|
|
|
await this.#setCount();
|
|
|
|
await this.#setCreds();
|
|
|
|
}
|
|
|
|
|
|
|
|
async #setCount() {
|
|
|
|
if (!this.#getCanLog()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-04-18 16:59:20 +05:30
|
|
|
this.#telemetryObject.counts = await getCounts(
|
|
|
|
this.#interestingDocs,
|
2022-04-19 11:29:36 +05:30
|
|
|
this.fyo
|
2022-04-18 16:59:20 +05:30
|
|
|
);
|
2022-03-18 15:39:17 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
async #setCreds() {
|
|
|
|
if (this.hasCreds) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-04-19 11:29:36 +05:30
|
|
|
const { url, token } = await this.fyo.auth.getTelemetryCreds();
|
2022-03-18 15:39:17 +05:30
|
|
|
this.#url = url;
|
|
|
|
this.#token = token;
|
|
|
|
}
|
|
|
|
|
|
|
|
#getCanLog(): boolean {
|
2022-04-19 11:29:36 +05:30
|
|
|
const telemetrySetting = this.fyo.config.get(
|
2022-04-18 16:59:20 +05:30
|
|
|
ConfigKeys.Telemetry
|
|
|
|
) as string;
|
2022-03-18 15:39:17 +05:30
|
|
|
return telemetrySetting === TelemetrySetting.allow;
|
|
|
|
}
|
|
|
|
|
|
|
|
#clear() {
|
|
|
|
// Delete only what varies
|
|
|
|
delete this.#telemetryObject.openTime;
|
|
|
|
delete this.#telemetryObject.closeTime;
|
|
|
|
delete this.#telemetryObject.errors;
|
|
|
|
delete this.#telemetryObject.counts;
|
|
|
|
delete this.#telemetryObject.timeline;
|
|
|
|
delete this.#telemetryObject.instanceId;
|
|
|
|
delete this.#telemetryObject.country;
|
2022-03-09 16:24:42 +05:30
|
|
|
}
|
2022-03-09 15:43:17 +05:30
|
|
|
}
|