Change directory management
This commit is contained in:
@@ -34,12 +34,9 @@ class MethodHandler(private val scope: CoroutineScope) : FlutterPlugin,
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun onAttachedToEngine(flutterPluginBinding: FlutterPlugin.FlutterPluginBinding) {
|
override fun onAttachedToEngine(flutterPluginBinding: FlutterPlugin.FlutterPluginBinding) {
|
||||||
val taskQueue = flutterPluginBinding.binaryMessenger.makeBackgroundTaskQueue()
|
|
||||||
channel = MethodChannel(
|
channel = MethodChannel(
|
||||||
flutterPluginBinding.binaryMessenger,
|
flutterPluginBinding.binaryMessenger,
|
||||||
channelName,
|
channelName,
|
||||||
StandardMethodCodec.INSTANCE,
|
|
||||||
taskQueue
|
|
||||||
)
|
)
|
||||||
channel!!.setMethodCallHandler(this)
|
channel!!.setMethodCallHandler(this)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -33,7 +33,7 @@ class PlatformSettingsHandler : FlutterPlugin, MethodChannel.MethodCallHandler,
|
|||||||
private lateinit var ignoreRequestResult: MethodChannel.Result
|
private lateinit var ignoreRequestResult: MethodChannel.Result
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
const val channelName = "com.hiddify.app/platform.settings"
|
const val channelName = "app.hiddify.com/platform"
|
||||||
|
|
||||||
const val REQUEST_IGNORE_BATTERY_OPTIMIZATIONS = 44
|
const val REQUEST_IGNORE_BATTERY_OPTIMIZATIONS = 44
|
||||||
val gson = Gson()
|
val gson = Gson()
|
||||||
|
|||||||
@@ -122,7 +122,7 @@ void initLoggers(
|
|||||||
final logToFile = debug || (!Platform.isAndroid && !Platform.isIOS);
|
final logToFile = debug || (!Platform.isAndroid && !Platform.isIOS);
|
||||||
if (logToFile) {
|
if (logToFile) {
|
||||||
_loggers.addPrinter(
|
_loggers.addPrinter(
|
||||||
FileLogPrinter(read(filesEditorServiceProvider).appLogsPath),
|
FileLogPrinter(read(filesEditorServiceProvider).appLogsFile.path),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
Loggy.initLoggy(
|
Loggy.initLoggy(
|
||||||
|
|||||||
@@ -34,15 +34,11 @@ class CoreFacadeImpl with ExceptionHandler, InfraLogger implements CoreFacade {
|
|||||||
return exceptionHandler(
|
return exceptionHandler(
|
||||||
() {
|
() {
|
||||||
loggy.debug("setting up singbox");
|
loggy.debug("setting up singbox");
|
||||||
loggy.debug("base dir: ${filesEditor.baseDir.path}");
|
|
||||||
loggy.debug("working dir: ${filesEditor.workingDir.path}");
|
|
||||||
loggy.debug("temp dir: ${filesEditor.tempDir.path}");
|
|
||||||
|
|
||||||
return singbox
|
return singbox
|
||||||
.setup(
|
.setup(
|
||||||
filesEditor.baseDir.path,
|
filesEditor.dirs.baseDir.path,
|
||||||
filesEditor.workingDir.path,
|
filesEditor.dirs.workingDir.path,
|
||||||
filesEditor.tempDir.path,
|
filesEditor.dirs.tempDir.path,
|
||||||
)
|
)
|
||||||
.map((r) {
|
.map((r) {
|
||||||
loggy.debug("setup complete");
|
loggy.debug("setup complete");
|
||||||
@@ -180,7 +176,7 @@ class CoreFacadeImpl with ExceptionHandler, InfraLogger implements CoreFacade {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Stream<Either<CoreServiceFailure, List<String>>> watchLogs() {
|
Stream<Either<CoreServiceFailure, List<String>>> watchLogs() {
|
||||||
return singbox.watchLogs(filesEditor.coreLogsPath).handleExceptions(
|
return singbox.watchLogs(filesEditor.coreLogsFile.path).handleExceptions(
|
||||||
(error, stackTrace) {
|
(error, stackTrace) {
|
||||||
loggy.warning("error watching logs", error, stackTrace);
|
loggy.warning("error watching logs", error, stackTrace);
|
||||||
return CoreServiceFailure.unexpected(error, stackTrace);
|
return CoreServiceFailure.unexpected(error, stackTrace);
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_hooks/flutter_hooks.dart';
|
import 'package:flutter_hooks/flutter_hooks.dart';
|
||||||
import 'package:fpdart/fpdart.dart';
|
import 'package:fpdart/fpdart.dart';
|
||||||
@@ -32,7 +31,7 @@ class LogsPage extends HookConsumerWidget with PresLogger {
|
|||||||
child: Text(t.logs.shareCoreLogs),
|
child: Text(t.logs.shareCoreLogs),
|
||||||
onTap: () async {
|
onTap: () async {
|
||||||
await UriUtils.tryShareOrLaunchFile(
|
await UriUtils.tryShareOrLaunchFile(
|
||||||
Uri.parse(filesEditor.coreLogsPath),
|
Uri.parse(filesEditor.coreLogsFile.path),
|
||||||
fileOrDir: filesEditor.logsDir.uri,
|
fileOrDir: filesEditor.logsDir.uri,
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@@ -41,7 +40,7 @@ class LogsPage extends HookConsumerWidget with PresLogger {
|
|||||||
child: Text(t.logs.shareAppLogs),
|
child: Text(t.logs.shareAppLogs),
|
||||||
onTap: () async {
|
onTap: () async {
|
||||||
await UriUtils.tryShareOrLaunchFile(
|
await UriUtils.tryShareOrLaunchFile(
|
||||||
Uri.parse(filesEditor.appLogsPath),
|
Uri.parse(filesEditor.appLogsFile.path),
|
||||||
fileOrDir: filesEditor.logsDir.uri,
|
fileOrDir: filesEditor.logsDir.uri,
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import 'package:go_router/go_router.dart';
|
|||||||
import 'package:hiddify/core/core_providers.dart';
|
import 'package:hiddify/core/core_providers.dart';
|
||||||
import 'package:hiddify/core/prefs/general_prefs.dart';
|
import 'package:hiddify/core/prefs/general_prefs.dart';
|
||||||
import 'package:hiddify/domain/singbox/rules.dart';
|
import 'package:hiddify/domain/singbox/rules.dart';
|
||||||
import 'package:hiddify/services/platform_settings.dart';
|
import 'package:hiddify/services/platform_services.dart';
|
||||||
import 'package:hiddify/services/service_providers.dart';
|
import 'package:hiddify/services/service_providers.dart';
|
||||||
import 'package:hiddify/utils/riverpod_utils.dart';
|
import 'package:hiddify/utils/riverpod_utils.dart';
|
||||||
import 'package:hiddify/utils/utils.dart';
|
import 'package:hiddify/utils/utils.dart';
|
||||||
@@ -23,7 +23,7 @@ Future<List<InstalledPackageInfo>> installedPackagesInfo(
|
|||||||
InstalledPackagesInfoRef ref,
|
InstalledPackagesInfoRef ref,
|
||||||
) async {
|
) async {
|
||||||
return ref
|
return ref
|
||||||
.watch(platformSettingsProvider)
|
.watch(platformServicesProvider)
|
||||||
.getInstalledPackages()
|
.getInstalledPackages()
|
||||||
.getOrElse((err) {
|
.getOrElse((err) {
|
||||||
_logger.error("error getting installed packages", err);
|
_logger.error("error getting installed packages", err);
|
||||||
@@ -38,7 +38,7 @@ Future<ImageProvider> packageIcon(
|
|||||||
) async {
|
) async {
|
||||||
ref.disposeDelay(const Duration(seconds: 10));
|
ref.disposeDelay(const Duration(seconds: 10));
|
||||||
final bytes = await ref
|
final bytes = await ref
|
||||||
.watch(platformSettingsProvider)
|
.watch(platformServicesProvider)
|
||||||
.getPackageIcon(packageName)
|
.getPackageIcon(packageName)
|
||||||
.getOrElse((err) {
|
.getOrElse((err) {
|
||||||
_logger.warning("error getting package icon", err);
|
_logger.warning("error getting package icon", err);
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ Future<bool> isIgnoringBatteryOptimizations(
|
|||||||
IsIgnoringBatteryOptimizationsRef ref,
|
IsIgnoringBatteryOptimizationsRef ref,
|
||||||
) async =>
|
) async =>
|
||||||
ref
|
ref
|
||||||
.watch(platformSettingsProvider)
|
.watch(platformServicesProvider)
|
||||||
.isIgnoringBatteryOptimizations()
|
.isIgnoringBatteryOptimizations()
|
||||||
.getOrElse((l) => false)
|
.getOrElse((l) => false)
|
||||||
.run();
|
.run();
|
||||||
@@ -35,7 +35,7 @@ class PlatformSettingsTiles extends HookConsumerWidget {
|
|||||||
enabled: enabled,
|
enabled: enabled,
|
||||||
onTap: () async {
|
onTap: () async {
|
||||||
await ref
|
await ref
|
||||||
.read(platformSettingsProvider)
|
.read(platformServicesProvider)
|
||||||
.requestIgnoreBatteryOptimizations()
|
.requestIgnoreBatteryOptimizations()
|
||||||
.run();
|
.run();
|
||||||
await Future.delayed(const Duration(seconds: 1));
|
await Future.delayed(const Duration(seconds: 1));
|
||||||
|
|||||||
@@ -3,82 +3,65 @@ import 'dart:io';
|
|||||||
import 'package:flutter/services.dart';
|
import 'package:flutter/services.dart';
|
||||||
import 'package:hiddify/domain/constants.dart';
|
import 'package:hiddify/domain/constants.dart';
|
||||||
import 'package:hiddify/gen/assets.gen.dart';
|
import 'package:hiddify/gen/assets.gen.dart';
|
||||||
|
import 'package:hiddify/services/platform_services.dart';
|
||||||
import 'package:hiddify/utils/utils.dart';
|
import 'package:hiddify/utils/utils.dart';
|
||||||
import 'package:path/path.dart' as p;
|
import 'package:path/path.dart' as p;
|
||||||
import 'package:path_provider/path_provider.dart';
|
import 'package:path_provider/path_provider.dart';
|
||||||
|
|
||||||
|
typedef Directories = ({
|
||||||
|
Directory baseDir,
|
||||||
|
Directory workingDir,
|
||||||
|
Directory tempDir
|
||||||
|
});
|
||||||
|
|
||||||
class FilesEditorService with InfraLogger {
|
class FilesEditorService with InfraLogger {
|
||||||
|
FilesEditorService(this.platformServices);
|
||||||
|
|
||||||
late final _methodChannel = const MethodChannel("com.hiddify.app/files.method");
|
final PlatformServices platformServices;
|
||||||
|
|
||||||
late final Directory baseDir;
|
late final Directories dirs;
|
||||||
late final Directory workingDir;
|
|
||||||
late final Directory tempDir;
|
|
||||||
late final Directory logsDir;
|
|
||||||
late final Directory _configsDir;
|
|
||||||
|
|
||||||
Future<Map<String, String>?> getPaths() async {
|
Directory get workingDir => dirs.workingDir;
|
||||||
try {
|
Directory get configsDir =>
|
||||||
final Map<dynamic, dynamic>? directoryMap = await _methodChannel.invokeMethod('get_paths');
|
Directory(p.join(workingDir.path, Constants.configsFolderName));
|
||||||
return directoryMap?.cast<String, String>();
|
Directory get logsDir => dirs.workingDir;
|
||||||
} on PlatformException catch (e) {
|
|
||||||
// print("Failed to get shared directory: '${e.message}'.");
|
File get appLogsFile => File(p.join(logsDir.path, "app.log"));
|
||||||
return null;
|
File get coreLogsFile => File(p.join(logsDir.path, "box.log"));
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> init() async {
|
Future<void> init() async {
|
||||||
if (Platform.isIOS) {
|
dirs = await platformServices.getPaths().getOrElse(
|
||||||
final paths = await getPaths();
|
(error) {
|
||||||
baseDir = Directory(paths!["base"]!);
|
loggy.error("error getting paths", error, StackTrace.current);
|
||||||
workingDir = Directory(paths["working"]!);
|
throw error;
|
||||||
tempDir = Directory(paths["temp"]!);
|
},
|
||||||
|
).run();
|
||||||
|
|
||||||
|
loggy.info("directories: $dirs");
|
||||||
|
|
||||||
|
if (!await dirs.baseDir.exists()) {
|
||||||
|
await dirs.baseDir.create(recursive: true);
|
||||||
|
}
|
||||||
|
if (!await dirs.workingDir.exists()) {
|
||||||
|
await dirs.workingDir.create(recursive: true);
|
||||||
|
}
|
||||||
|
if (!await configsDir.exists()) {
|
||||||
|
await configsDir.create(recursive: true);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (await appLogsFile.exists()) {
|
||||||
|
await appLogsFile.writeAsString("");
|
||||||
} else {
|
} else {
|
||||||
baseDir = await getApplicationSupportDirectory();
|
await appLogsFile.create(recursive: true);
|
||||||
if (Platform.isAndroid) {
|
|
||||||
final externalDir = await getExternalStorageDirectory();
|
|
||||||
workingDir = externalDir!;
|
|
||||||
} else if (Platform.isWindows || Platform.isLinux || Platform.isMacOS) {
|
|
||||||
workingDir = baseDir;
|
|
||||||
} else {
|
|
||||||
workingDir = await getApplicationDocumentsDirectory();
|
|
||||||
}
|
|
||||||
tempDir = await getTemporaryDirectory();
|
|
||||||
}
|
|
||||||
logsDir = workingDir;
|
|
||||||
|
|
||||||
loggy.debug("base dir: ${baseDir.path}");
|
|
||||||
loggy.debug("working dir: ${workingDir.path}");
|
|
||||||
loggy.debug("temp dir: ${tempDir.path}");
|
|
||||||
loggy.debug("logs dire: ${logsDir.path}");
|
|
||||||
|
|
||||||
_configsDir =
|
|
||||||
Directory(p.join(workingDir.path, Constants.configsFolderName));
|
|
||||||
if (!await baseDir.exists()) {
|
|
||||||
await baseDir.create(recursive: true);
|
|
||||||
}
|
|
||||||
if (!await workingDir.exists()) {
|
|
||||||
await workingDir.create(recursive: true);
|
|
||||||
}
|
|
||||||
if (!await _configsDir.exists()) {
|
|
||||||
await _configsDir.create(recursive: true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
final appLogFile = File(appLogsPath);
|
if (await coreLogsFile.exists()) {
|
||||||
if (await appLogFile.exists()) {
|
await coreLogsFile.writeAsString("");
|
||||||
await appLogFile.writeAsString("");
|
|
||||||
} else {
|
} else {
|
||||||
await appLogFile.create(recursive: true);
|
await coreLogsFile.create(recursive: true);
|
||||||
}
|
}
|
||||||
|
|
||||||
await _populateGeoAssets();
|
await _populateGeoAssets();
|
||||||
|
|
||||||
final coreLogFile = File(coreLogsPath);
|
|
||||||
if (await coreLogFile.exists()) {
|
|
||||||
await coreLogFile.writeAsString("");
|
|
||||||
} else {
|
|
||||||
await coreLogFile.create(recursive: true);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static Future<Directory> getDatabaseDirectory() async {
|
static Future<Directory> getDatabaseDirectory() async {
|
||||||
@@ -90,11 +73,8 @@ class FilesEditorService with InfraLogger {
|
|||||||
return getApplicationDocumentsDirectory();
|
return getApplicationDocumentsDirectory();
|
||||||
}
|
}
|
||||||
|
|
||||||
String get appLogsPath => p.join(logsDir.path, "app.log");
|
|
||||||
String get coreLogsPath => p.join(logsDir.path, "box.log");
|
|
||||||
|
|
||||||
String configPath(String fileName) {
|
String configPath(String fileName) {
|
||||||
return p.join(_configsDir.path, "$fileName.json");
|
return p.join(configsDir.path, "$fileName.json");
|
||||||
}
|
}
|
||||||
|
|
||||||
String tempConfigPath(String fileName) => configPath("temp_$fileName");
|
String tempConfigPath(String fileName) => configPath("temp_$fileName");
|
||||||
|
|||||||
@@ -1,16 +1,48 @@
|
|||||||
import 'dart:convert';
|
import 'dart:convert';
|
||||||
|
import 'dart:io';
|
||||||
|
|
||||||
import 'package:flutter/services.dart';
|
import 'package:flutter/services.dart';
|
||||||
import 'package:fpdart/fpdart.dart';
|
import 'package:fpdart/fpdart.dart';
|
||||||
import 'package:freezed_annotation/freezed_annotation.dart';
|
import 'package:freezed_annotation/freezed_annotation.dart';
|
||||||
|
import 'package:hiddify/services/files_editor_service.dart';
|
||||||
import 'package:hiddify/utils/utils.dart';
|
import 'package:hiddify/utils/utils.dart';
|
||||||
|
import 'package:path_provider/path_provider.dart';
|
||||||
|
|
||||||
part 'platform_settings.freezed.dart';
|
part 'platform_services.freezed.dart';
|
||||||
part 'platform_settings.g.dart';
|
part 'platform_services.g.dart';
|
||||||
|
|
||||||
class PlatformSettings with InfraLogger {
|
class PlatformServices with InfraLogger {
|
||||||
late final MethodChannel _methodChannel =
|
final _methodChannel = const MethodChannel("app.hiddify.com/platform");
|
||||||
const MethodChannel("com.hiddify.app/platform.settings");
|
|
||||||
|
TaskEither<String, Directories> getPaths() {
|
||||||
|
return TaskEither(
|
||||||
|
() async {
|
||||||
|
loggy.debug("getting paths");
|
||||||
|
final Directories dirs;
|
||||||
|
if (Platform.isIOS) {
|
||||||
|
final paths = await _methodChannel.invokeMethod<Map>("get_paths");
|
||||||
|
loggy.debug("paths: $paths");
|
||||||
|
dirs = (
|
||||||
|
baseDir: Directory(paths?["base"]! as String),
|
||||||
|
workingDir: Directory(paths?["working"]! as String),
|
||||||
|
tempDir: Directory(paths?["temp"]! as String),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
final baseDir = await getApplicationSupportDirectory();
|
||||||
|
final workingDir = Platform.isAndroid
|
||||||
|
? await getExternalStorageDirectory()
|
||||||
|
: baseDir;
|
||||||
|
final tempDir = await getTemporaryDirectory();
|
||||||
|
dirs = (
|
||||||
|
baseDir: baseDir,
|
||||||
|
workingDir: workingDir!,
|
||||||
|
tempDir: tempDir,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return right(dirs);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
TaskEither<String, bool> isIgnoringBatteryOptimizations() {
|
TaskEither<String, bool> isIgnoringBatteryOptimizations() {
|
||||||
return TaskEither(
|
return TaskEither(
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
import 'package:hiddify/data/data_providers.dart';
|
import 'package:hiddify/data/data_providers.dart';
|
||||||
import 'package:hiddify/services/cron_service.dart';
|
import 'package:hiddify/services/cron_service.dart';
|
||||||
import 'package:hiddify/services/files_editor_service.dart';
|
import 'package:hiddify/services/files_editor_service.dart';
|
||||||
import 'package:hiddify/services/platform_settings.dart';
|
import 'package:hiddify/services/platform_services.dart';
|
||||||
import 'package:hiddify/services/singbox/singbox_service.dart';
|
import 'package:hiddify/services/singbox/singbox_service.dart';
|
||||||
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
||||||
|
|
||||||
@@ -9,14 +9,14 @@ part 'service_providers.g.dart';
|
|||||||
|
|
||||||
@Riverpod(keepAlive: true)
|
@Riverpod(keepAlive: true)
|
||||||
FilesEditorService filesEditorService(FilesEditorServiceRef ref) =>
|
FilesEditorService filesEditorService(FilesEditorServiceRef ref) =>
|
||||||
FilesEditorService();
|
FilesEditorService(ref.watch(platformServicesProvider));
|
||||||
|
|
||||||
@Riverpod(keepAlive: true)
|
@Riverpod(keepAlive: true)
|
||||||
SingboxService singboxService(SingboxServiceRef ref) => SingboxService();
|
SingboxService singboxService(SingboxServiceRef ref) => SingboxService();
|
||||||
|
|
||||||
@riverpod
|
@Riverpod(keepAlive: true)
|
||||||
PlatformSettings platformSettings(PlatformSettingsRef ref) =>
|
PlatformServices platformServices(PlatformServicesRef ref) =>
|
||||||
PlatformSettings();
|
PlatformServices();
|
||||||
|
|
||||||
@Riverpod(keepAlive: true)
|
@Riverpod(keepAlive: true)
|
||||||
CronService cronService(CronServiceRef ref) {
|
CronService cronService(CronServiceRef ref) {
|
||||||
|
|||||||
Reference in New Issue
Block a user