Change app http client

This commit is contained in:
problematicconsumer
2024-01-03 21:15:55 +03:30
parent 7c8e632d86
commit 58f3e938a6
10 changed files with 138 additions and 38 deletions

View File

@@ -0,0 +1,95 @@
import 'dart:convert';
import 'package:dio/dio.dart';
import 'package:dio_smart_retry/dio_smart_retry.dart';
import 'package:flutter_loggy_dio/flutter_loggy_dio.dart';
import 'package:hiddify/utils/custom_loggers.dart';
class DioHttpClient with InfraLogger {
DioHttpClient({
required Duration timeout,
required String userAgent,
required bool debug,
}) {
_dio = Dio(
BaseOptions(
connectTimeout: timeout,
sendTimeout: timeout,
receiveTimeout: timeout,
headers: {"User-Agent": userAgent},
),
);
_dio.interceptors.add(
RetryInterceptor(
dio: _dio,
retryDelays: const [
Duration(seconds: 1),
Duration(seconds: 2),
Duration(seconds: 3),
],
),
);
if (debug) {
_dio.interceptors.add(LoggyDioInterceptor(requestHeader: true));
}
}
late final Dio _dio;
Future<Response<T>> get<T>(
String url, {
CancelToken? cancelToken,
String? userAgent,
({String username, String password})? credentials,
}) async {
return _dio.get<T>(
url,
cancelToken: cancelToken,
options: _options(url, userAgent: userAgent, credentials: credentials),
);
}
Future<Response> download(
String url,
String path, {
CancelToken? cancelToken,
String? userAgent,
({String username, String password})? credentials,
}) async {
return _dio.download(
url,
path,
cancelToken: cancelToken,
options: _options(url, userAgent: userAgent, credentials: credentials),
);
}
Options _options(
String url, {
String? userAgent,
({String username, String password})? credentials,
}) {
final uri = Uri.parse(url);
String? userInfo;
if (credentials != null) {
userInfo = "${credentials.username}:${credentials.password}";
} else if (uri.userInfo.isNotEmpty) {
userInfo = uri.userInfo;
}
String? basicAuth;
if (userInfo != null) {
basicAuth = "Basic ${base64.encode(utf8.encode(userInfo))}";
}
return Options(
headers: {
if (userAgent != null) "User-Agent": userAgent,
if (basicAuth != null) "authorization": basicAuth,
},
);
}
}

View File

@@ -1,26 +1,15 @@
import 'package:dio/dio.dart'; import 'package:flutter/foundation.dart';
import 'package:hiddify/core/app_info/app_info_provider.dart'; import 'package:hiddify/core/app_info/app_info_provider.dart';
import 'package:hiddify/core/http_client/dio_http_client.dart';
import 'package:riverpod_annotation/riverpod_annotation.dart'; import 'package:riverpod_annotation/riverpod_annotation.dart';
part 'http_client_provider.g.dart'; part 'http_client_provider.g.dart';
@Riverpod(keepAlive: true) @Riverpod(keepAlive: true)
Dio httpClient(HttpClientRef ref) { DioHttpClient httpClient(HttpClientRef ref) {
final dio = Dio( return DioHttpClient(
BaseOptions( timeout: const Duration(seconds: 15),
connectTimeout: const Duration(seconds: 15), userAgent: ref.watch(appInfoProvider).requireValue.userAgent,
sendTimeout: const Duration(seconds: 15), debug: kDebugMode,
receiveTimeout: const Duration(seconds: 15),
headers: {
"User-Agent": ref.watch(appInfoProvider).requireValue.userAgent,
},
),
); );
// https://github.com/dart-lang/http/issues/1047
// https://github.com/cfug/dio/issues/2042
// final debug = ref.read(debugModeNotifierProvider);
// if (debug && (Platform.isAndroid || Platform.isIOS || Platform.isMacOS)) {
// dio.httpClientAdapter = NativeAdapter();
// }
return dio;
} }

View File

@@ -8,5 +8,5 @@ part 'app_update_data_providers.g.dart';
AppUpdateRepository appUpdateRepository( AppUpdateRepository appUpdateRepository(
AppUpdateRepositoryRef ref, AppUpdateRepositoryRef ref,
) { ) {
return AppUpdateRepositoryImpl(dio: ref.watch(httpClientProvider)); return AppUpdateRepositoryImpl(httpClient: ref.watch(httpClientProvider));
} }

View File

