Change app http client
This commit is contained in:
95
lib/core/http_client/dio_http_client.dart
Normal file
95
lib/core/http_client/dio_http_client.dart
Normal 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,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -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;
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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));
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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());
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|||||||
@@ -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(
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|||||||
@@ -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)
|
||||||
|
|||||||
16
pubspec.lock
16
pubspec.lock
@@ -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:
|
||||||
|
|||||||
@@ -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:
|
||||||
|
|||||||
Reference in New Issue
Block a user