fix(YouTube - Video Id): Fix video id not showing the currently playing video (#3038)

Co-authored-by: oSumAtrIX <johan.melkonyan1@web.de>
This commit is contained in:
LisoUseInAIKyrios 2023-09-28 17:11:58 +04:00 committed by GitHub
parent d156833a61
commit f6f226ba28
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 82 additions and 77 deletions

View file

@ -51,9 +51,7 @@ object ReturnYouTubeDislikePatch : BytecodePatch(
override fun execute(context: BytecodeContext) { override fun execute(context: BytecodeContext) {
// region Inject newVideoLoaded event handler to update dislikes when a new video is loaded. // region Inject newVideoLoaded event handler to update dislikes when a new video is loaded.
// This patch needs a few adjustments and lots of testing before it can change to the new video id hook. VideoIdPatch.hookVideoId("$INTEGRATIONS_CLASS_DESCRIPTOR->newVideoLoaded(Ljava/lang/String;)V")
// There's a few corner cases and some weirdness when loading new videos (specifically with detecting shorts).
VideoIdPatch.legacyInjectCall("$INTEGRATIONS_CLASS_DESCRIPTOR->newVideoLoaded(Ljava/lang/String;)V")
// endregion // endregion

View file

@ -96,11 +96,8 @@ object SponsorBlockBytecodePatch : BytecodePatch(
/* /*
* Set current video id. * Set current video id.
*
* The new video id hook seems to work without issues,
* but it's easier to keep using this hook as it's well tested and has no known problems.
*/ */
VideoIdPatch.legacyInjectCallBackgroundPlay("$INTEGRATIONS_SEGMENT_PLAYBACK_CONTROLLER_CLASS_DESCRIPTOR->setCurrentVideoId(Ljava/lang/String;)V") VideoIdPatch.hookBackgroundPlayVideoId("$INTEGRATIONS_SEGMENT_PLAYBACK_CONTROLLER_CLASS_DESCRIPTOR->setCurrentVideoId(Ljava/lang/String;)V")
/* /*
* Seekbar drawing * Seekbar drawing

View file

@ -88,7 +88,9 @@ object SpoofSignaturePatch : BytecodePatch(
) )
// Hook the player parameters. // Hook the player parameters.
PlayerResponseMethodHookPatch.injectProtoBufferHook("$INTEGRATIONS_CLASS_DESCRIPTOR->spoofParameter(Ljava/lang/String;)Ljava/lang/String;") PlayerResponseMethodHookPatch + PlayerResponseMethodHookPatch.Hook.ProtoBufferParameter(
"$INTEGRATIONS_CLASS_DESCRIPTOR->spoofParameter(Ljava/lang/String;)Ljava/lang/String;"
)
// Force the seekbar time and chapters to always show up. // Force the seekbar time and chapters to always show up.
// This is used only if the storyboard spec fetch fails, or when viewing paid videos. // This is used only if the storyboard spec fetch fails, or when viewing paid videos.

View file

@ -109,9 +109,13 @@ object VideoInformationPatch : BytecodePatch(
} }
/* /*
* Inject call for video id * Inject call for video ids
*/ */
VideoIdPatch.injectCall("$INTEGRATIONS_CLASS_DESCRIPTOR->setVideoId(Ljava/lang/String;)V") val videoIdMethodDescriptor = "$INTEGRATIONS_CLASS_DESCRIPTOR->setVideoId(Ljava/lang/String;)V"
VideoIdPatch.hookVideoId(videoIdMethodDescriptor)
VideoIdPatch.hookBackgroundPlayVideoId(videoIdMethodDescriptor)
VideoIdPatch.hookPlayerResponseVideoId(
"$INTEGRATIONS_CLASS_DESCRIPTOR->setPlayerResponseVideoId(Ljava/lang/String;)V")
/* /*
* Set the video time method * Set the video time method

View file

@ -7,66 +7,58 @@ import app.revanced.patcher.extensions.InstructionExtensions.addInstructions
import app.revanced.patcher.patch.BytecodePatch import app.revanced.patcher.patch.BytecodePatch
import app.revanced.patcher.patch.annotation.Patch import app.revanced.patcher.patch.annotation.Patch
import app.revanced.patcher.util.proxy.mutableTypes.MutableMethod import app.revanced.patcher.util.proxy.mutableTypes.MutableMethod
import app.revanced.patches.youtube.misc.fix.playback.SpoofSignaturePatch
import app.revanced.patches.youtube.misc.integrations.IntegrationsPatch import app.revanced.patches.youtube.misc.integrations.IntegrationsPatch
import app.revanced.patches.youtube.video.playerresponse.fingerprint.PlayerParameterBuilderFingerprint import app.revanced.patches.youtube.video.playerresponse.fingerprint.PlayerParameterBuilderFingerprint
import app.revanced.patches.youtube.video.videoid.VideoIdPatch import java.io.Closeable
@Patch( @Patch(
dependencies = [IntegrationsPatch::class], dependencies = [IntegrationsPatch::class],
) )
object PlayerResponseMethodHookPatch : BytecodePatch( object PlayerResponseMethodHookPatch :
setOf( BytecodePatch(setOf(PlayerParameterBuilderFingerprint)),
PlayerParameterBuilderFingerprint, Closeable,
) MutableSet<PlayerResponseMethodHookPatch.Hook> by mutableSetOf() {
) { private const val VIDEO_ID_PARAMETER = 1
private const val playerResponseVideoIdParameter = 1 private const val PROTO_BUFFER_PARAMETER_PARAMETER = 3
private const val playerResponseProtoBufferParameter = 3
/**
* Insert index when adding a video id hook.
*/
private var playerResponseVideoIdInsertIndex = 0
/**
* Insert index when adding a proto buffer override.
* Must be after all video id hooks in the same method.
*/
private var playerResponseProtoBufferInsertIndex = 0
private lateinit var playerResponseMethod: MutableMethod private lateinit var playerResponseMethod: MutableMethod
override fun execute(context: BytecodeContext) { override fun execute(context: BytecodeContext) {
playerResponseMethod = PlayerParameterBuilderFingerprint.result?.mutableMethod
// Hook player parameter. ?: throw PlayerParameterBuilderFingerprint.exception
PlayerParameterBuilderFingerprint.result?.let {
playerResponseMethod = it.mutableMethod
} ?: throw PlayerParameterBuilderFingerprint.exception
} }
/** override fun close() {
* Modify the player parameter proto buffer value. fun hookVideoId(hook: Hook) = playerResponseMethod.addInstruction(
* Used exclusively by [SpoofSignaturePatch]. 0, "invoke-static {p$VIDEO_ID_PARAMETER}, $hook"
*/ )
fun injectProtoBufferHook(methodDescriptor: String) {
playerResponseMethod.addInstructions( fun hookProtoBufferParameter(hook: Hook) = playerResponseMethod.addInstructions(
playerResponseProtoBufferInsertIndex, 0,
""" """
invoke-static {p$playerResponseProtoBufferParameter}, $methodDescriptor invoke-static {p$PROTO_BUFFER_PARAMETER_PARAMETER}, $hook
move-result-object p$playerResponseProtoBufferParameter move-result-object p$PROTO_BUFFER_PARAMETER_PARAMETER
""" """
) )
playerResponseProtoBufferInsertIndex += 2
// Reverse the order in order to preserve insertion order of the hooks.
val beforeVideoIdHooks = filterIsInstance<Hook.ProtoBufferParameterBeforeVideoId>().asReversed()
val videoIdHooks = filterIsInstance<Hook.VideoId>().asReversed()
val afterVideoIdHooks = filterIsInstance<Hook.ProtoBufferParameter>().asReversed()
// Add the hooks in this specific order as they insert instructions at the beginning of the method.
afterVideoIdHooks.forEach(::hookProtoBufferParameter)
videoIdHooks.forEach(::hookVideoId)
beforeVideoIdHooks.forEach(::hookProtoBufferParameter)
} }
/** internal abstract class Hook(private val methodDescriptor: String) {
* Used by [VideoIdPatch]. internal class VideoId(methodDescriptor: String) : Hook(methodDescriptor)
*/
internal fun injectVideoIdHook(methodDescriptor: String) { internal class ProtoBufferParameter(methodDescriptor: String) : Hook(methodDescriptor)
playerResponseMethod.addInstruction( internal class ProtoBufferParameterBeforeVideoId(methodDescriptor: String) : Hook(methodDescriptor)
// Keep injection calls in the order they're added,
// and all video id hooks run before proto buffer hooks. override fun toString() = methodDescriptor
playerResponseVideoIdInsertIndex++,
"invoke-static {p$playerResponseVideoIdParameter}, $methodDescriptor"
)
playerResponseProtoBufferInsertIndex++
} }
} }

