mirror of
				https://github.com/immich-app/immich.git
				synced 2025-10-31 02:27:08 -04:00 
			
		
		
		
	* dispose store on isolate cleanup * do not listen for store updates in isolates --------- Co-authored-by: shenlong-tanwen <139912620+shalong-tanwen@users.noreply.github.com> Co-authored-by: Alex <alex.tran1502@gmail.com>
		
			
				
	
	
		
			95 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			95 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
| import 'dart:async';
 | |
| import 'dart:ui';
 | |
| 
 | |
| import 'package:flutter/services.dart';
 | |
| import 'package:hooks_riverpod/hooks_riverpod.dart';
 | |
| import 'package:immich_mobile/domain/services/log.service.dart';
 | |
| import 'package:immich_mobile/entities/store.entity.dart';
 | |
| import 'package:immich_mobile/providers/db.provider.dart';
 | |
| import 'package:immich_mobile/providers/infrastructure/cancel.provider.dart';
 | |
| import 'package:immich_mobile/providers/infrastructure/db.provider.dart';
 | |
| import 'package:immich_mobile/utils/bootstrap.dart';
 | |
| import 'package:immich_mobile/utils/debug_print.dart';
 | |
| import 'package:immich_mobile/utils/http_ssl_options.dart';
 | |
| import 'package:logging/logging.dart';
 | |
| import 'package:worker_manager/worker_manager.dart';
 | |
| 
 | |
| class InvalidIsolateUsageException implements Exception {
 | |
|   const InvalidIsolateUsageException();
 | |
| 
 | |
|   @override
 | |
|   String toString() => "IsolateHelper should only be used from the root isolate";
 | |
| }
 | |
| 
 | |
| // !! Should be used only from the root isolate
 | |
| Cancelable<T?> runInIsolateGentle<T>({
 | |
|   required Future<T> Function(ProviderContainer ref) computation,
 | |
|   String? debugLabel,
 | |
| }) {
 | |
|   final token = RootIsolateToken.instance;
 | |
|   if (token == null) {
 | |
|     throw const InvalidIsolateUsageException();
 | |
|   }
 | |
| 
 | |
|   return workerManager.executeGentle((cancelledChecker) async {
 | |
|     await runZonedGuarded(
 | |
|       () async {
 | |
|         BackgroundIsolateBinaryMessenger.ensureInitialized(token);
 | |
|         DartPluginRegistrant.ensureInitialized();
 | |
| 
 | |
|         final (isar, drift, logDb) = await Bootstrap.initDB();
 | |
|         await Bootstrap.initDomain(isar, drift, logDb, shouldBufferLogs: false, listenStoreUpdates: false);
 | |
|         final ref = ProviderContainer(
 | |
|           overrides: [
 | |
|             // TODO: Remove once isar is removed
 | |
|             dbProvider.overrideWithValue(isar),
 | |
|             isarProvider.overrideWithValue(isar),
 | |
|             cancellationProvider.overrideWithValue(cancelledChecker),
 | |
|             driftProvider.overrideWith(driftOverride(drift)),
 | |
|           ],
 | |
|         );
 | |
| 
 | |
|         Logger log = Logger("IsolateLogger");
 | |
| 
 | |
|         try {
 | |
|           HttpSSLOptions.apply(applyNative: false);
 | |
|           return await computation(ref);
 | |
|         } on CanceledError {
 | |
|           log.warning("Computation cancelled ${debugLabel == null ? '' : ' for $debugLabel'}");
 | |
|         } catch (error, stack) {
 | |
|           log.severe("Error in runInIsolateGentle ${debugLabel == null ? '' : ' for $debugLabel'}", error, stack);
 | |
|         } finally {
 | |
|           try {
 | |
|             ref.dispose();
 | |
| 
 | |
|             await Store.dispose();
 | |
|             await LogService.I.dispose();
 | |
|             await logDb.close();
 | |
|             await drift.close();
 | |
| 
 | |
|             // Close Isar safely
 | |
|             try {
 | |
|               if (isar.isOpen) {
 | |
|                 await isar.close();
 | |
|               }
 | |
|             } catch (e) {
 | |
|               dPrint(() => "Error closing Isar: $e");
 | |
|             }
 | |
|           } catch (error, stack) {
 | |
|             dPrint(() => "Error closing resources in isolate: $error, $stack");
 | |
|           } finally {
 | |
|             ref.dispose();
 | |
|             // Delay to ensure all resources are released
 | |
|             await Future.delayed(const Duration(seconds: 2));
 | |
|           }
 | |
|         }
 | |
|         return null;
 | |
|       },
 | |
|       (error, stack) {
 | |
|         dPrint(() => "Error in isolate $debugLabel zone: $error, $stack");
 | |
|       },
 | |
|     );
 | |
|     return null;
 | |
|   });
 | |
| }
 |