import 'package:cupertino_http/cupertino_http.dart'; import 'package:http_interceptor/http_interceptor.dart'; import 'package:mangayomi/eval/dart/model/m_bridge.dart'; import 'dart:async'; import 'dart:io'; import 'package:mangayomi/eval/dart/model/m_source.dart'; import 'package:mangayomi/main.dart'; import 'package:flutter_inappwebview/flutter_inappwebview.dart' as flutter_inappwebview; import 'package:mangayomi/models/settings.dart'; import 'package:cronet_http/cronet_http.dart'; import 'package:http/io_client.dart'; import 'package:mangayomi/utils/log/log.dart'; class MClient { MClient(); static Client httpClient() { if (Platform.isAndroid) { final engine = CronetEngine.build( enablePublicKeyPinningBypassForLocalTrustAnchors: true, enableHttp2: true, enableBrotli: true, cacheMode: CacheMode.memory, cacheMaxSize: 5 * 1024 * 1024); return CronetClient.fromCronetEngine(engine, closeEngine: true); } if (Platform.isIOS || Platform.isMacOS) { final config = URLSessionConfiguration.ephemeralSessionConfiguration() ..cache = URLCache.withCapacity(memoryCapacity: 5 * 1024 * 1024); return CupertinoClient.fromSessionConfiguration(config); } return IOClient(HttpClient()); } static InterceptedClient init( {MSource? source, Map? reqcopyWith}) { return InterceptedClient.build( client: httpClient(), interceptors: [MCookieManager(reqcopyWith), LoggerInterceptor()]); } static Map getCookiesPref(String url) { final cookiesList = isar.settings.getSync(227)!.cookiesList ?? []; if (cookiesList.isEmpty) return {}; final cookies = cookiesList .firstWhere( (element) => element.host == Uri.parse(url).host || Uri.parse(url).host.contains(element.host!), orElse: () => MCookie(cookie: ""), ) .cookie!; if (cookies.isEmpty) return {}; return {HttpHeaders.cookieHeader: cookies}; } static Future setCookie(String url, String ua, {String? cookie}) async { List cookies = []; if (Platform.isWindows || Platform.isLinux) { cookies = cookie ?.split(RegExp('(?<=)(,)(?=[^;]+?=)')) .where((cookie) => cookie.isNotEmpty) .toList() ?? []; } else { cookies = (await flutter_inappwebview.CookieManager.instance() .getCookies(url: Uri.parse(url))) .map((e) => "${e.name}=${e.value}") .toList(); } if (cookies.isNotEmpty) { final host = Uri.parse(url).host; final newCookie = cookies.join("; "); final settings = isar.settings.getSync(227); List? cookieList = []; for (var cookie in settings!.cookiesList ?? []) { if (cookie.host != host || (!host.contains(cookie.host))) { cookieList.add(cookie); } } cookieList.add(MCookie() ..host = host ..cookie = newCookie); isar.writeTxnSync( () => isar.settings.putSync(settings..cookiesList = cookieList)); } if (ua.isNotEmpty) { final settings = isar.settings.getSync(227); isar.writeTxnSync(() => isar.settings.putSync(settings!..userAgent = ua)); } } static void deleteAllCookies(String url) { final cookiesList = isar.settings.getSync(227)!.cookiesList ?? []; List? cookieList = []; for (var cookie in cookiesList) { if (!(cookie.host == Uri.parse(url).host || Uri.parse(url).host.contains(cookie.host!))) { cookieList.add(cookie); } } isar.writeTxnSync(() => isar.settings .putSync(isar.settings.getSync(227)!..cookiesList = cookieList)); } } class MCookieManager extends InterceptorContract { MCookieManager(this.reqcopyWith); Map? reqcopyWith; @override Future interceptRequest({ required BaseRequest request, }) async { final cookie = MClient.getCookiesPref(request.url.toString()); if (cookie.isNotEmpty) { final userAgent = isar.settings.getSync(227)!.userAgent!; if (request.headers[HttpHeaders.cookieHeader] == null) { request.headers.addAll(cookie); } if (request.headers[HttpHeaders.userAgentHeader] == null) { request.headers[HttpHeaders.userAgentHeader] = userAgent; } } try { if (reqcopyWith != null) { if (reqcopyWith!["followRedirects"] != null) { request.followRedirects = reqcopyWith!["followRedirects"]; } if (reqcopyWith!["maxRedirects"] != null) { request.maxRedirects = reqcopyWith!["maxRedirects"]; } if (reqcopyWith!["contentLength"] != null) { request.contentLength = reqcopyWith!["contentLength"]; } if (reqcopyWith!["persistentConnection"] != null) { request.persistentConnection = reqcopyWith!["persistentConnection"]; } } } catch (_) {} return request; } @override Future interceptResponse({ required BaseResponse response, }) async { return response; } } class LoggerInterceptor extends InterceptorContract { @override Future interceptRequest({ required BaseRequest request, }) async { Logger.add(LoggerLevel.info, '----- Request -----\n${request.toString()}\nheader: ${request.headers.toString()}'); return request; } @override Future interceptResponse({ required BaseResponse response, }) async { final cloudflare = [403, 503].contains(response.statusCode) && ["cloudflare-nginx", "cloudflare"].contains(response.headers["server"]); Logger.add(LoggerLevel.info, "----- Response -----\n${response.request?.method}: ${response.request?.url}, statusCode: ${response.statusCode} ${cloudflare ? "Failed to bypass Cloudflare" : ""}"); if (cloudflare) { botToast("${response.statusCode} Failed to bypass Cloudflare"); } return response; } }