View file

@ -26,8 +26,8 @@ object VideoIdPatch : BytecodePatch(
) )
) { ) {
private var videoIdRegister = 0 private var videoIdRegister = 0
private var insertIndex = 0 private var videoIdInsertIndex = 0
private lateinit var insertMethod: MutableMethod private lateinit var videoIdMethod: MutableMethod
private var backgroundPlaybackVideoIdRegister = 0 private var backgroundPlaybackVideoIdRegister = 0
private var backgroundPlaybackInsertIndex = 0 private var backgroundPlaybackInsertIndex = 0
@ -52,8 +52,8 @@ object VideoIdPatch : BytecodePatch(
} ?: throw VideoIdFingerprint.exception } ?: throw VideoIdFingerprint.exception
VideoIdFingerprint.setFields { method, index, register -> VideoIdFingerprint.setFields { method, index, register ->
insertMethod = method videoIdMethod = method
insertIndex = index videoIdInsertIndex = index
videoIdRegister = register videoIdRegister = register
} }
@ -65,21 +65,7 @@ object VideoIdPatch : BytecodePatch(
} }
/** /**
* Adds an invoke-static instruction, called with the new id when the video changes. * Hooks the new video id when the video changes.
*
* Called as soon as the player response is parsed, and called before many other hooks are
* updated such as [PlayerTypeHookPatch].
*
* Supports all videos and functions in all situations.
*
* Be aware, this can be called multiple times for the same video id.
*
* @param methodDescriptor which method to call. Params have to be `Ljava/lang/String;`
*/
fun injectCall(methodDescriptor: String) = PlayerResponseMethodHookPatch.injectVideoIdHook(methodDescriptor)
/**
* Adds an invoke-static instruction, called with the new id when the video changes.
* *
* Supports all videos (regular videos and Shorts). * Supports all videos (regular videos and Shorts).
* *
@ -89,10 +75,10 @@ object VideoIdPatch : BytecodePatch(
* *
* @param methodDescriptor which method to call. Params have to be `Ljava/lang/String;` * @param methodDescriptor which method to call. Params have to be `Ljava/lang/String;`
*/ */
fun legacyInjectCall( fun hookVideoId(
methodDescriptor: String methodDescriptor: String
) = insertMethod.addInstruction( ) = videoIdMethod.addInstruction(
insertIndex++, videoIdInsertIndex++,
"invoke-static {v$videoIdRegister}, $methodDescriptor" "invoke-static {v$videoIdRegister}, $methodDescriptor"
) )
@ -106,11 +92,37 @@ object VideoIdPatch : BytecodePatch(
* *
* @param methodDescriptor which method to call. Params have to be `Ljava/lang/String;` * @param methodDescriptor which method to call. Params have to be `Ljava/lang/String;`
*/ */
fun legacyInjectCallBackgroundPlay( fun hookBackgroundPlayVideoId(
methodDescriptor: String methodDescriptor: String
) = backgroundPlaybackMethod.addInstruction( ) = backgroundPlaybackMethod.addInstruction(
backgroundPlaybackInsertIndex++, // move-result-object offset backgroundPlaybackInsertIndex++, // move-result-object offset
"invoke-static {v$backgroundPlaybackVideoIdRegister}, $methodDescriptor" "invoke-static {v$backgroundPlaybackVideoIdRegister}, $methodDescriptor"
) )
/**
* Hooks the video id of every video when loaded.
* Supports all videos and functions in all situations.
*
* Hook is always called off the main thread.
*
* This hook is called as soon as the player response is parsed,
* and called before many other hooks are updated such as [PlayerTypeHookPatch].
*
* Note: The video id returned here may not be the current video that's being played.
* It's common for multiple Shorts to load at once in preparation
* for the user swiping to the next Short.
*
* For most use cases, you probably want to use
* [hookVideoId] or [hookBackgroundPlayVideoId] instead.
*
* Be aware, this can be called multiple times for the same video id.
*
* @param methodDescriptor which method to call. Params have to be `Ljava/lang/String;`
*/
fun hookPlayerResponseVideoId(methodDescriptor: String) {
PlayerResponseMethodHookPatch + PlayerResponseMethodHookPatch.Hook.VideoId(
methodDescriptor
)
}
} }