fix(twitter): make hide-promoted-ads
patch compatible with any version
Signed-off-by: oSumAtrIX <johan.melkonyan1@web.de>
This commit is contained in:
parent
e14893ec89
commit
3dbc5ff272
|
@ -1,18 +0,0 @@
|
||||||
package app.revanced.patches.twitter.ad.timeline.fingerprints
|
|
||||||
|
|
||||||
import app.revanced.patcher.fingerprint.method.impl.MethodFingerprint
|
|
||||||
import org.jf.dexlib2.Opcode
|
|
||||||
|
|
||||||
object TimelineTweetJsonParserFingerprint : MethodFingerprint(
|
|
||||||
opcodes = listOf(
|
|
||||||
Opcode.IPUT_OBJECT,
|
|
||||||
Opcode.GOTO,
|
|
||||||
Opcode.SGET_OBJECT,
|
|
||||||
Opcode.INVOKE_VIRTUAL,
|
|
||||||
Opcode.MOVE_RESULT_OBJECT,
|
|
||||||
Opcode.CHECK_CAST,
|
|
||||||
Opcode.IPUT_OBJECT,
|
|
||||||
Opcode.RETURN_VOID,
|
|
||||||
), strings = listOf("tweetPromotedMetadata", "promotedMetadata", "hasModeratedReplies", "conversationAnnotation"),
|
|
||||||
customFingerprint = { methodDef -> methodDef.name == "parseField" }
|
|
||||||
)
|
|
|
@ -1,77 +0,0 @@
|
||||||
package app.revanced.patches.twitter.ad.timeline.patch
|
|
||||||
|
|
||||||
import app.revanced.patcher.annotation.Description
|
|
||||||
import app.revanced.patcher.annotation.Name
|
|
||||||
import app.revanced.patcher.annotation.Version
|
|
||||||
import app.revanced.patcher.data.BytecodeContext
|
|
||||||
import app.revanced.patcher.extensions.addInstructions
|
|
||||||
import app.revanced.patcher.extensions.instruction
|
|
||||||
import app.revanced.patcher.fingerprint.method.impl.MethodFingerprint
|
|
||||||
import app.revanced.patcher.fingerprint.method.impl.MethodFingerprintResult
|
|
||||||
import app.revanced.patcher.patch.BytecodePatch
|
|
||||||
import app.revanced.patcher.patch.PatchResult
|
|
||||||
import app.revanced.patcher.patch.PatchResultError
|
|
||||||
import app.revanced.patcher.patch.PatchResultSuccess
|
|
||||||
import app.revanced.patcher.patch.annotations.Patch
|
|
||||||
import app.revanced.patcher.util.proxy.mutableTypes.MutableMethod
|
|
||||||
import app.revanced.patches.twitter.ad.timeline.annotations.TimelineAdsCompatibility
|
|
||||||
import app.revanced.patches.twitter.ad.timeline.fingerprints.TimelineTweetJsonParserFingerprint
|
|
||||||
import org.jf.dexlib2.Opcode
|
|
||||||
import org.jf.dexlib2.builder.BuilderInstruction
|
|
||||||
import org.jf.dexlib2.builder.instruction.BuilderInstruction22c
|
|
||||||
import org.jf.dexlib2.iface.instruction.ReferenceInstruction
|
|
||||||
import org.jf.dexlib2.iface.reference.FieldReference
|
|
||||||
import org.jf.dexlib2.iface.reference.StringReference
|
|
||||||
|
|
||||||
@Patch
|
|
||||||
@Name("timeline-ads")
|
|
||||||
@Description("Removes ads from the Twitter timeline. Might require clearing app data to remove already cached ads.")
|
|
||||||
@TimelineAdsCompatibility
|
|
||||||
@Version("0.0.1")
|
|
||||||
class TimelineAdsPatch : BytecodePatch(
|
|
||||||
listOf(TimelineTweetJsonParserFingerprint)
|
|
||||||
) {
|
|
||||||
override fun execute(context: BytecodeContext): PatchResult {
|
|
||||||
if (removePromotedAds())
|
|
||||||
return PatchResultError("The instruction for the tweet id field could not be found")
|
|
||||||
|
|
||||||
return PatchResultSuccess()
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun removePromotedAds(): Boolean {
|
|
||||||
val (parserFingerprintResult, parserMethod, instructions) = TimelineTweetJsonParserFingerprint.unwrap()
|
|
||||||
|
|
||||||
// Anchor index
|
|
||||||
val tweetIdFieldInstructionIndex = instructions.indexOfFirst { instruction ->
|
|
||||||
if (instruction.opcode.ordinal != Opcode.CONST_STRING.ordinal) return@indexOfFirst false
|
|
||||||
if (((instruction as? ReferenceInstruction)?.reference as StringReference).string != "tweetSocialProof") return@indexOfFirst false
|
|
||||||
|
|
||||||
// Use the above conditions as an anchor to find the index for the instruction with the field we need
|
|
||||||
return@indexOfFirst true
|
|
||||||
} - 2 // This is where the instruction with the field is located
|
|
||||||
|
|
||||||
// Reference to the tweetId field for of the timeline tweet
|
|
||||||
val tweetIdFieldReference =
|
|
||||||
(parserMethod.instruction(tweetIdFieldInstructionIndex) as? BuilderInstruction22c)?.reference as? FieldReference
|
|
||||||
?: return true
|
|
||||||
|
|
||||||
// Set the tweetId field to null
|
|
||||||
// This will cause twitter to not show the promoted ads, because we set it to null, when the tweet is promoted
|
|
||||||
parserFingerprintResult.mutableMethod.addInstructions(
|
|
||||||
parserFingerprintResult.scanResult.patternScanResult!!.startIndex + 1,
|
|
||||||
"""
|
|
||||||
const/4 v2, 0x0
|
|
||||||
iput-object v2, p0, Lcom/twitter/model/json/timeline/urt/JsonTimelineTweet;->${tweetIdFieldReference.name}:Ljava/lang/String;
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun MethodFingerprint.unwrap(): Triple<MethodFingerprintResult, MutableMethod, MutableList<BuilderInstruction>> {
|
|
||||||
val parserFingerprintResult = this.result!!
|
|
||||||
val parserMethod = parserFingerprintResult.mutableMethod
|
|
||||||
val instructions = parserMethod.implementation!!.instructions
|
|
||||||
|
|
||||||
return Triple(parserFingerprintResult, parserMethod, instructions)
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -0,0 +1,9 @@
|
||||||
|
package app.revanced.patches.twitter.misc.hook.json.fingerprints
|
||||||
|
|
||||||
|
import app.revanced.patcher.fingerprint.method.impl.MethodFingerprint
|
||||||
|
import org.jf.dexlib2.Opcode
|
||||||
|
|
||||||
|
object JsonHookPatchFingerprint : MethodFingerprint(
|
||||||
|
customFingerprint = { methodDef -> methodDef.name == "<clinit>" },
|
||||||
|
opcodes = listOf(Opcode.IGET_OBJECT)
|
||||||
|
)
|
|
@ -0,0 +1,10 @@
|
||||||
|
package app.revanced.patches.twitter.misc.hook.json.fingerprints
|
||||||
|
|
||||||
|
import app.revanced.patcher.fingerprint.method.impl.MethodFingerprint
|
||||||
|
|
||||||
|
object JsonInputStreamFingerprint : MethodFingerprint(
|
||||||
|
customFingerprint = { methodDef ->
|
||||||
|
if (methodDef.parameterTypes.size == 0) false
|
||||||
|
else methodDef.parameterTypes.first() == "Ljava/io/InputStream;"
|
||||||
|
}
|
||||||
|
)
|
|
@ -0,0 +1,7 @@
|
||||||
|
package app.revanced.patches.twitter.misc.hook.json.fingerprints
|
||||||
|
|
||||||
|
import app.revanced.patcher.fingerprint.method.impl.MethodFingerprint
|
||||||
|
|
||||||
|
object LoganSquareFingerprint : MethodFingerprint(
|
||||||
|
customFingerprint = { methodDef -> methodDef.definingClass.endsWith("LoganSquare;") }
|
||||||
|
)
|
|
@ -0,0 +1,114 @@
|
||||||
|
package app.revanced.patches.twitter.misc.hook.json.patch
|
||||||
|
|
||||||
|
import app.revanced.patcher.annotation.Description
|
||||||
|
import app.revanced.patcher.annotation.Name
|
||||||
|
import app.revanced.patcher.annotation.Version
|
||||||
|
import app.revanced.patcher.data.BytecodeContext
|
||||||
|
import app.revanced.patcher.extensions.addInstructions
|
||||||
|
import app.revanced.patcher.fingerprint.method.impl.MethodFingerprint.Companion.resolve
|
||||||
|
import app.revanced.patcher.fingerprint.method.impl.MethodFingerprintResult
|
||||||
|
import app.revanced.patcher.patch.*
|
||||||
|
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.LoganSquareFingerprint
|
||||||
|
import java.io.InvalidClassException
|
||||||
|
|
||||||
|
@Name("json-hook")
|
||||||
|
@Description("Hooks the stream which reads JSON responses.")
|
||||||
|
@Version("0.0.1")
|
||||||
|
class JsonHookPatch : BytecodePatch(
|
||||||
|
listOf(LoganSquareFingerprint)
|
||||||
|
) {
|
||||||
|
override fun execute(context: BytecodeContext): PatchResult {
|
||||||
|
// Make sure the integrations are present.
|
||||||
|
val jsonHookPatch = context.findClass { it.type == JSON_HOOK_PATCH_CLASS_DESCRIPTOR }
|
||||||
|
?: return PatchResultError("Could not find integrations.")
|
||||||
|
|
||||||
|
// Allow patch to inject hooks into the patches integrations.
|
||||||
|
jsonHookPatchFingerprintResult = JsonHookPatchFingerprint.also {
|
||||||
|
it.resolve(context, jsonHookPatch.immutableClass)
|
||||||
|
}.result ?: return PatchResultError("Unexpected integrations.")
|
||||||
|
|
||||||
|
// Conveniently find the type to hook a method in, via a named field.
|
||||||
|
val jsonFactory = LoganSquareFingerprint.result
|
||||||
|
?.classDef
|
||||||
|
?.fields
|
||||||
|
?.firstOrNull { it.name == "JSON_FACTORY" }
|
||||||
|
?.type
|
||||||
|
.let { type ->
|
||||||
|
context.findClass { it.type == type }?.mutableClass
|
||||||
|
} ?: return PatchResultError("Could not find required class.")
|
||||||
|
|
||||||
|
// Hook the methods first parameter.
|
||||||
|
JsonInputStreamFingerprint
|
||||||
|
.also { it.resolve(context, jsonFactory) }
|
||||||
|
.result
|
||||||
|
?.mutableMethod
|
||||||
|
?.addInstructions(
|
||||||
|
0,
|
||||||
|
"""
|
||||||
|
invoke-static { p1 }, $JSON_HOOK_PATCH_CLASS_DESCRIPTOR->parseJsonHook(Ljava/io/InputStream;)Ljava/io/InputStream;
|
||||||
|
move-result-object p1
|
||||||
|
"""
|
||||||
|
) ?: return PatchResultError("Could not find method to hook.")
|
||||||
|
|
||||||
|
return PatchResultSuccess()
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a hook class.
|
||||||
|
* The class has to extend on **JsonHook**.
|
||||||
|
* The class has to be a Kotlin object class, or at least have an INSTANCE field of itself.
|
||||||
|
*
|
||||||
|
* @param context The [BytecodeContext] of the current patch.
|
||||||
|
* @param descriptor The class descriptor of the hook.
|
||||||
|
*/
|
||||||
|
internal class Hook(context: BytecodeContext, private val descriptor: String) {
|
||||||
|
private 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 {
|
||||||
|
context.findClass { it.type == descriptor }?.let {
|
||||||
|
it.mutableClass.also { classDef ->
|
||||||
|
if (
|
||||||
|
classDef.superclass != JSON_HOOK_CLASS_DESCRIPTOR ||
|
||||||
|
!classDef.fields.any { field -> field.name == "INSTANCE" }
|
||||||
|
) throw InvalidClassException(classDef.type, "Not a hook class")
|
||||||
|
|
||||||
|
}
|
||||||
|
} ?: throw ClassNotFoundException("Failed to find hook class")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private companion object {
|
||||||
|
const val JSON_HOOK_CLASS_NAMESPACE = "app/revanced/twitter/patches/hook/json"
|
||||||
|
|
||||||
|
const val JSON_HOOK_PATCH_CLASS_DESCRIPTOR = "L$JSON_HOOK_CLASS_NAMESPACE/JsonHookPatch;"
|
||||||
|
|
||||||
|
const val BASE_PATCH_CLASS_NAME = "BaseJsonHook"
|
||||||
|
|
||||||
|
const val JSON_HOOK_CLASS_DESCRIPTOR = "L$JSON_HOOK_CLASS_NAMESPACE/$BASE_PATCH_CLASS_NAME;"
|
||||||
|
|
||||||
|
private lateinit var jsonHookPatchFingerprintResult: MethodFingerprintResult
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,17 @@
|
||||||
|
package app.revanced.patches.twitter.misc.hook.patch
|
||||||
|
|
||||||
|
import app.revanced.patcher.data.BytecodeContext
|
||||||
|
import app.revanced.patcher.patch.BytecodePatch
|
||||||
|
import app.revanced.patcher.patch.PatchResultError
|
||||||
|
import app.revanced.patcher.patch.PatchResultSuccess
|
||||||
|
import app.revanced.patcher.patch.annotations.DependsOn
|
||||||
|
import app.revanced.patches.twitter.misc.hook.json.patch.JsonHookPatch
|
||||||
|
|
||||||
|
@DependsOn([JsonHookPatch::class])
|
||||||
|
abstract class BaseHookPatchPatch(private val hookClassDescriptor: String) : BytecodePatch() {
|
||||||
|
override fun execute(context: BytecodeContext) = try {
|
||||||
|
PatchResultSuccess().also { JsonHookPatch.Hook(context, hookClassDescriptor).add() }
|
||||||
|
} catch (ex: Exception) {
|
||||||
|
PatchResultError(ex)
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,13 +1,13 @@
|
||||||
package app.revanced.patches.twitter.ad.timeline.annotations
|
package app.revanced.patches.twitter.misc.hook.patch.ads.annotations
|
||||||
|
|
||||||
import app.revanced.patcher.annotation.Compatibility
|
import app.revanced.patcher.annotation.Compatibility
|
||||||
import app.revanced.patcher.annotation.Package
|
import app.revanced.patcher.annotation.Package
|
||||||
|
|
||||||
@Compatibility(
|
@Compatibility(
|
||||||
[Package(
|
[Package(
|
||||||
"com.twitter.android", arrayOf("9.65.3-release.0")
|
"com.twitter.android"
|
||||||
)]
|
)]
|
||||||
)
|
)
|
||||||
@Target(AnnotationTarget.CLASS)
|
@Target(AnnotationTarget.CLASS)
|
||||||
@Retention(AnnotationRetention.RUNTIME)
|
@Retention(AnnotationRetention.RUNTIME)
|
||||||
internal annotation class TimelineAdsCompatibility
|
internal annotation class HideAdsCompatibility
|
|
@ -0,0 +1,22 @@
|
||||||
|
package app.revanced.patches.twitter.misc.hook.patch.ads.patch
|
||||||
|
|
||||||
|
import app.revanced.patcher.annotation.Description
|
||||||
|
import app.revanced.patcher.annotation.Name
|
||||||
|
import app.revanced.patcher.annotation.Version
|
||||||
|
import app.revanced.patcher.patch.annotations.DependsOn
|
||||||
|
import app.revanced.patcher.patch.annotations.Patch
|
||||||
|
import app.revanced.patches.twitter.misc.hook.json.patch.JsonHookPatch
|
||||||
|
import app.revanced.patches.twitter.misc.hook.patch.BaseHookPatchPatch
|
||||||
|
import app.revanced.patches.twitter.misc.hook.patch.ads.annotations.HideAdsCompatibility
|
||||||
|
|
||||||
|
@Patch
|
||||||
|
@Name("hide-ads")
|
||||||
|
@DependsOn([JsonHookPatch::class])
|
||||||
|
@Description("Hides ads.")
|
||||||
|
@HideAdsCompatibility
|
||||||
|
@Version("0.0.1")
|
||||||
|
class HideAdsPatch : BaseHookPatchPatch(HOOK_CLASS_DESCRIPTOR) {
|
||||||
|
private companion object {
|
||||||
|
const val HOOK_CLASS_DESCRIPTOR = "Lapp/revanced/twitter/patches/hook/patch/ads/AdsHook;"
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue