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:revanced_manager/app/app.locator.dart'; import 'package:revanced_manager/services/manager_api.dart'; @lazySingleton class GithubAPI { late Dio _dio = Dio(); late final ManagerAPI _managerAPI = locator(); final _cacheOptions = CacheOptions( store: MemCacheStore(), maxStale: const Duration(days: 1), priority: CachePriority.high, ); Future initialize(String repoUrl) async { try { _dio = Dio( BaseOptions( baseUrl: repoUrl, ), ); _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?> getLatestRelease( String repoName, ) async { try { final response = await _dio.get( '/repos/$repoName/releases', ); return response.data[0]; } on Exception catch (e) { if (kDebugMode) { print(e); } return null; } } Future?> getPatchesRelease( String repoName, String version, ) async { try { final response = await _dio.get( '/repos/$repoName/releases/tags/$version', ); return response.data; } on Exception catch (e) { if (kDebugMode) { print(e); } return null; } } Future?> getLatestPatchesRelease( String repoName, ) async { try { final response = await _dio.get( '/repos/$repoName/releases/latest', ); return response.data; } on Exception catch (e) { if (kDebugMode) { print(e); } return null; } } Future?> getLatestManagerRelease( String repoName, ) async { try { final response = await _dio.get( '/repos/$repoName/releases', ); final Map releases = response.data[0]; int updates = 0; final String currentVersion = await ManagerAPI().getCurrentManagerVersion(); while (response.data[updates]['tag_name'] != 'v$currentVersion') { updates++; } for (int i = 1; i < updates; i++) { releases.update( 'body', (value) => value + '\n' + '# ' + response.data[i]['tag_name'] + '\n' + response.data[i]['body'], ); } return releases; } on Exception catch (e) { if (kDebugMode) { print(e); } return null; } } Future getLatestReleaseFile( String extension, String repoName, ) async { try { final Map? release = await getLatestRelease(repoName); if (release != null) { final Map? asset = (release['assets'] as List).firstWhereOrNull( (asset) => (asset['name'] as String).endsWith(extension), ); if (asset != null) { return await DefaultCacheManager().getSingleFile( asset['browser_download_url'], ); } } } on Exception catch (e) { if (kDebugMode) { print(e); } } return null; } Future getPatchesReleaseFile( String extension, String repoName, String version, String url, ) async { try { if (url.isNotEmpty) { return await DefaultCacheManager().getSingleFile( url, ); } final Map? release = await getPatchesRelease(repoName, version); if (release != null) { final Map? asset = (release['assets'] as List).firstWhereOrNull( (asset) => (asset['name'] as String).endsWith(extension), ); if (asset != null) { final String downloadUrl = asset['browser_download_url']; if (extension == '.apk') { _managerAPI.setIntegrationsDownloadURL(downloadUrl); } else { _managerAPI.setPatchesDownloadURL(downloadUrl); } return await DefaultCacheManager().getSingleFile( downloadUrl, ); } } } on Exception catch (e) { if (kDebugMode) { print(e); } } return null; } }