import 'dart:convert'; import 'dart:io'; import 'package:flutter/services.dart'; import 'package:fpdart/fpdart.dart'; import 'package:hiddify/core/model/directories.dart'; import 'package:hiddify/singbox/model/singbox_config_option.dart'; import 'package:hiddify/singbox/model/singbox_outbound.dart'; import 'package:hiddify/singbox/model/singbox_stats.dart'; import 'package:hiddify/singbox/model/singbox_status.dart'; import 'package:hiddify/singbox/service/singbox_service.dart'; import 'package:hiddify/utils/custom_loggers.dart'; import 'package:rxdart/rxdart.dart'; class PlatformSingboxService with InfraLogger implements SingboxService { late final _methodChannel = const MethodChannel("com.hiddify.app/method"); late final _statusChannel = const EventChannel("com.hiddify.app/service.status", JSONMethodCodec()); late final _alertsChannel = const EventChannel("com.hiddify.app/service.alerts", JSONMethodCodec()); late final _logsChannel = const EventChannel("com.hiddify.app/service.logs"); late final ValueStream _status; @override Future init() async { loggy.debug("initializing"); final status = _statusChannel.receiveBroadcastStream().map(SingboxStatus.fromEvent); final alerts = _alertsChannel.receiveBroadcastStream().map(SingboxStatus.fromEvent); _status = ValueConnectableStream(Rx.merge([status, alerts])).autoConnect(); await _status.first; } @override TaskEither setup( Directories directories, bool debug, ) { return TaskEither( () async { if (!Platform.isIOS) { return right(unit); } await _methodChannel.invokeMethod("setup"); return right(unit); }, ); } @override TaskEither validateConfigByPath( String path, String tempPath, bool debug, ) { return TaskEither( () async { final message = await _methodChannel.invokeMethod( "parse_config", {"path": path, "tempPath": tempPath, "debug": debug}, ); if (message == null || message.isEmpty) return right(unit); return left(message); }, ); } @override TaskEither changeOptions(SingboxConfigOption options) { return TaskEither( () async { await _methodChannel.invokeMethod( "change_config_options", jsonEncode(options.toJson()), ); return right(unit); }, ); } @override TaskEither generateFullConfigByPath( String path, ) { return TaskEither( () async { final configJson = await _methodChannel.invokeMethod( "generate_config", {"path": path}, ); if (configJson == null || configJson.isEmpty) { return left("null response"); } return right(configJson); }, ); } @override TaskEither start( String path, String name, bool disableMemoryLimit, ) { return TaskEither( () async { loggy.debug("starting"); await _methodChannel.invokeMethod( "start", {"path": path, "name": name}, ); return right(unit); }, ); } @override TaskEither stop() { return TaskEither( () async { loggy.debug("stopping"); await _methodChannel.invokeMethod("stop"); return right(unit); }, ); } @override TaskEither restart( String path, String name, bool disableMemoryLimit, ) { return TaskEither( () async { loggy.debug("restarting"); await _methodChannel.invokeMethod( "restart", {"path": path, "name": name}, ); return right(unit); }, ); } @override TaskEither resetTunnel() { return TaskEither( () async { // only available on iOS (and macOS later) if (!Platform.isIOS) { throw UnimplementedError( "reset tunnel function unavailable on platform", ); } loggy.debug("resetting tunnel"); await _methodChannel.invokeMethod("reset"); return right(unit); }, ); } @override Stream> watchOutbounds() { const channel = EventChannel("com.hiddify.app/groups"); loggy.debug("watching outbounds"); return channel.receiveBroadcastStream().map( (event) { if (event case String _) { return (jsonDecode(event) as List).map((e) { return SingboxOutboundGroup.fromJson(e as Map); }).toList(); } loggy.error("[group client] unexpected type, msg: $event"); throw "invalid type"; }, ); } @override Stream watchStatus() => _status; @override Stream watchStats() { const channel = EventChannel("com.hiddify.app/stats", JSONMethodCodec()); loggy.debug("watching stats"); return channel.receiveBroadcastStream().map( (event) { if (event case Map _) { return SingboxStats.fromJson(event); } loggy.error("[stats client] unexpected type, msg: $event"); throw "invalid type"; }, ); } @override TaskEither selectOutbound(String groupTag, String outboundTag) { return TaskEither( () async { loggy.debug("selecting outbound"); await _methodChannel.invokeMethod( "select_outbound", {"groupTag": groupTag, "outboundTag": outboundTag}, ); return right(unit); }, ); } @override TaskEither urlTest(String groupTag) { return TaskEither( () async { await _methodChannel.invokeMethod( "url_test", {"groupTag": groupTag}, ); return right(unit); }, ); } @override Stream> watchLogs(String path) async* { yield* _logsChannel .receiveBroadcastStream() .map((event) => (event as List).map((e) => e as String).toList()); } @override TaskEither clearLogs() { return TaskEither( () async { await _methodChannel.invokeMethod("clear_logs"); return right(unit); }, ); } }