fix(YouTube - Client spoof): Removed unused code (#480)

Co-authored-by: oSumAtrIX <johan.melkonyan1@web.de>
This commit is contained in:
LisoUseInAIKyrios 2023-09-26 02:47:01 +04:00 committed by GitHub
parent 89993619fd
commit e6903bff95
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 93 additions and 55 deletions

View File

@ -1,13 +1,15 @@
package app.revanced.integrations.patches.spoof; package app.revanced.integrations.patches.spoof;
import static app.revanced.integrations.patches.spoof.requests.StoryBoardRendererRequester.fetchStoryboardRenderer;
import static app.revanced.integrations.utils.ReVancedUtils.containsAny;
import androidx.annotation.Nullable;
import app.revanced.integrations.patches.VideoInformation; import app.revanced.integrations.patches.VideoInformation;
import app.revanced.integrations.settings.SettingsEnum; import app.revanced.integrations.settings.SettingsEnum;
import app.revanced.integrations.shared.PlayerType; import app.revanced.integrations.shared.PlayerType;
import app.revanced.integrations.utils.LogHelper; import app.revanced.integrations.utils.LogHelper;
import static app.revanced.integrations.patches.spoof.requests.StoryBoardRendererRequester.fetchStoryboardsRenderer;
import static app.revanced.integrations.utils.ReVancedUtils.containsAny;
/** @noinspection unused*/ /** @noinspection unused*/
public class SpoofSignaturePatch { public class SpoofSignaturePatch {
/** /**
@ -31,16 +33,18 @@ public class SpoofSignaturePatch {
*/ */
private static final String SCRIM_PARAMETER = "SAFgAXgB"; private static final String SCRIM_PARAMETER = "SAFgAXgB";
/** /**
* Parameters used in YouTube Shorts. * Parameters used in YouTube Shorts.
*/ */
private static final String SHORTS_PLAYER_PARAMETERS = "8AEB"; private static final String SHORTS_PLAYER_PARAMETERS = "8AEB";
private static boolean isPlayingShorts; /**
* Last video id loaded. Used to prevent reloading the same spec multiple times.
*/
private static volatile String currentVideoId;
private static String storyboardRendererSpec = ""; @Nullable
private static int recommendedLevel; private static volatile StoryboardRenderer renderer;
/** /**
* Injection point. * Injection point.
@ -60,8 +64,7 @@ public class SpoofSignaturePatch {
// Shorts do not need to be spoofed. // Shorts do not need to be spoofed.
//noinspection AssignmentUsedAsCondition if (parameters.startsWith(SHORTS_PLAYER_PARAMETERS)) return parameters;
if (isPlayingShorts = parameters.startsWith(SHORTS_PLAYER_PARAMETERS)) return parameters;
boolean isPlayingFeed = PlayerType.getCurrent() == PlayerType.INLINE_MINIMAL && containsAny(parameters, AUTOPLAY_PARAMETERS); boolean isPlayingFeed = PlayerType.getCurrent() == PlayerType.INLINE_MINIMAL && containsAny(parameters, AUTOPLAY_PARAMETERS);
if (isPlayingFeed) return SettingsEnum.SPOOF_SIGNATURE_IN_FEED.getBoolean() ? if (isPlayingFeed) return SettingsEnum.SPOOF_SIGNATURE_IN_FEED.getBoolean() ?
@ -72,7 +75,13 @@ public class SpoofSignaturePatch {
// This will cause playback issues in the feed, but it's better than manipulating the history. // This will cause playback issues in the feed, but it's better than manipulating the history.
parameters; parameters;
fetchStoryboardsRenderer(VideoInformation.getVideoId()); String videoId = VideoInformation.getVideoId();
if (!videoId.equals(currentVideoId)) {
currentVideoId = videoId;
renderer = fetchStoryboardRenderer(videoId);
LogHelper.printDebug(() -> "Fetched: " + renderer);
}
return INCOGNITO_PARAMETERS; return INCOGNITO_PARAMETERS;
} }
@ -83,14 +92,18 @@ public class SpoofSignaturePatch {
return SettingsEnum.SPOOF_SIGNATURE.getBoolean(); return SettingsEnum.SPOOF_SIGNATURE.getBoolean();
} }
/** /**
* Injection point. * Injection point.
* Called from background threads and from the main thread.
*/ */
@Nullable
public static String getStoryboardRendererSpec(String originalStoryboardRendererSpec) { public static String getStoryboardRendererSpec(String originalStoryboardRendererSpec) {
if (!SettingsEnum.SPOOF_SIGNATURE.getBoolean()) return originalStoryboardRendererSpec; if (!SettingsEnum.SPOOF_SIGNATURE.getBoolean()) return originalStoryboardRendererSpec;
return storyboardRendererSpec; StoryboardRenderer currentRenderer = renderer;
if (currentRenderer == null) return originalStoryboardRendererSpec;
return currentRenderer.getSpec();
} }
/** /**
@ -99,14 +112,10 @@ public class SpoofSignaturePatch {
public static int getRecommendedLevel(int originalLevel) { public static int getRecommendedLevel(int originalLevel) {
if (!SettingsEnum.SPOOF_SIGNATURE.getBoolean()) return originalLevel; if (!SettingsEnum.SPOOF_SIGNATURE.getBoolean()) return originalLevel;
return recommendedLevel; StoryboardRenderer currentRenderer = renderer;
if (currentRenderer == null) return originalLevel;
return currentRenderer.getRecommendedLevel();
} }
public static void setStoryboardRendererSpec(String newlyLoadedStoryboardRendererSpec) {
storyboardRendererSpec = newlyLoadedStoryboardRendererSpec;
}
public static void setRecommendedLevel(int level) {
recommendedLevel = level;
}
} }

View File

@ -0,0 +1,32 @@
package app.revanced.integrations.patches.spoof;
import androidx.annotation.NonNull;
import org.jetbrains.annotations.NotNull;
public final class StoryboardRenderer {
private final String spec;
private final int recommendedLevel;
public StoryboardRenderer(String spec, int recommendedLevel) {
this.spec = spec;
this.recommendedLevel = recommendedLevel;
}
@NonNull
public String getSpec() {
return spec;
}
public int getRecommendedLevel() {
return recommendedLevel;
}
@NotNull
@Override
public String toString() {
return "StoryboardRenderer{" +
"spec='" + spec + '\'' +
", recommendedLevel=" + recommendedLevel +
'}';
}
}

View File

@ -6,21 +6,21 @@ import app.revanced.integrations.requests.Route;
import java.io.IOException; import java.io.IOException;
import java.net.HttpURLConnection; import java.net.HttpURLConnection;
/** @noinspection unused*/ final class PlayerRoutes {
public final class StoryBoardRendererRoutes {
private static final String YT_API_URL = "https://www.youtube.com/youtubei/v1/"; private static final String YT_API_URL = "https://www.youtube.com/youtubei/v1/";
static final Route GET_STORYBOARD_SPEC = new Route( static final Route.CompiledRoute POST_STORYBOARD_SPEC_RENDERER = new Route(
Route.Method.POST, Route.Method.POST,
"player" + "player" +
"?fields=storyboards.playerStoryboardSpecRenderer," + "?fields=storyboards.playerStoryboardSpecRenderer," +
"storyboards.playerLiveStoryboardSpecRenderer" "storyboards.playerLiveStoryboardSpecRenderer"
); ).compile();
private StoryBoardRendererRoutes() { private PlayerRoutes() {
} }
public static HttpURLConnection getPlayerResponseConnectionFromRoute() throws IOException { /** @noinspection SameParameterValue*/
var connection = Requester.getConnectionFromRoute(YT_API_URL, GET_STORYBOARD_SPEC); static HttpURLConnection getPlayerResponseConnectionFromRoute(Route.CompiledRoute route) throws IOException {
var connection = Requester.getConnectionFromCompiledRoute(YT_API_URL, route);
connection.setRequestProperty("User-Agent", "com.google.android.youtube/18.37.36 (Linux; U; Android 12; GB) gzip"); connection.setRequestProperty("User-Agent", "com.google.android.youtube/18.37.36 (Linux; U; Android 12; GB) gzip");
connection.setRequestProperty("X-Goog-Api-Format-Version", "2"); connection.setRequestProperty("X-Goog-Api-Format-Version", "2");
connection.setRequestProperty("Content-Type", "application/json"); connection.setRequestProperty("Content-Type", "application/json");

View File

@ -2,16 +2,18 @@ package app.revanced.integrations.patches.spoof.requests;
import androidx.annotation.NonNull; import androidx.annotation.NonNull;
import androidx.annotation.Nullable; import androidx.annotation.Nullable;
import app.revanced.integrations.patches.spoof.SpoofSignaturePatch; import app.revanced.integrations.patches.spoof.StoryboardRenderer;
import app.revanced.integrations.requests.Requester; import app.revanced.integrations.requests.Requester;
import app.revanced.integrations.utils.LogHelper; import app.revanced.integrations.utils.LogHelper;
import app.revanced.integrations.utils.ReVancedUtils; import app.revanced.integrations.utils.ReVancedUtils;
import org.json.JSONObject; import org.json.JSONObject;
import java.io.IOException;
import java.net.HttpURLConnection; import java.net.HttpURLConnection;
import java.net.SocketTimeoutException; import java.net.SocketTimeoutException;
import java.nio.charset.StandardCharsets; import java.nio.charset.StandardCharsets;
import java.util.Objects;
import static app.revanced.integrations.patches.spoof.requests.PlayerRoutes.POST_STORYBOARD_SPEC_RENDERER;
public class StoryBoardRendererRequester { public class StoryBoardRendererRequester {
private static final String INNER_TUBE_BODY = private static final String INNER_TUBE_BODY =
@ -30,14 +32,15 @@ public class StoryBoardRendererRequester {
private StoryBoardRendererRequester() { private StoryBoardRendererRequester() {
} }
// TODO: Find a way to increase the quality of SeekBar thumbnail previews @Nullable
public static void fetchStoryboardsRenderer(@NonNull String videoId) { public static StoryboardRenderer fetchStoryboardRenderer(@NonNull String videoId) {
ReVancedUtils.verifyOffMainThread();
try { try {
ReVancedUtils.verifyOffMainThread();
Objects.requireNonNull(videoId);
final byte[] innerTubeBody = String.format(INNER_TUBE_BODY, videoId).getBytes(StandardCharsets.UTF_8); final byte[] innerTubeBody = String.format(INNER_TUBE_BODY, videoId).getBytes(StandardCharsets.UTF_8);
HttpURLConnection connection = StoryBoardRendererRoutes.getPlayerResponseConnectionFromRoute(); HttpURLConnection connection = PlayerRoutes.getPlayerResponseConnectionFromRoute(POST_STORYBOARD_SPEC_RENDERER);
connection.getOutputStream().write(innerTubeBody, 0, innerTubeBody.length); connection.getOutputStream().write(innerTubeBody, 0, innerTubeBody.length);
final int responseCode = connection.getResponseCode(); final int responseCode = connection.getResponseCode();
@ -49,33 +52,23 @@ public class StoryBoardRendererRequester {
final String storyboardsRendererTag = storyboards.has("playerLiveStoryboardSpecRenderer") final String storyboardsRendererTag = storyboards.has("playerLiveStoryboardSpecRenderer")
? "playerLiveStoryboardSpecRenderer" ? "playerLiveStoryboardSpecRenderer"
: "playerStoryboardSpecRenderer"; : "playerStoryboardSpecRenderer";
final JSONObject storyboardsRenderer = storyboards.getJSONObject(storyboardsRendererTag);
final String storyboardsRendererSpec = storyboardsRenderer.getString("spec");
SpoofSignaturePatch.setStoryboardRendererSpec(storyboardsRendererSpec); final var renderer = storyboards.getJSONObject(storyboardsRendererTag);
SpoofSignaturePatch.setRecommendedLevel(storyboardsRenderer.getInt("recommendedLevel"));
LogHelper.printDebug(() -> "StoryBoard renderer spec: " + storyboardsRendererSpec);
return new StoryboardRenderer(
renderer.getString("spec"),
renderer.getInt("recommendedLevel")
);
} else { } else {
handleConnectionError("API not available: " + responseCode, null); LogHelper.printException(() -> "API not available: " + responseCode);
}
connection.disconnect(); connection.disconnect();
}
} catch (SocketTimeoutException ex) { } catch (SocketTimeoutException ex) {
handleConnectionError("API timed out", ex); LogHelper.printException(() -> "API timed out", ex);
} catch (IOException ex) {
handleConnectionError(String.format("Failed to fetch StoryBoard URL (%s)", ex.getMessage()), ex);
} catch (Exception ex) { } catch (Exception ex) {
handleConnectionError("Failed to fetch StoryBoard URL", ex); LogHelper.printException(() -> "Failed to fetch StoryBoard URL", ex);
}
} }
private static void handleConnectionError(@NonNull String toastMessage, @Nullable Exception ex) { return null;
if (ex != null)
LogHelper.printException(() -> toastMessage, ex);
else
LogHelper.printException(() -> toastMessage);
SpoofSignaturePatch.setStoryboardRendererSpec("");
} }
} }

View File

@ -16,7 +16,11 @@ public class Requester {
} }
public static HttpURLConnection getConnectionFromRoute(String apiUrl, Route route, String... params) throws IOException { public static HttpURLConnection getConnectionFromRoute(String apiUrl, Route route, String... params) throws IOException {
String url = apiUrl + route.compile(params).getCompiledRoute(); return getConnectionFromCompiledRoute(apiUrl, route.compile(params));
}
public static HttpURLConnection getConnectionFromCompiledRoute(String apiUrl, Route.CompiledRoute route) throws IOException {
String url = apiUrl + route.getCompiledRoute();
HttpURLConnection connection = (HttpURLConnection) new URL(url).openConnection(); HttpURLConnection connection = (HttpURLConnection) new URL(url).openConnection();
connection.setRequestMethod(route.getMethod().name()); connection.setRequestMethod(route.getMethod().name());
connection.setRequestProperty("User-agent", System.getProperty("http.agent") + ";revanced"); connection.setRequestProperty("User-agent", System.getProperty("http.agent") + ";revanced");