import 'dart:convert'; import 'dart:io'; import 'package:collection/collection.dart'; import 'package:dio/dio.dart'; import 'package:dio_http_cache_lts/dio_http_cache_lts.dart'; import 'package:flutter_cache_manager/flutter_cache_manager.dart'; import 'package:injectable/injectable.dart'; import 'package:revanced_manager/models/patch.dart'; import 'package:sentry_dio/sentry_dio.dart'; import 'package:sentry_flutter/sentry_flutter.dart'; @lazySingleton class GithubAPI { late Dio _dio = Dio(); final DioCacheManager _dioCacheManager = DioCacheManager(CacheConfig()); final Options _cacheOptions = buildCacheOptions( const Duration(hours: 6), maxStale: const Duration(days: 1), ); 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(_dioCacheManager.interceptor); _dio.addSentry( captureFailedRequests: true, ); } on Exception catch (e, s) { Sentry.captureException(e, stackTrace: s).ignore(); } } Future clearAllCache() async { try { await _dioCacheManager.clearAll(); } on Exception catch (e, s) { Sentry.captureException(e, stackTrace: s).ignore(); } } Future?> getLatestRelease(String repoName) async { try { final response = await _dio.get( '/repos/$repoName/releases', options: _cacheOptions, ); return response.data[0]; } on Exception catch (e, s) { Sentry.captureException(e, stackTrace: s).ignore(); 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(), }, options: _cacheOptions, ); 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, s) { Sentry.captureException(e, stackTrace: s).ignore(); return List.empty(); } } 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, s) { Sentry.captureException(e, stackTrace: s).ignore(); return null; } 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, s) { Sentry.captureException(e, stackTrace: s).ignore(); return List.empty(); } 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, s) { Sentry.captureException(e, stackTrace: s).ignore(); return ''; } } }