From f50a994b92a57cee074d969ec7965af7e87b7903 Mon Sep 17 00:00:00 2001 From: Dennis Fokin Date: Fri, 6 Dec 2024 12:40:49 +0100 Subject: [PATCH] Attempt to repair/restore a broken preferences file --- lib/desktop/init.dart | 43 ++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 42 insertions(+), 1 deletion(-) diff --git a/lib/desktop/init.dart b/lib/desktop/init.dart index 9a78868e9..26e5d748b 100755 --- a/lib/desktop/init.dart +++ b/lib/desktop/init.dart @@ -27,6 +27,10 @@ import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:local_notifier/local_notifier.dart'; import 'package:logging/logging.dart'; import 'package:material_symbols_icons/symbols.dart'; +import 'package:path/path.dart' as path; +import 'package:path_provider/path_provider.dart'; +import 'package:path_provider_linux/path_provider_linux.dart'; +import 'package:path_provider_windows/path_provider_windows.dart'; import 'package:screen_retriever/screen_retriever.dart'; import 'package:shared_preferences/shared_preferences.dart'; import 'package:window_manager/window_manager.dart'; @@ -125,7 +129,34 @@ Future initialize(List argv) async { _initLogging(args); await windowManager.ensureInitialized(); - final prefs = await SharedPreferences.getInstance(); + SharedPreferences prefs; + try { + prefs = await SharedPreferences.getInstance(); + } catch (error) { + // Attempt to repair the broken preferences file + String? directory; + if (Platform.isWindows) { + PathProviderWindows pathProvider = PathProviderWindows(); + directory = await pathProvider.getApplicationSupportPath(); + } else if (Platform.isLinux) { + PathProviderLinux pathProvider = PathProviderLinux(); + directory = await pathProvider.getApplicationSupportPath(); + } + if (directory == null) { + throw FormatException("Unable to find correct directory"); + } + String appDataPath = path.join(directory, 'shared_preferences.json'); + List repairedPreferences = await _repairPreferences(appDataPath); + await File(appDataPath).writeAsBytes(repairedPreferences); + + try { + prefs = await SharedPreferences.getInstance(); + } catch (error) { + // Unable to repair the preferences file, therefore delete it + await File(appDataPath).delete(); + prefs = await SharedPreferences.getInstance(); + } + } final windowManagerHelper = WindowManagerHelper.withPreferences(prefs); final isHidden = _getIsHidden(args, prefs); @@ -413,3 +444,13 @@ class _HelperWaiterState extends ConsumerState<_HelperWaiter> { } } } + +Future> _repairPreferences(String appDataPath) async { + List contents = await File(appDataPath).readAsBytes(); + var contentsGrowable = new List.from(contents); // Make the list growable + + // Remove any NUL characters + contentsGrowable.removeWhere((item) => item == 0); + + return contentsGrowable; +}