Files
smooth-app/packages/smooth_app/lib/helpers/analytics_helper.dart
Marvin Möltgen 1734a20381 ci: Return to entrypoint based scanner (#3874)
* ci: Return to entrypoint based scanner

* Update basic_test.dart
2023-04-14 17:07:37 +02:00

221 lines
6.3 KiB
Dart

import 'dart:async';
import 'package:flutter/foundation.dart';
import 'package:matomo_tracker/matomo_tracker.dart';
import 'package:openfoodfacts/openfoodfacts.dart';
import 'package:package_info_plus/package_info_plus.dart';
import 'package:sentry_flutter/sentry_flutter.dart';
import 'package:smooth_app/main.dart';
/// Category for Matomo Events
enum AnalyticsCategory {
userManagement(tag: 'user management'),
scanning(tag: 'scanning'),
share(tag: 'share'),
couldNotFindProduct(tag: 'could not find product'),
productEdit(tag: 'product edit'),
list(tag: 'list');
const AnalyticsCategory({required this.tag});
final String tag;
}
/// Event types for Matomo analytics
enum AnalyticsEvent {
scanAction(tag: 'scanned product', category: AnalyticsCategory.scanning),
scanStrangeRestart(
tag: 'strange restart', category: AnalyticsCategory.scanning),
scanStrangeRestop(
tag: 'strange restop', category: AnalyticsCategory.scanning),
shareProduct(tag: 'shared product', category: AnalyticsCategory.share),
loginAction(tag: 'logged in', category: AnalyticsCategory.userManagement),
registerAction(tag: 'register', category: AnalyticsCategory.userManagement),
logoutAction(tag: 'logged out', category: AnalyticsCategory.userManagement),
couldNotScanProduct(
tag: 'could not scan product',
category: AnalyticsCategory.couldNotFindProduct,
),
couldNotFindProduct(
tag: 'could not find product',
category: AnalyticsCategory.couldNotFindProduct,
),
openProductEditPage(
tag: 'opened product edit page',
category: AnalyticsCategory.productEdit,
),
shareList(tag: 'shared a list', category: AnalyticsCategory.list),
openListWeb(tag: 'open a list in wbe', category: AnalyticsCategory.list);
const AnalyticsEvent({required this.tag, required this.category});
final String tag;
final AnalyticsCategory category;
}
enum AnalyticsEditEvents {
basicDetails(name: 'BasicDetails'),
photos(name: 'Photos'),
powerEditScreen(name: 'Power Edit Screen'),
ingredients_and_Origins(name: 'Ingredient And Origins'),
categories(name: 'Categories'),
nutrition_Facts(name: 'Nutrition Facts'),
labelsAndCertifications(name: 'Labels And Certifications'),
packagingComponents(name: 'Packaging Components'),
recyclingInstructionsPhotos(name: 'Recycling Instructions Photos'),
stores(name: 'Stores'),
origins(name: 'Origins'),
traceabilityCodes(name: 'Traceability Codes'),
country(name: 'Country'),
otherDetails(name: 'Other Details');
const AnalyticsEditEvents({required this.name});
final String name;
}
/// Helper for logging usage of core features and exceptions
/// Logging:
/// - Errors and Problems (sentry)
/// - App start
/// - Product scan
/// - Product page open
/// - Knowledge panel open
/// - personalized ranking (without sharing the preferences)
/// - search
/// - external links
class AnalyticsHelper {
AnalyticsHelper._();
static bool _crashReports = false;
static String latestSearch = '';
/// Did the user allow the analytic reports?
static bool _allow = false;
static Future<void> initSentry({
required Function()? appRunner,
}) async {
final PackageInfo packageInfo = await PackageInfo.fromPlatform();
await SentryFlutter.init(
(SentryOptions options) {
options.dsn =
'https://22ec5d0489534b91ba455462d3736680@o241488.ingest.sentry.io/5376745';
options.sentryClientName =
'sentry.dart.smoothie/${packageInfo.version}';
// To set a uniform sample rate
options.tracesSampleRate = 1.0;
options.beforeSend = _beforeSend;
options.environment = appFlavour;
},
appRunner: appRunner,
);
}
static void setCrashReports(final bool crashReports) =>
_crashReports = crashReports;
static Future<void> setAnalyticsReports(final bool allow) async {
_allow = allow;
await MatomoTracker.instance.setOptOut(optout: false);
}
static FutureOr<SentryEvent?> _beforeSend(SentryEvent event,
{dynamic hint}) async {
if (!_crashReports) {
return null;
}
return event;
}
static Future<void> initMatomo(
final bool screenshotMode,
) async {
if (screenshotMode) {
setCrashReports(false);
setAnalyticsReports(false);
return;
}
try {
await MatomoTracker.instance.initialize(
url: 'https://analytics.openfoodfacts.org/matomo.php',
siteId: 2,
visitorId: uuid,
);
} catch (err) {
// With Hot Reload, this may trigger a late field already initialized
}
}
/// A UUID must be at least one 16 characters
static String? get uuid {
// if user opts out then track anonymously with userId containg zeros
if (kDebugMode) {
return 'smoothie_debug--';
}
if (!_allow) {
return '0' * 16;
}
return OpenFoodAPIConfiguration.uuid;
}
static void trackEvent(
AnalyticsEvent msg, {
int? eventValue,
String? barcode,
}) =>
MatomoTracker.instance.trackEvent(
eventName: msg.name,
eventCategory: msg.category.tag,
action: msg.name,
eventValue: eventValue ?? _formatBarcode(barcode),
);
static void trackProductEdit(
AnalyticsEditEvents editEventName, String barcode,
[bool saved = false]) =>
MatomoTracker.instance.trackEvent(
eventName: saved ? '${editEventName.name}-saved' : editEventName.name,
eventCategory: AnalyticsCategory.productEdit.tag,
action: editEventName.name,
eventValue: _formatBarcode(barcode),
);
static void trackSearch({
required String search,
String? searchCategory,
int? searchCount,
}) {
final String searchString = '$search,$searchCategory,$searchCount';
if (searchString == latestSearch) {
return;
}
latestSearch = searchString;
MatomoTracker.instance.trackSearch(
searchKeyword: search,
searchCount: searchCount,
searchCategory: searchCategory,
);
}
static void trackOutlink({required String url}) =>
MatomoTracker.instance.trackOutlink(url);
static int? _formatBarcode(String? barcode) {
if (barcode == null) {
return null;
}
const int fallback = 000000000;
try {
return int.tryParse(barcode) ?? fallback;
} on FormatException {
return fallback;
}
}
}