@@ -1,5 +1,5 @@
import 'package:dio/dio.dart';
import 'package:fpdart/fpdart.dart'; import 'package:fpdart/fpdart.dart';
import 'package:hiddify/core/http_client/dio_http_client.dart';
import 'package:hiddify/core/model/constants.dart'; import 'package:hiddify/core/model/constants.dart';
import 'package:hiddify/core/model/environment.dart'; import 'package:hiddify/core/model/environment.dart';
import 'package:hiddify/core/utils/exception_handler.dart'; import 'package:hiddify/core/utils/exception_handler.dart';
@@ -18,9 +18,9 @@ abstract interface class AppUpdateRepository {
class AppUpdateRepositoryImpl class AppUpdateRepositoryImpl
with ExceptionHandler, InfraLogger with ExceptionHandler, InfraLogger
implements AppUpdateRepository { implements AppUpdateRepository {
AppUpdateRepositoryImpl({required this.dio}); AppUpdateRepositoryImpl({required this.httpClient});
final Dio dio; final DioHttpClient httpClient;
@override @override
TaskEither<AppUpdateFailure, RemoteVersionEntity> getLatestVersion({ TaskEither<AppUpdateFailure, RemoteVersionEntity> getLatestVersion({
@@ -32,7 +32,8 @@ class AppUpdateRepositoryImpl
if (!release.allowCustomUpdateChecker) { if (!release.allowCustomUpdateChecker) {
throw Exception("custom update checkers are not supported"); throw Exception("custom update checkers are not supported");
} }
final response = await dio.get<List>(Constants.githubReleasesApiUrl); final response =
await httpClient.get<List>(Constants.githubReleasesApiUrl);
if (response.statusCode != 200 || response.data == null) { if (response.statusCode != 200 || response.data == null) {
loggy.warning("failed to fetch latest version info"); loggy.warning("failed to fetch latest version info");
return left(const AppUpdateFailure.unexpected()); return left(const AppUpdateFailure.unexpected());

View File

@@ -13,7 +13,7 @@ Future<GeoAssetRepository> geoAssetRepository(GeoAssetRepositoryRef ref) async {
final repo = GeoAssetRepositoryImpl( final repo = GeoAssetRepositoryImpl(
geoAssetDataSource: ref.watch(geoAssetDataSourceProvider), geoAssetDataSource: ref.watch(geoAssetDataSourceProvider),
geoAssetPathResolver: ref.watch(geoAssetPathResolverProvider), geoAssetPathResolver: ref.watch(geoAssetPathResolverProvider),
dio: ref.watch(httpClientProvider), httpClient: ref.watch(httpClientProvider),
); );
await repo.init().getOrElse((l) => throw l).run(); await repo.init().getOrElse((l) => throw l).run();
return repo; return repo;

View File

@@ -1,11 +1,11 @@
import 'dart:io'; import 'dart:io';
import 'package:dartx/dartx_io.dart'; import 'package:dartx/dartx_io.dart';
import 'package:dio/dio.dart';
import 'package:drift/drift.dart'; import 'package:drift/drift.dart';
import 'package:flutter/services.dart'; import 'package:flutter/services.dart';
import 'package:fpdart/fpdart.dart'; import 'package:fpdart/fpdart.dart';
import 'package:hiddify/core/database/app_database.dart'; import 'package:hiddify/core/database/app_database.dart';
import 'package:hiddify/core/http_client/dio_http_client.dart';
import 'package:hiddify/core/utils/exception_handler.dart'; import 'package:hiddify/core/utils/exception_handler.dart';
import 'package:hiddify/features/geo_asset/data/geo_asset_data_mapper.dart'; import 'package:hiddify/features/geo_asset/data/geo_asset_data_mapper.dart';
import 'package:hiddify/features/geo_asset/data/geo_asset_data_source.dart'; import 'package:hiddify/features/geo_asset/data/geo_asset_data_source.dart';
@@ -35,12 +35,12 @@ class GeoAssetRepositoryImpl
GeoAssetRepositoryImpl({ GeoAssetRepositoryImpl({
required this.geoAssetDataSource, required this.geoAssetDataSource,
required this.geoAssetPathResolver, required this.geoAssetPathResolver,
required this.dio, required this.httpClient,
}); });
final GeoAssetDataSource geoAssetDataSource; final GeoAssetDataSource geoAssetDataSource;
final GeoAssetPathResolver geoAssetPathResolver; final GeoAssetPathResolver geoAssetPathResolver;
final Dio dio; final DioHttpClient httpClient;
@override @override
TaskEither<GeoAssetFailure, Unit> init() { TaskEither<GeoAssetFailure, Unit> init() {
@@ -141,7 +141,7 @@ class GeoAssetRepositoryImpl
loggy.debug( loggy.debug(
"checking latest release of [${geoAsset.name}] on [${geoAsset.repositoryUrl}]", "checking latest release of [${geoAsset.name}] on [${geoAsset.repositoryUrl}]",
); );
final response = await dio.get<Map>(geoAsset.repositoryUrl); final response = await httpClient.get<Map>(geoAsset.repositoryUrl);
if (response.statusCode != 200 || response.data == null) { if (response.statusCode != 200 || response.data == null) {
return left( return left(
GeoAssetUnexpectedFailure.new( GeoAssetUnexpectedFailure.new(
@@ -180,7 +180,7 @@ class GeoAssetRepositoryImpl
loggy.debug("[${geoAsset.name}] download url: [$downloadUrl]"); loggy.debug("[${geoAsset.name}] download url: [$downloadUrl]");
final tempPath = "${file.path}.tmp"; final tempPath = "${file.path}.tmp";
await file.parent.create(recursive: true); await file.parent.create(recursive: true);
await dio.download(downloadUrl, tempPath); await httpClient.download(downloadUrl, tempPath);
await File(tempPath).rename(file.path); await File(tempPath).rename(file.path);
await geoAssetDataSource.patch( await geoAssetDataSource.patch(

View File

@@ -15,7 +15,7 @@ Future<ProfileRepository> profileRepository(ProfileRepositoryRef ref) async {
profileDataSource: ref.watch(profileDataSourceProvider), profileDataSource: ref.watch(profileDataSourceProvider),
profilePathResolver: ref.watch(profilePathResolverProvider), profilePathResolver: ref.watch(profilePathResolverProvider),
singbox: ref.watch(singboxServiceProvider), singbox: ref.watch(singboxServiceProvider),
dio: ref.watch(httpClientProvider), httpClient: ref.watch(httpClientProvider),
); );
await repo.init().getOrElse((l) => throw l).run(); await repo.init().getOrElse((l) => throw l).run();
return repo; return repo;

View File

@@ -1,9 +1,9 @@
import 'dart:io'; import 'dart:io';
import 'package:dio/dio.dart';
import 'package:drift/drift.dart'; import 'package:drift/drift.dart';
import 'package:fpdart/fpdart.dart'; import 'package:fpdart/fpdart.dart';
import 'package:hiddify/core/database/app_database.dart'; import 'package:hiddify/core/database/app_database.dart';
import 'package:hiddify/core/http_client/dio_http_client.dart';
import 'package:hiddify/core/utils/exception_handler.dart'; import 'package:hiddify/core/utils/exception_handler.dart';
import 'package:hiddify/features/profile/data/profile_data_mapper.dart'; import 'package:hiddify/features/profile/data/profile_data_mapper.dart';
import 'package:hiddify/features/profile/data/profile_data_source.dart'; import 'package:hiddify/features/profile/data/profile_data_source.dart';
@@ -16,7 +16,6 @@ import 'package:hiddify/singbox/service/singbox_service.dart';
import 'package:hiddify/utils/custom_loggers.dart'; import 'package:hiddify/utils/custom_loggers.dart';
import 'package:hiddify/utils/link_parsers.dart'; import 'package:hiddify/utils/link_parsers.dart';
import 'package:meta/meta.dart'; import 'package:meta/meta.dart';
import 'package:retry/retry.dart';
import 'package:uuid/uuid.dart'; import 'package:uuid/uuid.dart';
abstract interface class ProfileRepository { abstract interface class ProfileRepository {
@@ -63,13 +62,13 @@ class ProfileRepositoryImpl
required this.profileDataSource, required this.profileDataSource,
required this.profilePathResolver, required this.profilePathResolver,
required this.singbox, required this.singbox,
required this.dio, required this.httpClient,
}); });
final ProfileDataSource profileDataSource; final ProfileDataSource profileDataSource;
final ProfilePathResolver profilePathResolver; final ProfilePathResolver profilePathResolver;
final SingboxService singbox; final SingboxService singbox;
final Dio dio; final DioHttpClient httpClient;
@override @override
TaskEither<ProfileFailure, Unit> init() { TaskEither<ProfileFailure, Unit> init() {
@@ -366,11 +365,9 @@ class ProfileRepositoryImpl
() async { () async {
final file = profilePathResolver.file(fileName); final file = profilePathResolver.file(fileName);
final tempFile = profilePathResolver.tempFile(fileName); final tempFile = profilePathResolver.tempFile(fileName);
try { try {
final response = await retry( final response = await httpClient.download(url.trim(), tempFile.path);
() async => dio.download(url.trim(), tempFile.path),
maxAttempts: 3,
);
final headers = final headers =
await _populateHeaders(response.headers.map, tempFile.path); await _populateHeaders(response.headers.map, tempFile.path);
return await validateConfig(file.path, tempFile.path, false) return await validateConfig(file.path, tempFile.path, false)

View File

@@ -353,6 +353,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "5.4.0" version: "5.4.0"
dio_smart_retry:
dependency: "direct main"
description:
name: dio_smart_retry
sha256: "3d71450c19b4d91ef4c7d726a55a284bfc11eb3634f1f25006cdfab3f8595653"
url: "https://pub.dev"
source: hosted
version: "6.0.0"
drift: drift:
dependency: "direct main" dependency: "direct main"
description: description:
@@ -499,6 +507,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "2.0.2" version: "2.0.2"
flutter_loggy_dio:
dependency: "direct main"
description:
name: flutter_loggy_dio
sha256: "682fe1fe136ae11cc9094364a4256d831b8cbe627b1370bb8fb9f135415265e2"
url: "https://pub.dev"
source: hosted
version: "3.0.1"
flutter_native_splash: flutter_native_splash:
dependency: "direct main" dependency: "direct main"
description: description:

View File

@@ -72,6 +72,8 @@ dependencies:
native_dio_adapter: ^1.2.0 native_dio_adapter: ^1.2.0
flutter_displaymode: ^0.6.0 flutter_displaymode: ^0.6.0
windows_single_instance: ^1.0.1 windows_single_instance: ^1.0.1
flutter_loggy_dio: ^3.0.1
dio_smart_retry: ^6.0.0
dev_dependencies: dev_dependencies:
flutter_test: flutter_test: