2023-01-30 18:05:06 +05:30
|
|
|
import 'dart:developer';
|
2022-09-11 02:01:06 +01:00
|
|
|
import 'dart:io';
|
2023-01-30 18:05:06 +05:30
|
|
|
|
2022-09-11 02:01:06 +01:00
|
|
|
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';
|
2023-01-30 18:05:06 +05:30
|
|
|
import 'package:native_dio_client/native_dio_client.dart';
|
2022-09-11 02:01:06 +01:00
|
|
|
import 'package:revanced_manager/models/patch.dart';
|
2022-10-11 20:19:50 +05:30
|
|
|
import 'package:revanced_manager/utils/check_for_gms.dart';
|
2022-10-14 23:35:33 +05:30
|
|
|
import 'package:sentry_dio/sentry_dio.dart';
|
2023-01-30 18:05:06 +05:30
|
|
|
import 'package:sentry_flutter/sentry_flutter.dart';
|
|
|
|
import 'package:timeago/timeago.dart';
|
2022-09-11 02:01:06 +01:00
|
|
|
|
|
|
|
@lazySingleton
|
|
|
|
class RevancedAPI {
|
2022-09-19 00:28:26 +01:00
|
|
|
late Dio _dio = Dio();
|
2022-09-11 02:01:06 +01:00
|
|
|
final DioCacheManager _dioCacheManager = DioCacheManager(CacheConfig());
|
|
|
|
final Options _cacheOptions = buildCacheOptions(
|
2022-09-28 22:26:54 +05:30
|
|
|
const Duration(hours: 6),
|
|
|
|
maxStale: const Duration(days: 1),
|
2022-09-11 02:01:06 +01:00
|
|
|
);
|
|
|
|
|
2022-09-19 00:28:26 +01:00
|
|
|
Future<void> initialize(String apiUrl) async {
|
2022-10-14 23:35:33 +05:30
|
|
|
try {
|
2023-01-30 18:05:06 +05:30
|
|
|
final bool isGMSInstalled = await checkForGMS();
|
2022-10-11 20:19:50 +05:30
|
|
|
|
2022-10-14 23:35:33 +05:30
|
|
|
if (!isGMSInstalled) {
|
2023-01-30 18:05:06 +05:30
|
|
|
_dio = Dio(
|
|
|
|
BaseOptions(
|
|
|
|
baseUrl: apiUrl,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
log('ReVanced API: Using default engine + $isGMSInstalled');
|
2022-10-14 23:35:33 +05:30
|
|
|
} else {
|
2023-01-30 18:05:06 +05:30
|
|
|
_dio = Dio(
|
|
|
|
BaseOptions(
|
|
|
|
baseUrl: apiUrl,
|
|
|
|
),
|
|
|
|
)..httpClientAdapter = NativeAdapter();
|
|
|
|
log('ReVanced API: Using CronetEngine + $isGMSInstalled');
|
2022-10-14 23:35:33 +05:30
|
|
|
}
|
|
|
|
_dio.interceptors.add(_dioCacheManager.interceptor);
|
|
|
|
_dio.addSentry(
|
|
|
|
captureFailedRequests: true,
|
|
|
|
);
|
|
|
|
} on Exception catch (e, s) {
|
2023-02-21 18:06:56 +05:45
|
|
|
Sentry.captureException(e, stackTrace: s).ignore();
|
2022-10-11 20:19:50 +05:30
|
|
|
}
|
2022-09-11 02:01:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> clearAllCache() async {
|
2022-10-14 23:35:33 +05:30
|
|
|
try {
|
|
|
|
await _dioCacheManager.clearAll();
|
|
|
|
} on Exception catch (e, s) {
|
2023-02-21 18:06:56 +05:45
|
|
|
Sentry.captureException(e, stackTrace: s).ignore();
|
2022-10-14 23:35:33 +05:30
|
|
|
}
|
2022-09-11 02:01:06 +01:00
|
|
|
}
|
|
|
|
|
2022-09-19 00:28:26 +01:00
|
|
|
Future<Map<String, List<dynamic>>> getContributors() async {
|
2023-01-30 18:05:06 +05:30
|
|
|
final Map<String, List<dynamic>> contributors = {};
|
2022-09-11 02:01:06 +01:00
|
|
|
try {
|
2023-01-30 18:05:06 +05:30
|
|
|
final response = await _dio.get('/contributors', options: _cacheOptions);
|
|
|
|
final List<dynamic> repositories = response.data['repositories'];
|
|
|
|
for (final Map<String, dynamic> repo in repositories) {
|
|
|
|
final String name = repo['name'];
|
2022-09-11 02:01:06 +01:00
|
|
|
contributors[name] = repo['contributors'];
|
|
|
|
}
|
2022-10-14 23:35:33 +05:30
|
|
|
} on Exception catch (e, s) {
|
2023-02-21 18:06:56 +05:45
|
|
|
Sentry.captureException(e, stackTrace: s).ignore();
|
2022-09-12 09:18:03 +01:00
|
|
|
return {};
|
2022-09-11 02:01:06 +01:00
|
|
|
}
|
|
|
|
return contributors;
|
|
|
|
}
|
|
|
|
|
2022-09-19 00:28:26 +01:00
|
|
|
Future<List<Patch>> getPatches() async {
|
2022-09-11 02:01:06 +01:00
|
|
|
try {
|
2023-01-30 18:05:06 +05:30
|
|
|
final response = await _dio.get('/patches', options: _cacheOptions);
|
|
|
|
final List<dynamic> patches = response.data;
|
2022-09-11 02:01:06 +01:00
|
|
|
return patches.map((patch) => Patch.fromJson(patch)).toList();
|
2022-10-14 23:35:33 +05:30
|
|
|
} on Exception catch (e, s) {
|
2023-02-21 18:06:56 +05:45
|
|
|
Sentry.captureException(e, stackTrace: s).ignore();
|
2022-09-12 09:18:03 +01:00
|
|
|
return List.empty();
|
2022-09-11 02:01:06 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<Map<String, dynamic>?> _getLatestRelease(
|
|
|
|
String extension,
|
|
|
|
String repoName,
|
|
|
|
) async {
|
|
|
|
try {
|
2023-01-30 18:05:06 +05:30
|
|
|
final response = await _dio.get('/tools', options: _cacheOptions);
|
|
|
|
final List<dynamic> tools = response.data['tools'];
|
2022-09-11 02:01:06 +01:00
|
|
|
return tools.firstWhereOrNull(
|
|
|
|
(t) =>
|
|
|
|
t['repository'] == repoName &&
|
|
|
|
(t['name'] as String).endsWith(extension),
|
|
|
|
);
|
2022-10-14 23:35:33 +05:30
|
|
|
} on Exception catch (e, s) {
|
2023-02-21 18:06:56 +05:45
|
|
|
Sentry.captureException(e, stackTrace: s).ignore();
|
2022-09-11 02:01:06 +01:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<String?> getLatestReleaseVersion(
|
2022-09-18 23:12:30 +05:30
|
|
|
String extension,
|
|
|
|
String repoName,
|
|
|
|
) async {
|
2022-09-11 02:01:06 +01:00
|
|
|
try {
|
2023-01-30 18:05:06 +05:30
|
|
|
final Map<String, dynamic>? release = await _getLatestRelease(
|
2022-09-19 00:28:26 +01:00
|
|
|
extension,
|
|
|
|
repoName,
|
|
|
|
);
|
2022-09-11 02:01:06 +01:00
|
|
|
if (release != null) {
|
|
|
|
return release['version'];
|
|
|
|
}
|
2022-10-14 23:35:33 +05:30
|
|
|
} on Exception catch (e, s) {
|
2023-02-21 18:06:56 +05:45
|
|
|
Sentry.captureException(e, stackTrace: s).ignore();
|
2022-09-11 02:01:06 +01:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2022-09-19 00:28:26 +01:00
|
|
|
Future<File?> getLatestReleaseFile(String extension, String repoName) async {
|
2022-09-11 02:01:06 +01:00
|
|
|
try {
|
2023-01-30 18:05:06 +05:30
|
|
|
final Map<String, dynamic>? release = await _getLatestRelease(
|
2022-09-11 02:01:06 +01:00
|
|
|
extension,
|
|
|
|
repoName,
|
|
|
|
);
|
|
|
|
if (release != null) {
|
2023-01-30 18:05:06 +05:30
|
|
|
final String url = release['browser_download_url'];
|
2022-09-11 02:01:06 +01:00
|
|
|
return await DefaultCacheManager().getSingleFile(url);
|
|
|
|
}
|
2022-10-14 23:35:33 +05:30
|
|
|
} on Exception catch (e, s) {
|
2023-02-21 18:06:56 +05:45
|
|
|
Sentry.captureException(e, stackTrace: s).ignore();
|
2022-09-11 02:01:06 +01:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<String?> getLatestReleaseTime(
|
|
|
|
String extension,
|
|
|
|
String repoName,
|
|
|
|
) async {
|
|
|
|
try {
|
2023-01-30 18:05:06 +05:30
|
|
|
final Map<String, dynamic>? release = await _getLatestRelease(
|
2022-09-11 02:01:06 +01:00
|
|
|
extension,
|
|
|
|
repoName,
|
|
|
|
);
|
|
|
|
if (release != null) {
|
2023-01-30 18:05:06 +05:30
|
|
|
final DateTime timestamp =
|
|
|
|
DateTime.parse(release['timestamp'] as String);
|
2022-09-11 02:01:06 +01:00
|
|
|
return format(timestamp, locale: 'en_short');
|
|
|
|
}
|
2022-10-14 23:35:33 +05:30
|
|
|
} on Exception catch (e, s) {
|
2023-02-21 18:06:56 +05:45
|
|
|
Sentry.captureException(e, stackTrace: s).ignore();
|
2022-09-11 02:01:06 +01:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|