import 'dart:convert'; 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/models/patch.dart'; import 'package:revanced_manager/services/manager_api.dart'; @lazySingleton class GithubAPI { late Dio _dio = Dio(); final _cacheOptions = CacheOptions( store: MemCacheStore(), maxStale: const Duration(days: 1), priority: CachePriority.high, ); final Map repoAppPath = { 'com.google.android.youtube': 'youtube', 'com.google.android.apps.youtube.music': 'music', 'com.twitter.android': 'twitter', 'com.reddit.frontpage': 'reddit', 'com.zhiliaoapp.musically': 'tiktok', 'de.dwd.warnapp': 'warnwetter', 'com.garzotto.pflotsh.ecmwf_a': 'ecmwf', 'com.spotify.music': 'spotify', }; 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?> 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> getCommits( String packageName, String repoName, DateTime since, ) async { final String path = 'src/main/kotlin/app/revanced/patches/${repoAppPath[packageName]}'; try { final response = await _dio.get( '/repos/$repoName/commits', queryParameters: { 'path': path, 'since': since.toIso8601String(), }, ); final List commits = response.data; return commits .map( (commit) => commit['commit']['message'].split('\n')[0] + ' - ' + commit['commit']['author']['name'] + '\n' as String, ) .toList(); } on Exception catch (e) { if (kDebugMode) { print(e); } } return []; } 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> getPatches(String repoName) async { List patches = []; try { final File? f = await getLatestReleaseFile('.json', repoName); if (f != null) { final List list = jsonDecode(f.readAsStringSync()); patches = list.map((patch) => Patch.fromJson(patch)).toList(); } } on Exception catch (e) { if (kDebugMode) { print(e); } } return patches; } Future getLastestReleaseVersion(String repoName) async { try { final Map? release = await getLatestRelease(repoName); if (release != null) { return release['tag_name']; } else { return 'Unknown'; } } on Exception catch (e) { if (kDebugMode) { print(e); } return 'Unknown'; } } }