2023-04-20 04:45:46 +05:30
|
|
|
import 'dart:async';
|
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';
|
2023-04-18 13:27:26 +05:30
|
|
|
import 'package:dio_cache_interceptor/dio_cache_interceptor.dart';
|
2023-03-05 14:42:46 +05:30
|
|
|
import 'package:flutter/foundation.dart';
|
2022-09-11 02:01:06 +01:00
|
|
|
import 'package:flutter_cache_manager/flutter_cache_manager.dart';
|
|
|
|
import 'package:injectable/injectable.dart';
|
2023-09-29 18:39:07 +02:00
|
|
|
import 'package:synchronized/synchronized.dart';
|
2023-01-30 18:05:06 +05:30
|
|
|
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();
|
2023-04-20 04:45:46 +05:30
|
|
|
|
2023-09-29 18:39:07 +02:00
|
|
|
final Lock getToolsLock = Lock();
|
|
|
|
|
2023-04-18 13:27:26 +05:30
|
|
|
final _cacheOptions = CacheOptions(
|
|
|
|
store: MemCacheStore(),
|
2022-09-28 22:26:54 +05:30
|
|
|
maxStale: const Duration(days: 1),
|
2023-04-18 13:27:26 +05:30
|
|
|
priority: CachePriority.high,
|
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-05-06 05:39:46 +05:30
|
|
|
_dio = Dio(
|
|
|
|
BaseOptions(
|
|
|
|
baseUrl: apiUrl,
|
|
|
|
),
|
|
|
|
);
|
2022-10-11 20:19:50 +05:30
|
|
|
|
2023-04-18 13:27:26 +05:30
|
|
|
_dio.interceptors.add(DioCacheInterceptor(options: _cacheOptions));
|
2023-03-05 14:42:46 +05:30
|
|
|
} on Exception catch (e) {
|
|
|
|
if (kDebugMode) {
|
|
|
|
print(e);
|
|
|
|
}
|
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 {
|
2023-04-18 13:27:26 +05:30
|
|
|
await _cacheOptions.store!.clean();
|
2023-03-05 14:42:46 +05:30
|
|
|
} on Exception catch (e) {
|
|
|
|
if (kDebugMode) {
|
|
|
|
print(e);
|
|
|
|
}
|
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-04-18 13:27:26 +05:30
|
|
|
final response = await _dio.get('/contributors');
|
2023-01-30 18:05:06 +05:30
|
|
|
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'];
|
|
|
|
}
|
2023-03-05 14:42:46 +05:30
|
|
|
} on Exception catch (e) {
|
|
|
|
if (kDebugMode) {
|
|
|
|
print(e);
|
|
|
|
}
|
2022-09-12 09:18:03 +01:00
|
|
|
return {};
|
2022-09-11 02:01:06 +01:00
|
|
|
}
|
|
|
|
return contributors;
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<Map<String, dynamic>?> _getLatestRelease(
|
|
|
|
String extension,
|
|
|
|
String repoName,
|
2023-09-29 18:39:07 +02:00
|
|
|
) {
|
|
|
|
return getToolsLock.synchronized(() async {
|
|
|
|
try {
|
|
|
|
final response = await _dio.get('/tools');
|
|
|
|
final List<dynamic> tools = response.data['tools'];
|
|
|
|
return tools.firstWhereOrNull(
|
|
|
|
(t) =>
|
|
|
|
t['repository'] == repoName &&
|
|
|
|
(t['name'] as String).endsWith(extension),
|
|
|
|
);
|
|
|
|
} on Exception catch (e) {
|
|
|
|
if (kDebugMode) {
|
|
|
|
print(e);
|
|
|
|
}
|
|
|
|
return null;
|
2023-03-05 14:42:46 +05:30
|
|
|
}
|
2023-09-29 18:39:07 +02:00
|
|
|
});
|
2022-09-11 02:01:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
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'];
|
|
|
|
}
|
2023-03-05 14:42:46 +05:30
|
|
|
} on Exception catch (e) {
|
|
|
|
if (kDebugMode) {
|
|
|
|
print(e);
|
|
|
|
}
|
2022-09-11 02:01:06 +01:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2023-04-18 19:45:29 +05:30
|
|
|
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);
|
|
|
|
}
|
2023-03-05 14:42:46 +05:30
|
|
|
} on Exception catch (e) {
|
|
|
|
if (kDebugMode) {
|
|
|
|
print(e);
|
|
|
|
}
|
2022-09-11 02:01:06 +01:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2023-07-10 19:36:50 +07:00
|
|
|
StreamController<double> managerUpdateProgress =
|
|
|
|
StreamController<double>.broadcast();
|
2023-04-20 04:45:46 +05:30
|
|
|
|
|
|
|
void updateManagerDownloadProgress(int progress) {
|
|
|
|
managerUpdateProgress.add(progress.toDouble());
|
|
|
|
}
|
|
|
|
|
|
|
|
Stream<double> getManagerUpdateProgress() {
|
|
|
|
return managerUpdateProgress.stream;
|
|
|
|
}
|
|
|
|
|
|
|
|
void disposeManagerUpdateProgress() {
|
|
|
|
managerUpdateProgress.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<File?> downloadManager() async {
|
|
|
|
final Map<String, dynamic>? release = await _getLatestRelease(
|
|
|
|
'.apk',
|
|
|
|
'revanced/revanced-manager',
|
|
|
|
);
|
|
|
|
File? outputFile;
|
|
|
|
await for (final result in DefaultCacheManager().getFileStream(
|
|
|
|
release!['browser_download_url'] as String,
|
|
|
|
withProgress: true,
|
|
|
|
)) {
|
|
|
|
if (result is DownloadProgress) {
|
|
|
|
final totalSize = result.totalSize ?? 10000000;
|
|
|
|
final progress = (result.downloaded / totalSize * 100).round();
|
|
|
|
|
|
|
|
updateManagerDownloadProgress(progress);
|
|
|
|
} else if (result is FileInfo) {
|
2023-06-23 19:21:36 +05:45
|
|
|
disposeManagerUpdateProgress();
|
2023-04-20 04:45:46 +05:30
|
|
|
// The download is complete; convert the FileInfo object to a File object
|
|
|
|
outputFile = File(result.file.path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return outputFile;
|
|
|
|
}
|
|
|
|
|
2022-09-11 02:01:06 +01:00
|
|
|
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');
|
|
|
|
}
|
2023-03-05 14:42:46 +05:30
|
|
|
} on Exception catch (e) {
|
|
|
|
if (kDebugMode) {
|
|
|
|
print(e);
|
|
|
|
}
|
2022-09-11 02:01:06 +01:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|