2022-09-11 03:01:06 +02:00
|
|
|
import 'dart:convert';
|
2022-08-18 16:33:33 +02:00
|
|
|
import 'dart:io';
|
2022-09-11 03:01:06 +02:00
|
|
|
import 'package:collection/collection.dart';
|
|
|
|
import 'package:dio/dio.dart';
|
|
|
|
import 'package:dio_http_cache_lts/dio_http_cache_lts.dart';
|
2022-08-18 16:33:33 +02:00
|
|
|
import 'package:flutter_cache_manager/flutter_cache_manager.dart';
|
2022-09-12 02:41:53 +02:00
|
|
|
import 'package:injectable/injectable.dart';
|
2022-10-07 14:53:43 +02:00
|
|
|
import 'package:native_dio_client/native_dio_client.dart';
|
2022-09-11 03:01:06 +02:00
|
|
|
import 'package:revanced_manager/models/patch.dart';
|
2022-10-11 16:49:50 +02:00
|
|
|
import 'package:revanced_manager/utils/check_for_gms.dart';
|
2022-07-31 21:46:27 +02:00
|
|
|
|
2022-09-12 02:41:53 +02:00
|
|
|
@lazySingleton
|
2022-07-31 21:46:27 +02:00
|
|
|
class GithubAPI {
|
2022-10-11 16:49:50 +02:00
|
|
|
late Dio _dio = Dio();
|
2022-09-12 02:41:53 +02:00
|
|
|
final DioCacheManager _dioCacheManager = DioCacheManager(CacheConfig());
|
|
|
|
final Options _cacheOptions = buildCacheOptions(
|
2022-09-28 18:56:54 +02:00
|
|
|
const Duration(hours: 6),
|
|
|
|
maxStale: const Duration(days: 1),
|
2022-09-11 03:01:06 +02:00
|
|
|
);
|
2022-08-30 03:07:28 +02:00
|
|
|
final Map<String, String> 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',
|
2022-10-10 15:27:32 +02:00
|
|
|
'com.spotify.music': 'spotify',
|
2022-08-30 03:07:28 +02:00
|
|
|
};
|
|
|
|
|
2022-10-11 16:49:50 +02:00
|
|
|
void initialize() async {
|
|
|
|
bool isGMSInstalled = await checkForGMS();
|
|
|
|
|
|
|
|
if (!isGMSInstalled) {
|
|
|
|
_dio = Dio(BaseOptions(
|
|
|
|
baseUrl: 'https://api.github.com',
|
|
|
|
));
|
|
|
|
print('GitHub API: Using default engine + $isGMSInstalled');
|
|
|
|
} else {
|
|
|
|
_dio = Dio(BaseOptions(
|
|
|
|
baseUrl: 'https://api.github.com',
|
|
|
|
))
|
|
|
|
..httpClientAdapter = NativeAdapter();
|
|
|
|
print('ReVanced API: Using CronetEngine + $isGMSInstalled');
|
|
|
|
}
|
2022-09-11 03:01:06 +02:00
|
|
|
_dio.interceptors.add(_dioCacheManager.interceptor);
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> clearAllCache() async {
|
|
|
|
await _dioCacheManager.clearAll();
|
2022-08-18 18:32:58 +02:00
|
|
|
}
|
|
|
|
|
2022-09-11 03:01:06 +02:00
|
|
|
Future<Map<String, dynamic>?> _getLatestRelease(String repoName) async {
|
2022-08-09 02:16:33 +02:00
|
|
|
try {
|
2022-09-11 03:01:06 +02:00
|
|
|
var response = await _dio.get(
|
2022-09-19 01:28:26 +02:00
|
|
|
'/repos/$repoName/releases/latest',
|
2022-09-12 02:41:53 +02:00
|
|
|
options: _cacheOptions,
|
2022-08-09 02:16:33 +02:00
|
|
|
);
|
2022-09-11 03:01:06 +02:00
|
|
|
return response.data;
|
2022-08-09 02:16:33 +02:00
|
|
|
} on Exception {
|
2022-09-12 10:18:03 +02:00
|
|
|
return null;
|
2022-08-09 02:16:33 +02:00
|
|
|
}
|
2022-07-31 21:46:27 +02:00
|
|
|
}
|
2022-08-01 20:06:27 +02:00
|
|
|
|
2022-09-11 03:01:06 +02:00
|
|
|
Future<List<String>> getCommits(
|
|
|
|
String packageName,
|
|
|
|
String repoName,
|
|
|
|
DateTime since,
|
|
|
|
) async {
|
|
|
|
String path =
|
|
|
|
'src/main/kotlin/app/revanced/patches/${repoAppPath[packageName]}';
|
2022-08-09 02:16:33 +02:00
|
|
|
try {
|
2022-09-11 03:01:06 +02:00
|
|
|
var response = await _dio.get(
|
2022-09-19 01:28:26 +02:00
|
|
|
'/repos/$repoName/commits',
|
2022-09-11 03:01:06 +02:00
|
|
|
queryParameters: {
|
|
|
|
'path': path,
|
|
|
|
'since': since.toIso8601String(),
|
|
|
|
},
|
2022-09-12 02:41:53 +02:00
|
|
|
options: _cacheOptions,
|
2022-08-09 02:16:33 +02:00
|
|
|
);
|
2022-09-11 03:01:06 +02:00
|
|
|
List<dynamic> commits = response.data;
|
|
|
|
return commits
|
2022-10-08 18:36:45 +02:00
|
|
|
.map(
|
|
|
|
(commit) => (commit['commit']['message']).split('\n')[0] +
|
|
|
|
' - ' +
|
|
|
|
commit['commit']['author']['name'] +
|
|
|
|
'\n' as String,
|
|
|
|
)
|
2022-09-11 03:01:06 +02:00
|
|
|
.toList();
|
2022-10-08 18:36:45 +02:00
|
|
|
} catch (e) {
|
2022-09-12 10:18:03 +02:00
|
|
|
return List.empty();
|
2022-08-09 02:16:33 +02:00
|
|
|
}
|
2022-08-01 20:12:38 +02:00
|
|
|
}
|
2022-08-12 20:07:16 +02:00
|
|
|
|
2022-09-11 03:01:06 +02:00
|
|
|
Future<File?> getLatestReleaseFile(String extension, String repoName) async {
|
|
|
|
try {
|
|
|
|
Map<String, dynamic>? release = await _getLatestRelease(repoName);
|
|
|
|
if (release != null) {
|
|
|
|
Map<String, dynamic>? asset =
|
|
|
|
(release['assets'] as List<dynamic>).firstWhereOrNull(
|
|
|
|
(asset) => (asset['name'] as String).endsWith(extension),
|
|
|
|
);
|
|
|
|
if (asset != null) {
|
|
|
|
return await DefaultCacheManager().getSingleFile(
|
|
|
|
asset['browser_download_url'],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} on Exception {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return null;
|
2022-08-26 03:01:53 +02:00
|
|
|
}
|
|
|
|
|
2022-09-11 03:01:06 +02:00
|
|
|
Future<List<Patch>> getPatches(String repoName) async {
|
|
|
|
List<Patch> patches = [];
|
|
|
|
try {
|
|
|
|
File? f = await getLatestReleaseFile('.json', repoName);
|
|
|
|
if (f != null) {
|
|
|
|
List<dynamic> list = jsonDecode(f.readAsStringSync());
|
|
|
|
patches = list.map((patch) => Patch.fromJson(patch)).toList();
|
|
|
|
}
|
|
|
|
} on Exception {
|
2022-09-12 10:18:03 +02:00
|
|
|
return List.empty();
|
2022-09-11 03:01:06 +02:00
|
|
|
}
|
|
|
|
return patches;
|
2022-08-12 20:07:16 +02:00
|
|
|
}
|
2022-07-31 21:46:27 +02:00
|
|
|
}
|