fix(twitter): correctly resolve to integrations methods
This commit is contained in:
parent
2f799b7b73
commit
c655416a91
|
@ -5,5 +5,9 @@ import org.jf.dexlib2.Opcode
|
||||||
|
|
||||||
object JsonHookPatchFingerprint : MethodFingerprint(
|
object JsonHookPatchFingerprint : MethodFingerprint(
|
||||||
customFingerprint = { methodDef, _ -> methodDef.name == "<clinit>" },
|
customFingerprint = { methodDef, _ -> methodDef.name == "<clinit>" },
|
||||||
opcodes = listOf(Opcode.IGET_OBJECT)
|
opcodes = listOf(
|
||||||
|
Opcode.INVOKE_INTERFACE, // Add dummy hook to hooks list.
|
||||||
|
// Add hooks to the hooks list.
|
||||||
|
Opcode.INVOKE_STATIC // Call buildList.
|
||||||
|
)
|
||||||
)
|
)
|
|
@ -5,13 +5,18 @@ import app.revanced.patcher.annotation.Name
|
||||||
import app.revanced.patcher.annotation.Version
|
import app.revanced.patcher.annotation.Version
|
||||||
import app.revanced.patcher.data.BytecodeContext
|
import app.revanced.patcher.data.BytecodeContext
|
||||||
import app.revanced.patcher.extensions.addInstructions
|
import app.revanced.patcher.extensions.addInstructions
|
||||||
|
import app.revanced.patcher.extensions.removeInstructions
|
||||||
|
import app.revanced.patcher.fingerprint.method.impl.MethodFingerprint
|
||||||
import app.revanced.patcher.fingerprint.method.impl.MethodFingerprint.Companion.resolve
|
import app.revanced.patcher.fingerprint.method.impl.MethodFingerprint.Companion.resolve
|
||||||
import app.revanced.patcher.fingerprint.method.impl.MethodFingerprintResult
|
import app.revanced.patcher.patch.BytecodePatch
|
||||||
import app.revanced.patcher.patch.*
|
import app.revanced.patcher.patch.PatchResult
|
||||||
|
import app.revanced.patcher.patch.PatchResultError
|
||||||
|
import app.revanced.patcher.patch.PatchResultSuccess
|
||||||
import app.revanced.patcher.patch.annotations.RequiresIntegrations
|
import app.revanced.patcher.patch.annotations.RequiresIntegrations
|
||||||
import app.revanced.patches.twitter.misc.hook.json.fingerprints.JsonHookPatchFingerprint
|
import app.revanced.patches.twitter.misc.hook.json.fingerprints.JsonHookPatchFingerprint
|
||||||
import app.revanced.patches.twitter.misc.hook.json.fingerprints.JsonInputStreamFingerprint
|
import app.revanced.patches.twitter.misc.hook.json.fingerprints.JsonInputStreamFingerprint
|
||||||
import app.revanced.patches.twitter.misc.hook.json.fingerprints.LoganSquareFingerprint
|
import app.revanced.patches.twitter.misc.hook.json.fingerprints.LoganSquareFingerprint
|
||||||
|
import java.io.Closeable
|
||||||
import java.io.InvalidClassException
|
import java.io.InvalidClassException
|
||||||
|
|
||||||
@Name("json-hook")
|
@Name("json-hook")
|
||||||
|
@ -20,16 +25,16 @@ import java.io.InvalidClassException
|
||||||
@RequiresIntegrations
|
@RequiresIntegrations
|
||||||
class JsonHookPatch : BytecodePatch(
|
class JsonHookPatch : BytecodePatch(
|
||||||
listOf(LoganSquareFingerprint)
|
listOf(LoganSquareFingerprint)
|
||||||
) {
|
), Closeable {
|
||||||
override fun execute(context: BytecodeContext): PatchResult {
|
override fun execute(context: BytecodeContext): PatchResult {
|
||||||
// Make sure the integrations are present.
|
JsonHookPatchFingerprint.also {
|
||||||
val jsonHookPatch = context.findClass { it.type == JSON_HOOK_PATCH_CLASS_DESCRIPTOR }
|
// Make sure the integrations are present.
|
||||||
?: return PatchResultError("Could not find integrations.")
|
val jsonHookPatch = context.findClass { classDef -> classDef.type == JSON_HOOK_PATCH_CLASS_DESCRIPTOR }
|
||||||
|
?: throw PatchResultError("Could not find integrations.")
|
||||||
|
|
||||||
// Allow patch to inject hooks into the patches integrations.
|
if (!it.resolve(context, jsonHookPatch.immutableClass))
|
||||||
jsonHookPatchFingerprintResult = JsonHookPatchFingerprint.also {
|
throw PatchResultError("Unexpected integrations.")
|
||||||
it.resolve(context, jsonHookPatch.immutableClass)
|
}.let { hooks = JsonHookPatchHook(it) }
|
||||||
}.result ?: return PatchResultError("Unexpected integrations.")
|
|
||||||
|
|
||||||
// Conveniently find the type to hook a method in, via a named field.
|
// Conveniently find the type to hook a method in, via a named field.
|
||||||
val jsonFactory = LoganSquareFingerprint.result
|
val jsonFactory = LoganSquareFingerprint.result
|
||||||
|
@ -64,30 +69,10 @@ class JsonHookPatch : BytecodePatch(
|
||||||
*
|
*
|
||||||
* @param context The [BytecodeContext] of the current patch.
|
* @param context The [BytecodeContext] of the current patch.
|
||||||
* @param descriptor The class descriptor of the hook.
|
* @param descriptor The class descriptor of the hook.
|
||||||
|
* @throws ClassNotFoundException If the class could not be found.
|
||||||
*/
|
*/
|
||||||
internal class Hook(context: BytecodeContext, private val descriptor: String) {
|
internal class Hook(context: BytecodeContext, internal val descriptor: String) {
|
||||||
private var added = false
|
internal var added = false
|
||||||
|
|
||||||
/**
|
|
||||||
* Add the hook.
|
|
||||||
*/
|
|
||||||
internal fun add() {
|
|
||||||
if (added) return
|
|
||||||
|
|
||||||
jsonHookPatchFingerprintResult.apply {
|
|
||||||
mutableMethod.apply {
|
|
||||||
addInstructions(
|
|
||||||
scanResult.patternScanResult!!.startIndex,
|
|
||||||
"""
|
|
||||||
sget-object v1, $descriptor->INSTANCE:$descriptor
|
|
||||||
invoke-virtual {v0, v1}, Lkotlin/collections/builders/ListBuilder;->add(Ljava/lang/Object;)Z
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
added = true
|
|
||||||
}
|
|
||||||
|
|
||||||
init {
|
init {
|
||||||
context.findClass { it.type == descriptor }?.let {
|
context.findClass { it.type == descriptor }?.let {
|
||||||
|
@ -102,15 +87,67 @@ class JsonHookPatch : BytecodePatch(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private companion object {
|
/**
|
||||||
const val JSON_HOOK_CLASS_NAMESPACE = "app/revanced/twitter/patches/hook/json"
|
* A hook for the [JsonHookPatch].
|
||||||
|
*
|
||||||
|
* @param jsonHookPatchFingerprint The [JsonHookPatchFingerprint] to hook.
|
||||||
|
*/
|
||||||
|
internal class JsonHookPatchHook(jsonHookPatchFingerprint: MethodFingerprint): Closeable {
|
||||||
|
private val jsonHookPatchFingerprintResult = jsonHookPatchFingerprint.result!!
|
||||||
|
private val jsonHookPatchIndex = jsonHookPatchFingerprintResult.scanResult.patternScanResult!!.endIndex
|
||||||
|
|
||||||
const val JSON_HOOK_PATCH_CLASS_DESCRIPTOR = "L$JSON_HOOK_CLASS_NAMESPACE/JsonHookPatch;"
|
/**
|
||||||
|
* Add a hook to the [JsonHookPatch].
|
||||||
|
* Will not add the hook if it's already added.
|
||||||
|
*
|
||||||
|
* @param hook The [Hook] to add.
|
||||||
|
*/
|
||||||
|
fun addHook(hook: Hook) {
|
||||||
|
if (hook.added) return
|
||||||
|
|
||||||
const val BASE_PATCH_CLASS_NAME = "BaseJsonHook"
|
jsonHookPatchFingerprintResult.mutableMethod.apply {
|
||||||
|
// Insert hooks right before calling buildList.
|
||||||
|
val insertIndex = jsonHookPatchIndex
|
||||||
|
|
||||||
const val JSON_HOOK_CLASS_DESCRIPTOR = "L$JSON_HOOK_CLASS_NAMESPACE/$BASE_PATCH_CLASS_NAME;"
|
addInstructions(
|
||||||
|
insertIndex,
|
||||||
|
"""
|
||||||
|
sget-object v1, ${hook.descriptor}->INSTANCE:${hook.descriptor}
|
||||||
|
invoke-interface {v0, v1}, Ljava/util/List;->add(Ljava/lang/Object;)Z
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
private lateinit var jsonHookPatchFingerprintResult: MethodFingerprintResult
|
hook.added = true
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun close() {
|
||||||
|
// Remove hooks.add(dummyHook).
|
||||||
|
jsonHookPatchFingerprintResult.mutableMethod.apply {
|
||||||
|
val addDummyHookIndex = jsonHookPatchIndex - 2
|
||||||
|
|
||||||
|
removeInstructions(addDummyHookIndex, 2)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override fun close() = hooks.close()
|
||||||
|
|
||||||
|
internal companion object {
|
||||||
|
private const val JSON_HOOK_CLASS_NAMESPACE = "app/revanced/twitter/patches/hook/json"
|
||||||
|
|
||||||
|
private const val JSON_HOOK_PATCH_CLASS_DESCRIPTOR = "L$JSON_HOOK_CLASS_NAMESPACE/JsonHookPatch;"
|
||||||
|
|
||||||
|
private const val BASE_PATCH_CLASS_NAME = "BaseJsonHook"
|
||||||
|
|
||||||
|
private const val JSON_HOOK_CLASS_DESCRIPTOR = "L$JSON_HOOK_CLASS_NAMESPACE/$BASE_PATCH_CLASS_NAME;"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The [JsonHookPatchHook] of the [JsonHookPatch].
|
||||||
|
*
|
||||||
|
* @see JsonHookPatchHook
|
||||||
|
*/
|
||||||
|
internal lateinit var hooks: JsonHookPatchHook
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -4,13 +4,13 @@ import app.revanced.patcher.data.BytecodeContext
|
||||||
import app.revanced.patcher.patch.BytecodePatch
|
import app.revanced.patcher.patch.BytecodePatch
|
||||||
import app.revanced.patcher.patch.PatchResultError
|
import app.revanced.patcher.patch.PatchResultError
|
||||||
import app.revanced.patcher.patch.PatchResultSuccess
|
import app.revanced.patcher.patch.PatchResultSuccess
|
||||||
import app.revanced.patcher.patch.annotations.DependsOn
|
|
||||||
import app.revanced.patches.twitter.misc.hook.json.patch.JsonHookPatch
|
import app.revanced.patches.twitter.misc.hook.json.patch.JsonHookPatch
|
||||||
|
|
||||||
@DependsOn([JsonHookPatch::class])
|
|
||||||
abstract class BaseHookPatchPatch(private val hookClassDescriptor: String) : BytecodePatch() {
|
abstract class BaseHookPatchPatch(private val hookClassDescriptor: String) : BytecodePatch() {
|
||||||
override fun execute(context: BytecodeContext) = try {
|
override fun execute(context: BytecodeContext) = try {
|
||||||
PatchResultSuccess().also { JsonHookPatch.Hook(context, hookClassDescriptor).add() }
|
JsonHookPatch.hooks.addHook(JsonHookPatch.Hook(context, hookClassDescriptor))
|
||||||
|
|
||||||
|
PatchResultSuccess()
|
||||||
} catch (ex: Exception) {
|
} catch (ex: Exception) {
|
||||||
PatchResultError(ex)
|
PatchResultError(ex)
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,8 +15,4 @@ import app.revanced.patches.twitter.misc.hook.patch.ads.annotations.HideAdsCompa
|
||||||
@Description("Hides ads.")
|
@Description("Hides ads.")
|
||||||
@HideAdsCompatibility
|
@HideAdsCompatibility
|
||||||
@Version("0.0.1")
|
@Version("0.0.1")
|
||||||
class HideAdsPatch : BaseHookPatchPatch(HOOK_CLASS_DESCRIPTOR) {
|
class HideAdsPatch : BaseHookPatchPatch("Lapp/revanced/twitter/patches/hook/patch/ads/AdsHook;")
|
||||||
private companion object {
|
|
||||||
const val HOOK_CLASS_DESCRIPTOR = "Lapp/revanced/twitter/patches/hook/patch/ads/AdsHook;"
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -15,9 +15,6 @@ import app.revanced.patches.twitter.misc.hook.patch.recommendation.annotations.H
|
||||||
@Description("Hides recommended users.")
|
@Description("Hides recommended users.")
|
||||||
@HideRecommendedUsersCompatibility
|
@HideRecommendedUsersCompatibility
|
||||||
@Version("0.0.1")
|
@Version("0.0.1")
|
||||||
class HideRecommendedUsersPatch : BaseHookPatchPatch(HOOK_CLASS_DESCRIPTOR) {
|
class HideRecommendedUsersPatch : BaseHookPatchPatch(
|
||||||
private companion object {
|
"Lapp/revanced/twitter/patches/hook/patch/recommendation/RecommendedUsersHook;"
|
||||||
const val HOOK_CLASS_DESCRIPTOR =
|
)
|
||||||
"Lapp/revanced/twitter/patches/hook/patch/recommendation/RecommendedUsersHook;"
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in a new issue