Follow up notification action isolation handling

fix: possible database corruption by allow multiple instances
This commit is contained in:
OfficialDakari 2025-11-13 20:19:51 +05:00
parent fc218e6d65
commit 9dbb75dada
6 changed files with 44 additions and 167 deletions

View File

@ -3,6 +3,9 @@ import 'dart:ui';
import 'package:matrix/matrix.dart'; import 'package:matrix/matrix.dart';
abstract class AppConfig { abstract class AppConfig {
static const String pushIsolatePortName = 'push_isolate';
static const String mainIsolatePortName = 'main_isolate';
static String _applicationName = 'Extera'; static String _applicationName = 'Extera';
static String get applicationName => _applicationName; static String get applicationName => _applicationName;

View File

@ -1,3 +1,6 @@
import 'dart:isolate';
import 'dart:ui';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:collection/collection.dart'; import 'package:collection/collection.dart';
@ -15,9 +18,20 @@ import 'config/setting_keys.dart';
import 'utils/background_push.dart'; import 'utils/background_push.dart';
import 'widgets/fluffy_chat_app.dart'; import 'widgets/fluffy_chat_app.dart';
ReceivePort? mainIsolateReceivePort;
void main() async { void main() async {
Logs().i('Welcome to ${AppConfig.applicationName} <3'); Logs().i('Welcome to ${AppConfig.applicationName} <3');
if (PlatformInfos.isAndroid) {
final port = mainIsolateReceivePort = ReceivePort();
IsolateNameServer.removePortNameMapping('main_isolate');
IsolateNameServer.registerPortWithName(
port.sendPort,
'main_isolate',
);
}
// Our background push shared isolate accesses flutter-internal things very early in the startup proccess // Our background push shared isolate accesses flutter-internal things very early in the startup proccess
// To make sure that the parts of flutter needed are started up already, we need to ensure that the // To make sure that the parts of flutter needed are started up already, we need to ensure that the
// widget bindings are initialized already. // widget bindings are initialized already.

View File

@ -23,6 +23,7 @@ import 'dart:io';
import 'dart:isolate'; import 'dart:isolate';
import 'dart:ui'; import 'dart:ui';
import 'package:extera_next/main.dart';
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
@ -75,6 +76,20 @@ class BackgroundPush {
void _init() async { void _init() async {
try { try {
mainIsolateReceivePort?.listen(
(message) async {
try {
await notificationTap(
NotificationResponseJson.fromJsonString(message),
client: client,
router: FluffyChatApp.router,
l10n: l10n,
);
} catch (e, s) {
Logs().wtf('Main Notification Tap crashed', e, s);
}
},
);
if (PlatformInfos.isAndroid) { if (PlatformInfos.isAndroid) {
final port = ReceivePort(); final port = ReceivePort();
IsolateNameServer.removePortNameMapping('background_tab_port'); IsolateNameServer.removePortNameMapping('background_tab_port');
@ -106,8 +121,7 @@ class BackgroundPush {
response, response,
client: client, client: client,
router: FluffyChatApp.router, router: FluffyChatApp.router,
l10n: l10n l10n: l10n),
),
onDidReceiveBackgroundNotificationResponse: notificationTapBackground, onDidReceiveBackgroundNotificationResponse: notificationTapBackground,
); );
Logs().v('Flutter Local Notifications initialized'); Logs().v('Flutter Local Notifications initialized');
@ -312,12 +326,8 @@ class BackgroundPush {
final response = details.notificationResponse; final response = details.notificationResponse;
if (response != null) { if (response != null) {
notificationTap( notificationTap(response,
response, client: client, router: FluffyChatApp.router, l10n: l10n);
client: client,
router: FluffyChatApp.router,
l10n: l10n
);
} }
}); });
} }

View File

@ -1,149 +0,0 @@
import 'dart:convert';
import 'dart:io';
import 'package:flutter/foundation.dart' hide Key;
import 'package:flutter/services.dart';
import 'package:flutter_secure_storage/flutter_secure_storage.dart';
import 'package:hive/hive.dart';
import 'package:matrix/matrix.dart';
import 'package:path_provider/path_provider.dart';
import 'package:universal_html/html.dart' as html;
// ignore: deprecated_member_use
class FlutterHiveCollectionsDatabase extends HiveCollectionsDatabase {
FlutterHiveCollectionsDatabase(
super.name,
String super.path, {
super.key,
});
static const String _cipherStorageKey = 'hive_encryption_key';
static Future<FlutterHiveCollectionsDatabase> databaseBuilder(
Client client,
) async {
Logs().d('Open Hive...');
HiveAesCipher? hiverCipher;
try {
// Workaround for secure storage is calling Platform.operatingSystem on web
if (kIsWeb) {
// ignore: unawaited_futures
html.window.navigator.storage?.persist();
throw MissingPluginException();
}
const secureStorage = FlutterSecureStorage();
final containsEncryptionKey =
await secureStorage.read(key: _cipherStorageKey) != null;
if (!containsEncryptionKey) {
// do not try to create a buggy secure storage for new Linux users
if (Platform.isLinux) throw MissingPluginException();
final key = Hive.generateSecureKey();
await secureStorage.write(
key: _cipherStorageKey,
value: base64UrlEncode(key),
);
}
// workaround for if we just wrote to the key and it still doesn't exist
final rawEncryptionKey = await secureStorage.read(key: _cipherStorageKey);
if (rawEncryptionKey == null) throw MissingPluginException();
hiverCipher = HiveAesCipher(base64Url.decode(rawEncryptionKey));
} on MissingPluginException catch (_) {
const FlutterSecureStorage()
.delete(key: _cipherStorageKey)
.catchError((_) {});
Logs().i('Hive encryption is not supported on this platform');
} catch (e, s) {
const FlutterSecureStorage()
.delete(key: _cipherStorageKey)
.catchError((_) {});
Logs().w('Unable to init Hive encryption', e, s);
}
final db = FlutterHiveCollectionsDatabase(
'hive_collections_${client.clientName.replaceAll(' ', '_').toLowerCase()}',
await findDatabasePath(client),
key: hiverCipher,
);
try {
await db.open();
} catch (e, s) {
Logs().w('Unable to open Hive. Delete database and storage key...', e, s);
const FlutterSecureStorage().delete(key: _cipherStorageKey);
await db.clear().catchError((_) {});
await Hive.deleteFromDisk();
rethrow;
}
Logs().d('Hive is ready');
return db;
}
static Future<String> findDatabasePath(Client client) async {
var path = client.clientName;
if (!kIsWeb) {
Directory directory;
try {
if (Platform.isLinux) {
directory = await getApplicationSupportDirectory();
} else {
directory = await getApplicationDocumentsDirectory();
}
} catch (_) {
try {
directory = await getLibraryDirectory();
} catch (_) {
directory = Directory.current;
}
}
// do not destroy your stable FluffyChat in debug mode
directory = Directory(
directory.uri.resolve(kDebugMode ? 'hive_debug' : 'hive').toFilePath(),
);
directory.create(recursive: true);
path = directory.path;
}
return path;
}
@override
int get maxFileSize => supportsFileStoring ? 100 * 1000 * 1000 : 0;
@override
bool get supportsFileStoring => !kIsWeb;
Future<String> _getFileStoreDirectory() async {
try {
try {
return (await getTemporaryDirectory()).path;
} catch (_) {
return (await getApplicationDocumentsDirectory()).path;
}
} catch (_) {
return (await getDownloadsDirectory())!.path;
}
}
@override
Future<Uint8List?> getFile(Uri mxcUri) async {
if (!supportsFileStoring) return null;
final tempDirectory = await _getFileStoreDirectory();
final file =
File('$tempDirectory/${Uri.encodeComponent(mxcUri.toString())}');
if (await file.exists() == false) return null;
final bytes = await file.readAsBytes();
return bytes;
}
@override
Future storeFile(Uri mxcUri, Uint8List bytes, int time) async {
if (!supportsFileStoring) return null;
final tempDirectory = await _getFileStoreDirectory();
final file =
File('$tempDirectory/${Uri.encodeComponent(mxcUri.toString())}');
if (await file.exists()) return;
await file.writeAsBytes(bytes);
return;
}
}

View File

@ -113,7 +113,7 @@ Future<MatrixSdkDatabase> _constructDatabase(String clientName) async {
version: 1, version: 1,
// most important : apply encryption when opening the DB // most important : apply encryption when opening the DB
onConfigure: helper?.applyPragmaKey, onConfigure: helper?.applyPragmaKey,
singleInstance: false), ),
); );
return await MatrixSdkDatabase.init( return await MatrixSdkDatabase.init(

View File

@ -46,11 +46,10 @@ extension NotificationResponseJson on NotificationResponse {
void notificationTapBackground( void notificationTapBackground(
NotificationResponse notificationResponse, NotificationResponse notificationResponse,
) async { ) async {
Logs().i('Notification tap in background'); final sendPort = IsolateNameServer.lookupPortByName(AppConfig.mainIsolatePortName);
final sendPort = IsolateNameServer.lookupPortByName('background_tab_port');
if (sendPort != null) { if (sendPort != null) {
sendPort.send(notificationResponse.toJsonString()); sendPort.send(notificationResponse.toJsonString());
Logs().i('Notification tap sent to main isolate');
return; return;
} }