2022-04-19 11:29:36 +05:30
|
|
|
import { Fyo } from 'fyo';
|
2022-07-18 16:43:56 +05:30
|
|
|
import { Noun, Telemetry, Verb } from './types';
|
2023-06-28 10:25:39 +05:30
|
|
|
import { ModelNameEnum } from 'models/types';
|
2022-03-18 15:39:17 +05:30
|
|
|
|
|
|
|
/**
|
|
|
|
* # Telemetry
|
2022-05-23 13:39:07 +05:30
|
|
|
* Used to check if people are using Books or not. All logging
|
|
|
|
* happens using navigator.sendBeacon
|
2022-03-18 15:39:17 +05:30
|
|
|
*
|
|
|
|
* ## `start`
|
2022-05-23 13:39:07 +05:30
|
|
|
* Used to initialize state. It should be called before any logging and after an
|
|
|
|
* instance has loaded.
|
2022-03-18 15:39:17 +05:30
|
|
|
* It is called on three events:
|
2022-07-18 16:43:56 +05:30
|
|
|
* 1. When Desk is opened, i.e. when the usage starts, this also sends a
|
|
|
|
* Opened instance log.
|
|
|
|
* 2. On visibility change if not started, eg: when user minimizes Books and
|
2022-03-18 15:39:17 +05:30
|
|
|
* then comes back later.
|
2022-07-18 16:43:56 +05:30
|
|
|
* 3. When `log` is called, but telemetry isn't initialized.
|
2022-03-18 15:39:17 +05:30
|
|
|
*
|
|
|
|
* ## `log`
|
2022-05-23 13:39:07 +05:30
|
|
|
* Used to log activity.
|
2022-03-18 15:39:17 +05:30
|
|
|
*
|
|
|
|
* ## `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.
|
|
|
|
*/
|
2022-03-09 15:43:17 +05:30
|
|
|
|
2023-06-28 10:25:39 +05:30
|
|
|
const ignoreList: string[] = [
|
|
|
|
ModelNameEnum.AccountingLedgerEntry,
|
|
|
|
ModelNameEnum.StockLedgerEntry,
|
|
|
|
];
|
|
|
|
|
2022-04-18 16:59:20 +05:30
|
|
|
export class TelemetryManager {
|
2023-06-21 16:08:39 +05:30
|
|
|
#url = '';
|
|
|
|
#token = '';
|
2022-03-09 15:43:17 +05:30
|
|
|
#started = false;
|
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
|
|
|
get hasCreds() {
|
|
|
|
return !!this.#url && !!this.#token;
|
|
|
|
}
|
|
|
|
|
|
|
|
get started() {
|
|
|
|
return this.#started;
|
|
|
|
}
|
|
|
|
|
2022-07-18 16:43:56 +05:30
|
|
|
async start(isOpened?: boolean) {
|
2022-05-23 13:39:07 +05:30
|
|
|
this.#started = true;
|
|
|
|
await this.#setCreds();
|
2022-03-14 18:26:26 +05:30
|
|
|
|
2022-07-18 16:43:56 +05:30
|
|
|
if (isOpened) {
|
2022-05-28 16:44:38 +05:30
|
|
|
this.log(Verb.Opened, 'instance');
|
2022-05-23 13:39:07 +05:30
|
|
|
} else {
|
2022-05-28 16:44:38 +05:30
|
|
|
this.log(Verb.Resumed, 'instance');
|
2022-03-09 15:43:17 +05:30
|
|
|
}
|
2022-05-23 13:39:07 +05:30
|
|
|
}
|
2022-03-09 15:43:17 +05:30
|
|
|
|
2022-05-23 13:39:07 +05:30
|
|
|
stop() {
|
|
|
|
if (!this.started) {
|
2022-03-10 13:21:29 +05:30
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-05-28 16:44:38 +05:30
|
|
|
this.log(Verb.Closed, 'instance');
|
2022-05-23 13:39:07 +05:30
|
|
|
this.#started = false;
|
2022-03-09 15:43:17 +05:30
|
|
|
}
|
2022-03-09 16:24:42 +05:30
|
|
|
|
2022-05-23 13:39:07 +05:30
|
|
|
log(verb: Verb, noun: Noun, more?: Record<string, unknown>) {
|
2022-05-23 14:58:33 +05:30
|
|
|
if (!this.#started && this.fyo.db.isConnected) {
|
2023-06-22 14:22:54 +05:30
|
|
|
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
2022-05-23 13:39:07 +05:30
|
|
|
this.start().then(() => this.#sendBeacon(verb, noun, more));
|
|
|
|
return;
|
2022-03-09 16:50:48 +05:30
|
|
|
}
|
|
|
|
|
2022-05-23 13:39:07 +05:30
|
|
|
this.#sendBeacon(verb, noun, more);
|
2022-03-09 16:50:48 +05:30
|
|
|
}
|
|
|
|
|
2022-05-28 16:44:38 +05:30
|
|
|
async logOpened() {
|
|
|
|
await this.#setCreds();
|
|
|
|
this.#sendBeacon(Verb.Opened, 'app');
|
|
|
|
}
|
|
|
|
|
2022-05-23 13:39:07 +05:30
|
|
|
#sendBeacon(verb: Verb, noun: Noun, more?: Record<string, unknown>) {
|
2023-06-28 10:25:39 +05:30
|
|
|
if (
|
|
|
|
!this.hasCreds ||
|
|
|
|
this.fyo.store.skipTelemetryLogging ||
|
|
|
|
ignoreList.includes(noun)
|
|
|
|
) {
|
2022-05-23 13:39:07 +05:30
|
|
|
return;
|
|
|
|
}
|
2022-03-10 14:35:47 +05:30
|
|
|
|
2022-05-23 13:39:07 +05:30
|
|
|
const telemetryData: Telemetry = this.#getTelemtryData(verb, noun, more);
|
2022-03-18 15:39:17 +05:30
|
|
|
const data = JSON.stringify({
|
|
|
|
token: this.#token,
|
2022-05-23 13:39:07 +05:30
|
|
|
telemetryData,
|
2022-03-18 15:39:17 +05:30
|
|
|
});
|
2022-03-10 17:36:28 +05:30
|
|
|
|
2022-03-15 16:28:43 +05:30
|
|
|
navigator.sendBeacon(this.#url, data);
|
|
|
|
}
|
|
|
|
|
2022-03-18 15:39:17 +05:30
|
|
|
async #setCreds() {
|
|
|
|
if (this.hasCreds) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-05-27 15:09:24 +05:30
|
|
|
const { telemetryUrl, tokenString } = await this.fyo.auth.getCreds();
|
|
|
|
this.#url = telemetryUrl;
|
|
|
|
this.#token = tokenString;
|
2022-03-18 15:39:17 +05:30
|
|
|
}
|
|
|
|
|
2022-05-23 13:39:07 +05:30
|
|
|
#getTelemtryData(
|
|
|
|
verb: Verb,
|
|
|
|
noun: Noun,
|
|
|
|
more?: Record<string, unknown>
|
|
|
|
): Telemetry {
|
2023-06-21 16:08:39 +05:30
|
|
|
const countryCode = this.fyo.singles.SystemSettings?.countryCode;
|
2022-05-23 13:39:07 +05:30
|
|
|
return {
|
2022-07-18 16:43:56 +05:30
|
|
|
country: countryCode ?? '',
|
|
|
|
language: this.fyo.store.language,
|
|
|
|
deviceId:
|
2023-06-21 16:08:39 +05:30
|
|
|
this.fyo.store.deviceId || (this.fyo.config.get('deviceId') ?? '-'),
|
2022-07-18 16:43:56 +05:30
|
|
|
instanceId: this.fyo.store.instanceId,
|
|
|
|
version: this.fyo.store.appVersion,
|
|
|
|
openCount: this.fyo.store.openCount,
|
|
|
|
timestamp: new Date().toISOString().replace('T', ' ').slice(0, -1),
|
|
|
|
platform: this.fyo.store.platform,
|
2022-05-23 13:39:07 +05:30
|
|
|
verb,
|
|
|
|
noun,
|
|
|
|
more,
|
|
|
|
};
|
2022-03-18 15:39:17 +05:30
|
|
|
}
|
2022-03-09 15:43:17 +05:30
|
|
|
}
|