Files
PiliPlus/lib/utils/accounts/account_manager/account_mgr.dart
My-Responsitories 521ca3ad18 tweaks (#1788)
* tweak

* opt: show bar

* opt: crc32

* opt: appsign

* opt: Get

* opt: compress only if large

* opt: wbi

* tweak

Signed-off-by: bggRGjQaUbCoE <githubaccount56556@proton.me>

---------

Signed-off-by: My-Responsitories <107370289+My-Responsitories@users.noreply.github.com>
Co-authored-by: bggRGjQaUbCoE <githubaccount56556@proton.me>
2025-12-23 12:57:19 +08:00

272 lines
8.7 KiB
Dart

// edit from package:dio_cookie_manager
import 'dart:async';
import 'dart:io';
import 'package:PiliPlus/http/api.dart';
import 'package:PiliPlus/http/constants.dart';
import 'package:PiliPlus/models/common/account_type.dart';
import 'package:PiliPlus/utils/accounts.dart';
import 'package:PiliPlus/utils/accounts/account.dart';
import 'package:PiliPlus/utils/accounts/api_type.dart';
import 'package:PiliPlus/utils/app_sign.dart';
import 'package:PiliPlus/utils/extension/string_ext.dart';
import 'package:PiliPlus/utils/platform_utils.dart';
import 'package:PiliPlus/utils/storage_pref.dart';
import 'package:connectivity_plus/connectivity_plus.dart';
import 'package:dio/dio.dart';
import 'package:flutter/foundation.dart' show kDebugMode;
import 'package:flutter/material.dart';
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
final _setCookieReg = RegExp('(?<=)(,)(?=[^;]+?=)');
class AccountManager extends Interceptor {
AccountManager();
String blockServer = Pref.blockServer;
static String getCookies(List<Cookie> cookies) {
// Sort cookies by path (longer path first).
cookies.sort((a, b) {
if (a.path == null && b.path == null) {
return 0;
} else if (a.path == null) {
return -1;
} else if (b.path == null) {
return 1;
} else {
return b.path!.length.compareTo(a.path!.length);
}
});
return cookies.map((cookie) => '${cookie.name}=${cookie.value}').join('; ');
}
@override
void onRequest(RequestOptions options, RequestInterceptorHandler handler) {
final path = options.path;
late final Account account = options.extra['account'] ?? _findAccount(path);
if (account is NoAccount || _skipCookie(path)) return handler.next(options);
if (!account.isLogin && path == Api.heartBeat) {
return handler.reject(
DioException.requestCancelled(requestOptions: options, reason: null),
false,
);
}
final isApp = path.startsWith(HttpString.appBaseUrl);
if (isApp && options.responseType == ResponseType.bytes) {
options.headers.addAll(account.grpcHeaders);
return handler.next(options);
}
options.headers
..addAll(account.headers)
..['referer'] ??= HttpString.baseUrl;
// app端不需要管理cookie
if (isApp) {
// if (kDebugMode) debugPrint('is app: ${options.path}');
final dataPtr = (options.method == 'POST' && options.data is Map
? (options.data as Map).cast<String, dynamic>()
: options.queryParameters);
if (dataPtr.isNotEmpty) {
if (!account.accessKey.isNullOrEmpty) {
dataPtr['access_key'] = account.accessKey!;
}
AppSign.appSign(dataPtr);
// if (kDebugMode) debugPrint(dataPtr.toString());
}
return handler.next(options);
} else {
account.cookieJar
.loadForRequest(options.uri)
.then((cookies) {
final previousCookies =
options.headers[HttpHeaders.cookieHeader] as String?;
final newCookies = getCookies([
...?previousCookies
?.split(';')
.where((e) => e.isNotEmpty)
.map(Cookie.fromSetCookieValue),
...cookies,
]);
options.headers[HttpHeaders.cookieHeader] = newCookies.isNotEmpty
? newCookies
: '';
handler.next(options);
})
.catchError((dynamic e, StackTrace s) {
final err = DioException(
requestOptions: options,
error: e,
stackTrace: s,
);
handler.reject(err, true);
});
}
}
@override
void onResponse(Response response, ResponseInterceptorHandler handler) {
final options = response.requestOptions;
final path = options.path;
if (options.extra['account'] is NoAccount ||
path.startsWith(HttpString.appBaseUrl) ||
_skipCookie(path)) {
return handler.next(response);
} else {
final future = _saveCookies(
response,
).whenComplete(() => handler.next(response));
assert(() {
future.catchError(
(Object e, StackTrace s) {
throw DioException(
requestOptions: response.requestOptions,
error: e,
stackTrace: s,
);
},
);
return true;
}());
}
}
@override
void onError(DioException err, ErrorInterceptorHandler handler) {
if (err.requestOptions.responseType == ResponseType.stream) {
return handler.next(err);
}
if (err.requestOptions.method != 'POST') {
toast(err);
}
if (err.response != null &&
!err.response!.requestOptions.path.startsWith(HttpString.appBaseUrl)) {
_saveCookies(
err.response!,
).whenComplete(() => handler.next(err)).catchError(
(dynamic e, StackTrace s) {
final error = DioException(
requestOptions: err.response!.requestOptions,
error: e,
stackTrace: s,
);
handler.next(error);
},
);
} else {
handler.next(err);
}
}
static void toast(DioException err) {
const List<String> skipShow = [
'heartbeat',
'history/report',
'roomEntryAction',
'seg.so',
'online/total',
'github',
'hdslb.com',
'biliimg.com',
'site/getCoin',
];
String url = err.requestOptions.uri.toString();
if (kDebugMode) debugPrint('🌹🌹ApiInterceptor: $url');
if (skipShow.any((i) => url.contains(i)) ||
(url.contains('skipSegments') && err.requestOptions.method == 'GET')) {
// skip
} else {
dioError(err).then((res) => SmartDialog.showToast(res + url));
}
}
Future<void> _saveCookies(Response response) async {
final Account account =
response.requestOptions.extra['account'] ??
_findAccount(response.requestOptions.path);
final setCookies = response.headers[HttpHeaders.setCookieHeader];
if (setCookies == null || setCookies.isEmpty) {
return;
}
final List<Cookie> cookies = setCookies
.map((str) => str.split(_setCookieReg))
.expand((cookie) => cookie)
.where((cookie) => cookie.isNotEmpty)
.map(Cookie.fromSetCookieValue)
.toList();
final statusCode = response.statusCode ?? 0;
final locations = response.headers[HttpHeaders.locationHeader] ?? const [];
final isRedirectRequest = statusCode >= 300 && statusCode < 400;
final originalUri = response.requestOptions.uri;
final realUri = originalUri.resolveUri(response.realUri);
await account.cookieJar.saveFromResponse(realUri, cookies);
if (isRedirectRequest && locations.isNotEmpty) {
final originalUri = response.realUri;
await Future.wait(
locations.map(
(location) => account.cookieJar.saveFromResponse(
// Resolves the location based on the current Uri.
originalUri.resolve(location),
cookies,
),
),
);
}
await account.onChange();
}
bool _skipCookie(String path) {
return path.startsWith(blockServer) ||
path.contains('hdslb.com') ||
path.contains('biliimg.com');
}
Account _findAccount(String path) => ApiType.loginApi.contains(path)
? AnonymousAccount()
: Accounts.get(
AccountType.values.firstWhere(
(i) => ApiType.apiTypeSet[i]?.contains(path) == true,
orElse: () => AccountType.main,
),
);
static Future<String> dioError(DioException error) async {
switch (error.type) {
case DioExceptionType.badCertificate:
return '证书有误!';
case DioExceptionType.badResponse:
return '服务器异常,请稍后重试!';
case DioExceptionType.cancel:
return '请求已被取消,请重新请求';
case DioExceptionType.connectionError:
return '连接错误,请检查网络设置';
case DioExceptionType.connectionTimeout:
return '网络连接超时,请检查网络设置';
case DioExceptionType.receiveTimeout:
return '响应超时,请稍后重试!';
case DioExceptionType.sendTimeout:
return '发送请求超时,请检查网络设置';
case DioExceptionType.unknown:
String desc;
try {
desc = PlatformUtils.isMobile
? (await Connectivity().checkConnectivity()).first.desc
: '';
} catch (_) {
desc = '';
}
return '$desc网络异常 ${error.error}';
}
}
}
extension _ConnectivityResultExt on ConnectivityResult {
String get desc => const ['蓝牙', 'Wi-Fi', '局域', '流量', '', '代理', '其他'][index];
}