mirror of
https://github.com/topjohnwu/Magisk.git
synced 2025-01-12 14:33:37 +00:00
Provide upgrade path for stubs
This commit is contained in:
parent
d459859361
commit
676e9c6593
@ -7,20 +7,17 @@ import android.content.Intent
|
|||||||
import android.os.Build
|
import android.os.Build
|
||||||
import android.webkit.MimeTypeMap
|
import android.webkit.MimeTypeMap
|
||||||
import androidx.core.app.NotificationCompat
|
import androidx.core.app.NotificationCompat
|
||||||
import com.topjohnwu.magisk.ProcessPhoenix
|
|
||||||
import com.topjohnwu.magisk.R
|
import com.topjohnwu.magisk.R
|
||||||
import com.topjohnwu.magisk.extensions.chooser
|
import com.topjohnwu.magisk.extensions.chooser
|
||||||
import com.topjohnwu.magisk.extensions.exists
|
import com.topjohnwu.magisk.extensions.exists
|
||||||
import com.topjohnwu.magisk.extensions.provide
|
import com.topjohnwu.magisk.extensions.provide
|
||||||
import com.topjohnwu.magisk.intent
|
import com.topjohnwu.magisk.intent
|
||||||
import com.topjohnwu.magisk.isRunningAsStub
|
|
||||||
import com.topjohnwu.magisk.model.entity.internal.Configuration.*
|
import com.topjohnwu.magisk.model.entity.internal.Configuration.*
|
||||||
import com.topjohnwu.magisk.model.entity.internal.Configuration.Flash.Secondary
|
import com.topjohnwu.magisk.model.entity.internal.Configuration.Flash.Secondary
|
||||||
import com.topjohnwu.magisk.model.entity.internal.DownloadSubject
|
import com.topjohnwu.magisk.model.entity.internal.DownloadSubject
|
||||||
import com.topjohnwu.magisk.model.entity.internal.DownloadSubject.*
|
import com.topjohnwu.magisk.model.entity.internal.DownloadSubject.*
|
||||||
import com.topjohnwu.magisk.ui.flash.FlashActivity
|
import com.topjohnwu.magisk.ui.flash.FlashActivity
|
||||||
import com.topjohnwu.magisk.utils.APKInstall
|
import com.topjohnwu.magisk.utils.APKInstall
|
||||||
import com.topjohnwu.magisk.utils.DynAPK
|
|
||||||
import org.koin.core.get
|
import org.koin.core.get
|
||||||
import java.io.File
|
import java.io.File
|
||||||
import kotlin.random.Random.Default.nextInt
|
import kotlin.random.Random.Default.nextInt
|
||||||
@ -65,15 +62,7 @@ open class DownloadService : RemoteFileService() {
|
|||||||
) {
|
) {
|
||||||
remove(id)
|
remove(id)
|
||||||
when (subject.configuration) {
|
when (subject.configuration) {
|
||||||
is APK.Upgrade -> {
|
is APK.Upgrade -> APKInstall.install(this, subject.file)
|
||||||
if (isRunningAsStub) {
|
|
||||||
subject.file.copyTo(DynAPK.update(this), overwrite = true)
|
|
||||||
subject.file.delete()
|
|
||||||
ProcessPhoenix.triggerRebirth(this)
|
|
||||||
} else {
|
|
||||||
APKInstall.install(this, subject.file)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
is APK.Restore -> Unit
|
is APK.Restore -> Unit
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,28 +1,48 @@
|
|||||||
package com.topjohnwu.magisk.model.download
|
package com.topjohnwu.magisk.model.download
|
||||||
|
|
||||||
import com.topjohnwu.magisk.BuildConfig
|
import com.topjohnwu.magisk.*
|
||||||
import com.topjohnwu.magisk.Config
|
import com.topjohnwu.magisk.extensions.writeTo
|
||||||
import com.topjohnwu.magisk.R
|
|
||||||
import com.topjohnwu.magisk.isRunningAsStub
|
|
||||||
import com.topjohnwu.magisk.model.entity.internal.Configuration.APK.Restore
|
import com.topjohnwu.magisk.model.entity.internal.Configuration.APK.Restore
|
||||||
import com.topjohnwu.magisk.model.entity.internal.Configuration.APK.Upgrade
|
import com.topjohnwu.magisk.model.entity.internal.Configuration.APK.Upgrade
|
||||||
import com.topjohnwu.magisk.model.entity.internal.DownloadSubject
|
import com.topjohnwu.magisk.model.entity.internal.DownloadSubject
|
||||||
|
import com.topjohnwu.magisk.utils.DynAPK
|
||||||
import com.topjohnwu.magisk.utils.PatchAPK
|
import com.topjohnwu.magisk.utils.PatchAPK
|
||||||
import com.topjohnwu.superuser.Shell
|
import com.topjohnwu.superuser.Shell
|
||||||
import java.io.File
|
import java.io.File
|
||||||
|
|
||||||
private fun RemoteFileService.patchPackage(apk: File, id: Int) {
|
private fun RemoteFileService.patch(apk: File, id: Int) {
|
||||||
if (!isRunningAsStub && packageName != BuildConfig.APPLICATION_ID) {
|
if (packageName == BuildConfig.APPLICATION_ID)
|
||||||
update(id) { notification ->
|
return
|
||||||
notification.setProgress(0, 0, true)
|
|
||||||
.setProgress(0, 0, true)
|
update(id) { notification ->
|
||||||
.setContentTitle(getString(R.string.hide_manager_title))
|
notification.setProgress(0, 0, true)
|
||||||
.setContentText("")
|
.setProgress(0, 0, true)
|
||||||
}
|
.setContentTitle(getString(R.string.hide_manager_title))
|
||||||
val patched = File(apk.parent, "patched.apk")
|
.setContentText("")
|
||||||
PatchAPK.patch(apk, patched, packageName, applicationInfo.nonLocalizedLabel.toString())
|
}
|
||||||
|
val patched = File(apk.parent, "patched.apk")
|
||||||
|
PatchAPK.patch(apk, patched, packageName, applicationInfo.nonLocalizedLabel.toString())
|
||||||
|
apk.delete()
|
||||||
|
patched.renameTo(apk)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun RemoteFileService.upgrade(apk: File, id: Int) {
|
||||||
|
if (isRunningAsStub) {
|
||||||
|
// Move to upgrade location
|
||||||
|
apk.copyTo(DynAPK.update(this), overwrite = true)
|
||||||
apk.delete()
|
apk.delete()
|
||||||
patched.renameTo(apk)
|
if (ClassMap.data!!.version < Info.remote.stub.versionCode) {
|
||||||
|
// We also want to upgrade stub
|
||||||
|
service.fetchFile(Info.remote.stub.link).blockingGet().byteStream().use {
|
||||||
|
it.writeTo(apk)
|
||||||
|
}
|
||||||
|
patch(apk, id)
|
||||||
|
} else {
|
||||||
|
// Simply relaunch the app
|
||||||
|
ProcessPhoenix.triggerRebirth(this)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
patch(apk, id)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -41,6 +61,6 @@ private fun RemoteFileService.restore(apk: File, id: Int) {
|
|||||||
|
|
||||||
fun RemoteFileService.handleAPK(subject: DownloadSubject.Manager)
|
fun RemoteFileService.handleAPK(subject: DownloadSubject.Manager)
|
||||||
= when (subject.configuration) {
|
= when (subject.configuration) {
|
||||||
is Upgrade -> patchPackage(subject.file, subject.hashCode())
|
is Upgrade -> upgrade(subject.file, subject.hashCode())
|
||||||
is Restore -> restore(subject.file, subject.hashCode())
|
is Restore -> restore(subject.file, subject.hashCode())
|
||||||
}
|
}
|
||||||
|
@ -22,7 +22,7 @@ import java.io.InputStream
|
|||||||
|
|
||||||
abstract class RemoteFileService : NotificationService() {
|
abstract class RemoteFileService : NotificationService() {
|
||||||
|
|
||||||
private val service: GithubRawServices by inject()
|
val service: GithubRawServices by inject()
|
||||||
|
|
||||||
override val defaultNotification: NotificationCompat.Builder
|
override val defaultNotification: NotificationCompat.Builder
|
||||||
get() = Notifications.progress(this, "")
|
get() = Notifications.progress(this, "")
|
||||||
|
@ -138,6 +138,11 @@ object PatchAPK {
|
|||||||
|
|
||||||
fun patch(apk: File, out: File, pkg: String, label: String): Boolean {
|
fun patch(apk: File, out: File, pkg: String, label: String): Boolean {
|
||||||
try {
|
try {
|
||||||
|
if (apk.length() < 1 shl 18) {
|
||||||
|
// APK is smaller than 256K, must be stub
|
||||||
|
return patch(apk.path, out.path, pkg, label)
|
||||||
|
}
|
||||||
|
|
||||||
// Try using the new APK to patch itself
|
// Try using the new APK to patch itself
|
||||||
val loader = DynamicClassLoader(apk)
|
val loader = DynamicClassLoader(apk)
|
||||||
val cls = loader.loadClass("a.a")
|
val cls = loader.loadClass("a.a")
|
||||||
@ -152,9 +157,8 @@ object PatchAPK {
|
|||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
Timber.e(e)
|
Timber.e(e)
|
||||||
// Fallback to use the current implementation
|
// Fallback to use the current implementation
|
||||||
patch(apk.path, out.path, pkg, label)
|
return patch(apk.path, out.path, pkg, label)
|
||||||
}
|
}
|
||||||
return false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fun hideManager(context: Context, label: String) {
|
fun hideManager(context: Context, label: String) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user