import 'dart:async'; import 'dart:developer'; import 'dart:io'; import 'package:collection/collection.dart'; import 'package:dio/dio.dart'; import 'package:dio_cache_interceptor/dio_cache_interceptor.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter_cache_manager/flutter_cache_manager.dart'; import 'package:injectable/injectable.dart'; import 'package:native_dio_adapter/native_dio_adapter.dart'; import 'package:revanced_manager/models/patch.dart'; import 'package:revanced_manager/utils/check_for_gms.dart'; import 'package:timeago/timeago.dart'; @lazySingleton class RevancedAPI { late Dio _dio = Dio(); final _cacheOptions = CacheOptions( store: MemCacheStore(), maxStale: const Duration(days: 1), priority: CachePriority.high, ); Future initialize(String apiUrl) async { try { final bool isGMSInstalled = await checkForGMS(); if (!isGMSInstalled) { _dio = Dio( BaseOptions( baseUrl: apiUrl, ), ); log('ReVanced API: Using default engine + $isGMSInstalled'); } else { if (Platform.isIOS || Platform.isMacOS || Platform.isAndroid) { final CronetEngine androidCronetEngine = await CronetEngine.build( userAgent: 'ReVanced Manager', enableBrotli: true, enableQuic: true, ); _dio.httpClientAdapter = NativeAdapter(androidCronetEngine: androidCronetEngine); _dio = Dio( BaseOptions( baseUrl: apiUrl, ), ); } log('ReVanced API: Using CronetEngine + $isGMSInstalled'); } _dio.interceptors.add(DioCacheInterceptor(options: _cacheOptions)); } on Exception catch (e) { if (kDebugMode) { print(e); } } } Future clearAllCache() async { try { await _cacheOptions.store!.clean(); } on Exception catch (e) { if (kDebugMode) { print(e); } } } Future>> getContributors() async { final Map> contributors = {}; try { final response = await _dio.get('/contributors'); final List repositories = response.data['repositories']; for (final Map repo in repositories) { final String name = repo['name']; contributors[name] = repo['contributors']; } } on Exception catch (e) { if (kDebugMode) { print(e); } return {}; } return contributors; } Future> getPatches() async { try { final response = await _dio.get('/patches'); final List patches = response.data; return patches.map((patch) => Patch.fromJson(patch)).toList(); } on Exception catch (e) { if (kDebugMode) { print(e); } return List.empty(); } } Future?> _getLatestRelease( String extension, String repoName, ) async { try { final response = await _dio.get('/tools'); final List tools = response.data['tools']; return tools.firstWhereOrNull( (t) => t['repository'] == repoName && (t['name'] as String).endsWith(extension), ); } on Exception catch (e) { if (kDebugMode) { print(e); } return null; } } Future getLatestReleaseVersion( String extension, String repoName, ) async { try { final Map? release = await _getLatestRelease( extension, repoName, ); if (release != null) { return release['version']; } } on Exception catch (e) { if (kDebugMode) { print(e); } return null; } return null; } Future getLatestReleaseFile( String extension, String repoName, ) async { try { final Map? release = await _getLatestRelease( extension, repoName, ); if (release != null) { final String url = release['browser_download_url']; return await DefaultCacheManager().getSingleFile(url); } } on Exception catch (e) { if (kDebugMode) { print(e); } return null; } return null; } StreamController managerUpdateProgress = StreamController(); void updateManagerDownloadProgress(int progress) { managerUpdateProgress.add(progress.toDouble()); } Stream getManagerUpdateProgress() { return managerUpdateProgress.stream; } void disposeManagerUpdateProgress() { managerUpdateProgress.close(); } Future downloadManager() async { final Map? release = await _getLatestRelease( '.apk', 'revanced/revanced-manager', ); File? outputFile; await for (final result in DefaultCacheManager().getFileStream( release!['browser_download_url'] as String, withProgress: true, )) { if (result is DownloadProgress) { final totalSize = result.totalSize ?? 10000000; final progress = (result.downloaded / totalSize * 100).round(); updateManagerDownloadProgress(progress); } else if (result is FileInfo) { // The download is complete; convert the FileInfo object to a File object outputFile = File(result.file.path); } } return outputFile; } Future getLatestReleaseTime( String extension, String repoName, ) async { try { final Map? release = await _getLatestRelease( extension, repoName, ); if (release != null) { final DateTime timestamp = DateTime.parse(release['timestamp'] as String); return format(timestamp, locale: 'en_short'); } } on Exception catch (e) { if (kDebugMode) { print(e); } return null; } return null; } }