2022-07-05 22:20:24 +02:00
|
|
|
import 'package:drift/drift.dart';
|
2022-03-27 19:55:31 +02:00
|
|
|
import 'package:equatable/equatable.dart';
|
|
|
|
import 'package:event_bus/event_bus.dart';
|
|
|
|
import 'package:flutter/foundation.dart';
|
|
|
|
import 'package:kiwi/kiwi.dart';
|
|
|
|
import 'package:logging/logging.dart';
|
|
|
|
import 'package:nc_photos/debug_util.dart';
|
|
|
|
import 'package:nc_photos/di_container.dart';
|
|
|
|
import 'package:nc_photos/entity/album.dart';
|
2022-07-05 22:20:24 +02:00
|
|
|
import 'package:nc_photos/entity/album/data_source.dart';
|
2022-03-27 19:55:31 +02:00
|
|
|
import 'package:nc_photos/entity/face.dart';
|
|
|
|
import 'package:nc_photos/entity/face/data_source.dart';
|
|
|
|
import 'package:nc_photos/entity/favorite.dart';
|
|
|
|
import 'package:nc_photos/entity/favorite/data_source.dart';
|
|
|
|
import 'package:nc_photos/entity/file.dart';
|
|
|
|
import 'package:nc_photos/entity/file/data_source.dart';
|
2022-05-05 16:06:47 +02:00
|
|
|
import 'package:nc_photos/entity/local_file.dart';
|
|
|
|
import 'package:nc_photos/entity/local_file/data_source.dart';
|
2022-03-27 19:55:31 +02:00
|
|
|
import 'package:nc_photos/entity/person.dart';
|
|
|
|
import 'package:nc_photos/entity/person/data_source.dart';
|
|
|
|
import 'package:nc_photos/entity/share.dart';
|
|
|
|
import 'package:nc_photos/entity/share/data_source.dart';
|
|
|
|
import 'package:nc_photos/entity/sharee.dart';
|
|
|
|
import 'package:nc_photos/entity/sharee/data_source.dart';
|
2022-07-05 22:20:24 +02:00
|
|
|
import 'package:nc_photos/entity/sqlite_table.dart' as sql;
|
|
|
|
import 'package:nc_photos/entity/sqlite_table_isolate.dart' as sql_isolate;
|
2022-03-27 19:55:31 +02:00
|
|
|
import 'package:nc_photos/entity/tag.dart';
|
|
|
|
import 'package:nc_photos/entity/tag/data_source.dart';
|
|
|
|
import 'package:nc_photos/entity/tagged_file.dart';
|
|
|
|
import 'package:nc_photos/entity/tagged_file/data_source.dart';
|
|
|
|
import 'package:nc_photos/k.dart' as k;
|
|
|
|
import 'package:nc_photos/mobile/android/android_info.dart';
|
|
|
|
import 'package:nc_photos/mobile/self_signed_cert_manager.dart';
|
|
|
|
import 'package:nc_photos/platform/features.dart' as features;
|
|
|
|
import 'package:nc_photos/platform/k.dart' as platform_k;
|
|
|
|
import 'package:nc_photos/pref.dart';
|
|
|
|
import 'package:nc_photos/pref_util.dart' as pref_util;
|
2022-06-12 07:45:51 +02:00
|
|
|
import 'package:visibility_detector/visibility_detector.dart';
|
2022-03-27 19:55:31 +02:00
|
|
|
|
2022-07-05 22:20:24 +02:00
|
|
|
enum InitIsolateType {
|
|
|
|
main,
|
|
|
|
service,
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> init(InitIsolateType isolateType) async {
|
2022-06-06 10:59:42 +02:00
|
|
|
if (_hasInitedInThisIsolate) {
|
|
|
|
_log.warning("[initAppLaunch] Already initialized in this isolate");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
initLog();
|
2022-07-05 22:20:24 +02:00
|
|
|
_initDrift();
|
2022-03-27 19:55:31 +02:00
|
|
|
_initKiwi();
|
|
|
|
await _initPref();
|
|
|
|
await _initAccountPrefs();
|
|
|
|
await _initDeviceInfo();
|
|
|
|
_initEquatable();
|
|
|
|
if (features.isSupportSelfSignedCert) {
|
|
|
|
_initSelfSignedCertManager();
|
|
|
|
}
|
2022-07-05 22:20:24 +02:00
|
|
|
await _initDiContainer(isolateType);
|
2022-06-12 07:45:51 +02:00
|
|
|
_initVisibilityDetector();
|
2022-06-06 10:59:42 +02:00
|
|
|
|
|
|
|
_hasInitedInThisIsolate = true;
|
2022-03-27 19:55:31 +02:00
|
|
|
}
|
|
|
|
|
2022-06-06 10:59:42 +02:00
|
|
|
void initLog() {
|
|
|
|
if (_hasInitedInThisIsolate) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-03-27 19:55:31 +02:00
|
|
|
Logger.root.level = kReleaseMode ? Level.WARNING : Level.ALL;
|
|
|
|
Logger.root.onRecord.listen((record) {
|
|
|
|
// dev.log(
|
|
|
|
// "${record.level.name} ${record.time}: ${record.message}",
|
|
|
|
// time: record.time,
|
|
|
|
// sequenceNumber: record.sequenceNumber,
|
|
|
|
// level: record.level.value,
|
|
|
|
// name: record.loggerName,
|
|
|
|
// );
|
|
|
|
String msg =
|
|
|
|
"[${record.loggerName}] ${record.level.name} ${record.time}: ${record.message}";
|
|
|
|
if (record.error != null) {
|
|
|
|
msg += " (throw: ${record.error.runtimeType} { ${record.error} })";
|
|
|
|
}
|
|
|
|
if (record.stackTrace != null) {
|
|
|
|
msg += "\nStack Trace:\n${record.stackTrace}";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (kDebugMode) {
|
|
|
|
// show me colors!
|
|
|
|
int color;
|
|
|
|
if (record.level >= Level.SEVERE) {
|
|
|
|
color = 91;
|
|
|
|
} else if (record.level >= Level.WARNING) {
|
|
|
|
color = 33;
|
|
|
|
} else if (record.level >= Level.INFO) {
|
|
|
|
color = 34;
|
|
|
|
} else if (record.level >= Level.FINER) {
|
|
|
|
color = 32;
|
|
|
|
} else {
|
|
|
|
color = 90;
|
|
|
|
}
|
|
|
|
msg = "\x1B[${color}m$msg\x1B[0m";
|
|
|
|
}
|
2022-07-10 21:13:45 +02:00
|
|
|
debugPrint(msg, wrapWidth: 1024);
|
2022-03-27 19:55:31 +02:00
|
|
|
LogCapturer().onLog(msg);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-07-05 22:20:24 +02:00
|
|
|
void _initDrift() {
|
|
|
|
driftRuntimeOptions.debugPrint = _debugPrintSql;
|
|
|
|
}
|
|
|
|
|
2022-03-27 19:55:31 +02:00
|
|
|
Future<void> _initPref() async {
|
|
|
|
final provider = PrefSharedPreferencesProvider();
|
|
|
|
await provider.init();
|
|
|
|
final pref = Pref.scoped(provider);
|
|
|
|
Pref.setGlobalInstance(pref);
|
|
|
|
|
|
|
|
if (Pref().getLastVersion() == null) {
|
|
|
|
if (Pref().getSetupProgress() == null) {
|
|
|
|
// new install
|
|
|
|
await Pref().setLastVersion(k.version);
|
|
|
|
} else {
|
|
|
|
// v6 is the last version without saving the version number in pref
|
|
|
|
await Pref().setLastVersion(6);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _initAccountPrefs() async {
|
|
|
|
for (final a in Pref().getAccounts3Or([])) {
|
|
|
|
try {
|
|
|
|
AccountPref.setGlobalInstance(a, await pref_util.loadAccountPref(a));
|
|
|
|
} catch (e, stackTrace) {
|
|
|
|
_log.shout("[_initAccountPrefs] Failed reading pref for account: $a", e,
|
|
|
|
stackTrace);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _initDeviceInfo() async {
|
|
|
|
if (platform_k.isAndroid) {
|
|
|
|
await AndroidInfo.init();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _initKiwi() {
|
|
|
|
final kiwi = KiwiContainer();
|
|
|
|
kiwi.registerInstance<EventBus>(EventBus());
|
|
|
|
}
|
|
|
|
|
|
|
|
void _initEquatable() {
|
|
|
|
EquatableConfig.stringify = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void _initSelfSignedCertManager() {
|
|
|
|
SelfSignedCertManager().init();
|
|
|
|
}
|
|
|
|
|
2022-07-05 22:20:24 +02:00
|
|
|
Future<void> _initDiContainer(InitIsolateType isolateType) async {
|
|
|
|
final c = DiContainer.late();
|
|
|
|
c.pref = Pref();
|
|
|
|
c.sqliteDb = await _createDb(isolateType);
|
|
|
|
|
|
|
|
c.albumRepo = AlbumRepo(AlbumCachedDataSource(c));
|
|
|
|
c.albumRepoLocal = AlbumRepo(AlbumSqliteDbDataSource(c));
|
|
|
|
c.faceRepo = const FaceRepo(FaceRemoteDataSource());
|
|
|
|
c.fileRepo = FileRepo(FileCachedDataSource(c));
|
|
|
|
c.fileRepoRemote = const FileRepo(FileWebdavDataSource());
|
|
|
|
c.fileRepoLocal = FileRepo(FileSqliteDbDataSource(c));
|
|
|
|
c.personRepo = const PersonRepo(PersonRemoteDataSource());
|
|
|
|
c.shareRepo = ShareRepo(ShareRemoteDataSource());
|
|
|
|
c.shareeRepo = ShareeRepo(ShareeRemoteDataSource());
|
|
|
|
c.favoriteRepo = const FavoriteRepo(FavoriteRemoteDataSource());
|
|
|
|
c.tagRepo = const TagRepo(TagRemoteDataSource());
|
|
|
|
c.taggedFileRepo = const TaggedFileRepo(TaggedFileRemoteDataSource());
|
|
|
|
|
2022-05-05 16:06:47 +02:00
|
|
|
if (platform_k.isAndroid) {
|
|
|
|
// local file currently only supported on Android
|
2022-07-05 22:20:24 +02:00
|
|
|
c.localFileRepo = const LocalFileRepo(LocalFileMediaStoreDataSource());
|
2022-05-05 16:06:47 +02:00
|
|
|
}
|
2022-07-05 22:20:24 +02:00
|
|
|
|
|
|
|
KiwiContainer().registerInstance<DiContainer>(c);
|
2022-03-27 19:55:31 +02:00
|
|
|
}
|
|
|
|
|
2022-06-12 07:45:51 +02:00
|
|
|
void _initVisibilityDetector() {
|
|
|
|
VisibilityDetectorController.instance.updateInterval = Duration.zero;
|
|
|
|
}
|
|
|
|
|
2022-07-05 22:20:24 +02:00
|
|
|
Future<sql.SqliteDb> _createDb(InitIsolateType isolateType) async {
|
|
|
|
switch (isolateType) {
|
|
|
|
case InitIsolateType.main:
|
|
|
|
// use driftIsolate to prevent DB blocking the UI thread
|
|
|
|
if (platform_k.isWeb) {
|
|
|
|
// no isolate support on web
|
|
|
|
return sql.SqliteDb();
|
|
|
|
} else {
|
|
|
|
return sql_isolate.createDb();
|
|
|
|
}
|
|
|
|
|
|
|
|
case InitIsolateType.service:
|
|
|
|
// service already runs in an isolate
|
|
|
|
return sql.SqliteDb();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _debugPrintSql(String log) {
|
|
|
|
_log.finer(log);
|
|
|
|
}
|
|
|
|
|
2022-03-27 19:55:31 +02:00
|
|
|
final _log = Logger("app_init");
|
2022-06-06 10:59:42 +02:00
|
|
|
var _hasInitedInThisIsolate = false;
|