2022-08-29 18:44:45 +02:00
|
|
|
import 'dart:convert';
|
2022-08-06 15:04:18 +02:00
|
|
|
import 'dart:io';
|
2022-08-13 11:56:30 +02:00
|
|
|
import 'package:app_installer/app_installer.dart';
|
|
|
|
import 'package:device_apps/device_apps.dart';
|
2022-08-06 15:04:18 +02:00
|
|
|
import 'package:flutter/services.dart';
|
2022-08-09 01:01:06 +02:00
|
|
|
import 'package:injectable/injectable.dart';
|
2022-08-13 11:56:30 +02:00
|
|
|
import 'package:path_provider/path_provider.dart';
|
2022-08-25 01:51:47 +02:00
|
|
|
import 'package:revanced_manager/app/app.locator.dart';
|
2022-08-06 23:35:35 +02:00
|
|
|
import 'package:revanced_manager/models/patch.dart';
|
2022-08-14 20:40:34 +02:00
|
|
|
import 'package:revanced_manager/models/patched_application.dart';
|
2022-08-18 16:33:33 +02:00
|
|
|
import 'package:revanced_manager/services/manager_api.dart';
|
2022-08-14 20:40:34 +02:00
|
|
|
import 'package:revanced_manager/services/root_api.dart';
|
2022-08-13 11:56:30 +02:00
|
|
|
import 'package:share_extend/share_extend.dart';
|
2022-08-06 15:04:18 +02:00
|
|
|
|
2022-08-09 01:01:06 +02:00
|
|
|
@lazySingleton
|
2022-08-09 02:20:50 +02:00
|
|
|
class PatcherAPI {
|
2022-09-05 04:32:36 +02:00
|
|
|
static const patcherChannel = MethodChannel('app.revanced.manager/patcher');
|
2022-08-25 01:51:47 +02:00
|
|
|
final ManagerAPI _managerAPI = locator<ManagerAPI>();
|
2022-08-18 16:33:33 +02:00
|
|
|
final RootAPI _rootAPI = RootAPI();
|
2022-08-30 03:18:20 +02:00
|
|
|
late Directory _tmpDir;
|
2022-09-07 19:25:12 +02:00
|
|
|
late File _keyStoreFile;
|
2022-08-29 18:44:45 +02:00
|
|
|
List<Patch> _patches = [];
|
2022-08-13 11:56:30 +02:00
|
|
|
File? _outFile;
|
2022-08-09 01:01:06 +02:00
|
|
|
|
2022-08-29 18:44:45 +02:00
|
|
|
Future<void> initialize() async {
|
|
|
|
await _loadPatches();
|
2022-08-30 03:18:20 +02:00
|
|
|
Directory appCache = await getTemporaryDirectory();
|
|
|
|
_tmpDir = Directory('${appCache.path}/patcher');
|
2022-09-07 19:25:12 +02:00
|
|
|
_keyStoreFile = File('${appCache.path}/revanced-manager.keystore');
|
2022-08-30 03:18:20 +02:00
|
|
|
cleanPatcher();
|
|
|
|
}
|
|
|
|
|
|
|
|
void cleanPatcher() {
|
|
|
|
if (_tmpDir.existsSync()) {
|
|
|
|
_tmpDir.deleteSync(recursive: true);
|
|
|
|
}
|
2022-08-19 20:13:43 +02:00
|
|
|
}
|
|
|
|
|
2022-08-29 18:44:45 +02:00
|
|
|
Future<void> _loadPatches() async {
|
|
|
|
try {
|
|
|
|
if (_patches.isEmpty) {
|
|
|
|
File? patchJsonFile = await _managerAPI.downloadPatches('.json');
|
|
|
|
if (patchJsonFile != null) {
|
|
|
|
List<dynamic> list = json.decode(patchJsonFile.readAsStringSync());
|
|
|
|
_patches = list.map((patch) => Patch.fromJson(patch)).toList();
|
2022-08-22 02:31:27 +02:00
|
|
|
}
|
2022-08-06 15:04:18 +02:00
|
|
|
}
|
2022-08-29 18:44:45 +02:00
|
|
|
} on Exception {
|
|
|
|
_patches = List.empty();
|
2022-08-06 15:04:18 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-13 11:56:30 +02:00
|
|
|
Future<List<ApplicationWithIcon>> getFilteredInstalledApps() async {
|
2022-08-29 18:44:45 +02:00
|
|
|
List<ApplicationWithIcon> filteredApps = [];
|
|
|
|
await _loadPatches();
|
|
|
|
for (Patch patch in _patches) {
|
|
|
|
for (Package package in patch.compatiblePackages) {
|
2022-08-22 02:31:27 +02:00
|
|
|
try {
|
2022-08-29 18:44:45 +02:00
|
|
|
if (!filteredApps.any((app) => app.packageName == package.name)) {
|
|
|
|
ApplicationWithIcon? app =
|
|
|
|
await DeviceApps.getApp(package.name, true)
|
|
|
|
as ApplicationWithIcon?;
|
|
|
|
if (app != null) {
|
|
|
|
filteredApps.add(app);
|
2022-08-06 15:04:18 +02:00
|
|
|
}
|
|
|
|
}
|
2022-08-29 18:44:45 +02:00
|
|
|
} catch (e) {
|
|
|
|
continue;
|
2022-08-06 15:04:18 +02:00
|
|
|
}
|
2022-08-17 19:44:27 +02:00
|
|
|
}
|
|
|
|
}
|
2022-08-29 18:44:45 +02:00
|
|
|
return filteredApps;
|
2022-08-17 19:44:27 +02:00
|
|
|
}
|
|
|
|
|
2022-08-29 18:44:45 +02:00
|
|
|
Future<List<Patch>> getFilteredPatches(String packageName) async {
|
|
|
|
await _loadPatches();
|
|
|
|
return _patches
|
|
|
|
.where((patch) =>
|
|
|
|
!patch.name.contains('settings') &&
|
|
|
|
patch.compatiblePackages.any((pack) => pack.name == packageName))
|
|
|
|
.toList();
|
2022-08-13 11:56:30 +02:00
|
|
|
}
|
|
|
|
|
2022-08-29 18:44:45 +02:00
|
|
|
Future<List<Patch>> getAppliedPatches(List<String> appliedPatches) async {
|
|
|
|
await _loadPatches();
|
|
|
|
return _patches
|
|
|
|
.where((patch) => appliedPatches.contains(patch.name))
|
|
|
|
.toList();
|
2022-08-13 11:56:30 +02:00
|
|
|
}
|
|
|
|
|
2022-08-17 13:48:03 +02:00
|
|
|
Future<void> runPatcher(
|
2022-08-29 18:44:45 +02:00
|
|
|
String packageName,
|
2022-08-17 13:48:03 +02:00
|
|
|
String originalFilePath,
|
|
|
|
List<Patch> selectedPatches,
|
|
|
|
) async {
|
2022-08-29 18:44:45 +02:00
|
|
|
bool mergeIntegrations = selectedPatches.any(
|
|
|
|
(patch) => patch.dependencies.contains('integrations'),
|
|
|
|
);
|
|
|
|
bool resourcePatching = selectedPatches.any(
|
|
|
|
(patch) => patch.dependencies.any((dep) => dep.contains('resource-')),
|
2022-08-17 13:48:03 +02:00
|
|
|
);
|
2022-08-29 18:44:45 +02:00
|
|
|
bool includeSettings = selectedPatches.any(
|
|
|
|
(patch) => patch.dependencies.contains('settings'),
|
|
|
|
);
|
|
|
|
if (includeSettings) {
|
|
|
|
try {
|
|
|
|
Patch settingsPatch = _patches.firstWhere(
|
|
|
|
(patch) =>
|
|
|
|
patch.name.contains('settings') &&
|
|
|
|
patch.compatiblePackages.any((pack) => pack.name == packageName),
|
|
|
|
);
|
|
|
|
selectedPatches.add(settingsPatch);
|
|
|
|
} catch (e) {
|
|
|
|
// ignore
|
|
|
|
}
|
|
|
|
}
|
|
|
|
File? patchBundleFile = await _managerAPI.downloadPatches('.jar');
|
|
|
|
File? integrationsFile;
|
|
|
|
if (mergeIntegrations) {
|
|
|
|
integrationsFile = await _managerAPI.downloadIntegrations('.apk');
|
|
|
|
}
|
|
|
|
if (patchBundleFile != null) {
|
2022-08-30 03:18:20 +02:00
|
|
|
_tmpDir.createSync();
|
|
|
|
Directory workDir = _tmpDir.createTempSync('tmp-');
|
2022-08-29 18:44:45 +02:00
|
|
|
File inputFile = File('${workDir.path}/base.apk');
|
|
|
|
File patchedFile = File('${workDir.path}/patched.apk');
|
|
|
|
_outFile = File('${workDir.path}/out.apk');
|
|
|
|
Directory cacheDir = Directory('${workDir.path}/cache');
|
|
|
|
cacheDir.createSync();
|
|
|
|
await patcherChannel.invokeMethod(
|
|
|
|
'runPatcher',
|
|
|
|
{
|
|
|
|
'patchBundleFilePath': patchBundleFile.path,
|
|
|
|
'originalFilePath': originalFilePath,
|
|
|
|
'inputFilePath': inputFile.path,
|
|
|
|
'patchedFilePath': patchedFile.path,
|
|
|
|
'outFilePath': _outFile!.path,
|
|
|
|
'integrationsPath': mergeIntegrations ? integrationsFile!.path : '',
|
|
|
|
'selectedPatches': selectedPatches.map((p) => p.name).toList(),
|
|
|
|
'cacheDirPath': cacheDir.path,
|
|
|
|
'mergeIntegrations': mergeIntegrations,
|
|
|
|
'resourcePatching': resourcePatching,
|
2022-09-07 19:25:12 +02:00
|
|
|
'keyStoreFilePath': _keyStoreFile.path,
|
2022-08-29 18:44:45 +02:00
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
2022-08-13 11:56:30 +02:00
|
|
|
}
|
|
|
|
|
2022-08-14 20:40:34 +02:00
|
|
|
Future<bool> installPatchedFile(PatchedApplication patchedApp) async {
|
2022-08-13 11:56:30 +02:00
|
|
|
if (_outFile != null) {
|
|
|
|
try {
|
2022-09-05 14:43:13 +02:00
|
|
|
if (patchedApp.isRooted) {
|
2022-09-06 15:40:49 +02:00
|
|
|
bool hasRootPermissions = await _rootAPI.hasRootPermissions();
|
|
|
|
if (hasRootPermissions) {
|
|
|
|
return _rootAPI.installApp(
|
|
|
|
patchedApp.packageName,
|
|
|
|
patchedApp.apkFilePath,
|
|
|
|
_outFile!.path,
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
2022-08-13 11:56:30 +02:00
|
|
|
} else {
|
|
|
|
await AppInstaller.installApk(_outFile!.path);
|
2022-08-17 18:07:00 +02:00
|
|
|
return await DeviceApps.isAppInstalled(patchedApp.packageName);
|
2022-08-13 11:56:30 +02:00
|
|
|
}
|
|
|
|
} on Exception {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-09-06 15:40:49 +02:00
|
|
|
void sharePatchedFile(String appName, String version) {
|
2022-08-13 11:56:30 +02:00
|
|
|
if (_outFile != null) {
|
2022-08-14 04:07:28 +02:00
|
|
|
String prefix = appName.toLowerCase().replaceAll(' ', '-');
|
2022-09-06 15:40:49 +02:00
|
|
|
String newName = '$prefix-revanced_v$version.apk';
|
|
|
|
int lastSeparator = _outFile!.path.lastIndexOf('/');
|
|
|
|
File share = _outFile!.renameSync(
|
|
|
|
_outFile!.path.substring(0, lastSeparator + 1) + newName,
|
|
|
|
);
|
2022-08-14 20:40:34 +02:00
|
|
|
ShareExtend.share(share.path, 'file');
|
2022-08-13 11:56:30 +02:00
|
|
|
}
|
2022-08-06 15:04:18 +02:00
|
|
|
}
|
2022-08-15 04:31:36 +02:00
|
|
|
|
|
|
|
Future<bool> checkOldPatch(PatchedApplication patchedApp) async {
|
|
|
|
if (patchedApp.isRooted) {
|
2022-08-25 01:51:47 +02:00
|
|
|
return await _rootAPI.isAppInstalled(patchedApp.packageName);
|
2022-08-15 04:31:36 +02:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> deleteOldPatch(PatchedApplication patchedApp) async {
|
|
|
|
if (patchedApp.isRooted) {
|
2022-08-18 16:33:33 +02:00
|
|
|
await _rootAPI.deleteApp(patchedApp.packageName, patchedApp.apkFilePath);
|
2022-08-15 04:31:36 +02:00
|
|
|
}
|
|
|
|
}
|
2022-09-06 15:40:49 +02:00
|
|
|
|
|
|
|
void shareLog(String logs) {
|
|
|
|
ShareExtend.share(logs, 'text');
|
|
|
|
}
|
2022-08-06 15:04:18 +02:00
|
|
|
}
|