Merge pull request #473 from TeamPiped/federated-geo-bypass

Implement geo restrictions bypass.
This commit is contained in:
Kavin 2022-11-27 00:33:27 +00:00 committed by GitHub
commit a9b16ee727
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
19 changed files with 416 additions and 63 deletions

View file

@ -1,6 +1,7 @@
plugins { plugins {
id "com.github.johnrengelman.shadow" version "7.1.2" id "com.github.johnrengelman.shadow" version "7.1.2"
id "java" id "java"
id "io.freefair.lombok" version "6.5.1"
id "eclipse" id "eclipse"
} }

View file

@ -60,6 +60,9 @@ MATRIX_SERVER:https://matrix-client.matrix.org
# If not present, will work in anon mode # If not present, will work in anon mode
#MATRIX_TOKEN:INSERT_HERE #MATRIX_TOKEN:INSERT_HERE
# Geo Restriction Checker for federated bypassing of Geo Restrictions
#GEO_RESTRICTION_CHECKER_URL:INSERT_HERE
# Hibernate properties # Hibernate properties
hibernate.connection.url:jdbc:postgresql://postgres:5432/piped hibernate.connection.url:jdbc:postgresql://postgres:5432/piped
hibernate.connection.driver_class:org.postgresql.Driver hibernate.connection.driver_class:org.postgresql.Driver

View file

@ -1,11 +1,13 @@
package me.kavin.piped.consts; package me.kavin.piped.consts;
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper; import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.json.JsonMapper; import com.fasterxml.jackson.databind.json.JsonMapper;
import com.fasterxml.jackson.databind.node.JsonNodeFactory; import com.fasterxml.jackson.databind.node.JsonNodeFactory;
import com.fasterxml.jackson.databind.node.ObjectNode; import com.fasterxml.jackson.databind.node.ObjectNode;
import it.unimi.dsi.fastutil.objects.Object2ObjectOpenHashMap; import it.unimi.dsi.fastutil.objects.Object2ObjectOpenHashMap;
import me.kavin.piped.utils.PageMixin; import me.kavin.piped.utils.PageMixin;
import me.kavin.piped.utils.RequestUtils;
import me.kavin.piped.utils.resp.ListLinkHandlerMixin; import me.kavin.piped.utils.resp.ListLinkHandlerMixin;
import okhttp3.OkHttpClient; import okhttp3.OkHttpClient;
import okhttp3.brotli.BrotliInterceptor; import okhttp3.brotli.BrotliInterceptor;
@ -23,6 +25,7 @@ import java.net.InetSocketAddress;
import java.net.ProxySelector; import java.net.ProxySelector;
import java.util.List; import java.util.List;
import java.util.Properties; import java.util.Properties;
import java.util.regex.Pattern;
public class Constants { public class Constants {
@ -80,11 +83,16 @@ public class Constants {
public static final String MATRIX_TOKEN; public static final String MATRIX_TOKEN;
public static final String GEO_RESTRICTION_CHECKER_URL;
public static final String YOUTUBE_COUNTRY;
public static final String VERSION; public static final String VERSION;
public static final ObjectMapper mapper = JsonMapper.builder() public static final ObjectMapper mapper = JsonMapper.builder()
.addMixIn(Page.class, PageMixin.class) .addMixIn(Page.class, PageMixin.class)
.addMixIn(ListLinkHandler.class, ListLinkHandlerMixin.class) .addMixIn(ListLinkHandler.class, ListLinkHandlerMixin.class)
.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false)
.build(); .build();
public static final Object2ObjectOpenHashMap<String, String> hibernateProperties = new Object2ObjectOpenHashMap<>(); public static final Object2ObjectOpenHashMap<String, String> hibernateProperties = new Object2ObjectOpenHashMap<>();
@ -130,6 +138,7 @@ public class Constants {
}); });
MATRIX_SERVER = getProperty(prop, "MATRIX_SERVER", "https://matrix-client.matrix.org"); MATRIX_SERVER = getProperty(prop, "MATRIX_SERVER", "https://matrix-client.matrix.org");
MATRIX_TOKEN = getProperty(prop, "MATRIX_TOKEN"); MATRIX_TOKEN = getProperty(prop, "MATRIX_TOKEN");
GEO_RESTRICTION_CHECKER_URL = getProperty(prop, "GEO_RESTRICTION_CHECKER_URL");
prop.forEach((_key, _value) -> { prop.forEach((_key, _value) -> {
String key = String.valueOf(_key), value = String.valueOf(_value); String key = String.valueOf(_key), value = String.valueOf(_value);
if (key.startsWith("hibernate")) if (key.startsWith("hibernate"))
@ -164,6 +173,18 @@ public class Constants {
} }
h2client = builder.build(); h2client = builder.build();
h2_no_redir_client = builder_noredir.build(); h2_no_redir_client = builder_noredir.build();
String temp = null;
try {
var html = RequestUtils.sendGet("https://www.youtube.com/");
var regex = Pattern.compile("GL\":\"([A-Z]{2})\"", Pattern.MULTILINE);
var matcher = regex.matcher(html);
if (matcher.find()) {
temp = matcher.group(1);
}
} catch (Exception ignored) {
System.err.println("Failed to get country from YouTube!");
}
YOUTUBE_COUNTRY = temp;
VERSION = new File("VERSION").exists() ? VERSION = new File("VERSION").exists() ?
IOUtils.toString(new FileReader("VERSION")) : IOUtils.toString(new FileReader("VERSION")) :
"unknown"; "unknown";

View file

@ -9,6 +9,8 @@ import it.unimi.dsi.fastutil.objects.ObjectArrayList;
import me.kavin.piped.consts.Constants; import me.kavin.piped.consts.Constants;
import me.kavin.piped.utils.*; import me.kavin.piped.utils.*;
import me.kavin.piped.utils.obj.*; import me.kavin.piped.utils.obj.*;
import me.kavin.piped.utils.obj.federation.FederatedGeoBypassRequest;
import me.kavin.piped.utils.obj.federation.FederatedGeoBypassResponse;
import me.kavin.piped.utils.obj.federation.FederatedVideoInfo; import me.kavin.piped.utils.obj.federation.FederatedVideoInfo;
import me.kavin.piped.utils.resp.InvalidRequestResponse; import me.kavin.piped.utils.resp.InvalidRequestResponse;
import me.kavin.piped.utils.resp.VideoResolvedResponse; import me.kavin.piped.utils.resp.VideoResolvedResponse;
@ -18,20 +20,21 @@ import org.schabi.newpipe.extractor.ListExtractor;
import org.schabi.newpipe.extractor.Page; import org.schabi.newpipe.extractor.Page;
import org.schabi.newpipe.extractor.comments.CommentsInfo; import org.schabi.newpipe.extractor.comments.CommentsInfo;
import org.schabi.newpipe.extractor.comments.CommentsInfoItem; import org.schabi.newpipe.extractor.comments.CommentsInfoItem;
import org.schabi.newpipe.extractor.exceptions.GeographicRestrictionException;
import org.schabi.newpipe.extractor.stream.StreamInfo; import org.schabi.newpipe.extractor.stream.StreamInfo;
import org.schabi.newpipe.extractor.stream.StreamType;
import org.schabi.newpipe.extractor.utils.JsonUtils; import org.schabi.newpipe.extractor.utils.JsonUtils;
import java.io.IOException; import java.io.IOException;
import java.util.List; import java.util.List;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException; import java.util.concurrent.TimeoutException;
import static java.nio.charset.StandardCharsets.UTF_8; import static java.nio.charset.StandardCharsets.UTF_8;
import static me.kavin.piped.consts.Constants.YOUTUBE_SERVICE; import static me.kavin.piped.consts.Constants.YOUTUBE_SERVICE;
import static me.kavin.piped.consts.Constants.mapper; import static me.kavin.piped.consts.Constants.mapper;
import static me.kavin.piped.utils.CollectionUtils.collectRelatedItems; import static me.kavin.piped.utils.URLUtils.rewriteURL;
import static me.kavin.piped.utils.URLUtils.*; import static me.kavin.piped.utils.URLUtils.substringYouTube;
import static org.schabi.newpipe.extractor.NewPipe.getPreferredContentCountry; import static org.schabi.newpipe.extractor.NewPipe.getPreferredContentCountry;
import static org.schabi.newpipe.extractor.NewPipe.getPreferredLocalization; import static org.schabi.newpipe.extractor.NewPipe.getPreferredLocalization;
import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.getJsonPostResponse; import static org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper.getJsonPostResponse;
@ -48,6 +51,9 @@ public class StreamHandlers {
try { try {
return StreamInfo.getInfo("https://www.youtube.com/watch?v=" + videoId); return StreamInfo.getInfo("https://www.youtube.com/watch?v=" + videoId);
} catch (Exception e) { } catch (Exception e) {
if (e instanceof GeographicRestrictionException) {
return null;
}
transaction.setThrowable(e); transaction.setThrowable(e);
ExceptionUtils.rethrow(e); ExceptionUtils.rethrow(e);
} finally { } finally {
@ -98,21 +104,93 @@ public class StreamHandlers {
return null; return null;
}); });
final List<Subtitle> subtitles = new ObjectArrayList<>(); StreamInfo info = null;
final List<ChapterSegment> chapters = new ObjectArrayList<>();
final StreamInfo info = futureStream.get(10, TimeUnit.SECONDS); try {
info = futureStream.get(10, TimeUnit.SECONDS);
} catch (ExecutionException e) {
if (!(e.getCause() instanceof GeographicRestrictionException)) {
ExceptionUtils.rethrow(e);
}
}
info.getStreamSegments().forEach(segment -> chapters.add(new ChapterSegment(segment.getTitle(), rewriteURL(segment.getPreviewUrl()), if (info == null) {
segment.getStartTimeSeconds()))); // We're geo restricted
info.getSubtitles() if (Constants.MATRIX_TOKEN != null && Constants.GEO_RESTRICTION_CHECKER_URL != null) {
.forEach(subtitle -> subtitles.add(new Subtitle(rewriteURL(subtitle.getContent()),
subtitle.getFormat().getMimeType(), subtitle.getDisplayLanguageName(),
subtitle.getLanguageTag(), subtitle.isAutoGenerated())));
final List<PipedStream> videoStreams = new ObjectArrayList<>(); List<String> allowedCountries = new ObjectArrayList<>();
final List<PipedStream> audioStreams = new ObjectArrayList<>();
{
var restrictedTree = RequestUtils.sendGetJson(Constants.GEO_RESTRICTION_CHECKER_URL + "/api/region/check?video_id=" + videoId);
var it = restrictedTree.get("regions").elements();
while (it.hasNext()) {
var region = it.next();
allowedCountries.add(region.textValue());
}
}
if (allowedCountries.isEmpty())
throw new GeographicRestrictionException("Federated bypass failed, video not available in any region");
MatrixHelper.sendEvent("video.piped.stream.bypass.request", new FederatedGeoBypassRequest(videoId, allowedCountries));
var listener = new WaitingListener(10_000);
GeoRestrictionBypassHelper.makeRequest(videoId, listener);
listener.waitFor();
FederatedGeoBypassResponse federatedGeoBypassResponse = GeoRestrictionBypassHelper.getResponse(videoId);
if (federatedGeoBypassResponse == null)
throw new GeographicRestrictionException("Federated bypass failed, likely not authorized or no suitable instances found for country");
Streams streams = federatedGeoBypassResponse.getData();
String lbryId;
try {
lbryId = futureLbryId.get(2, TimeUnit.SECONDS);
} catch (Exception e) {
lbryId = null;
}
if (lbryId != null) {
streams.lbryId = lbryId;
}
String lbryURL;
try {
lbryURL = futureLBRY.get(3, TimeUnit.SECONDS);
} catch (Exception e) {
lbryURL = null;
}
if (lbryURL != null)
streams.videoStreams.add(0, new PipedStream(lbryURL, "MP4", "LBRY", "video/mp4", false));
// Attempt to get dislikes calculating with the RYD API rating
if (streams.dislikes < 0 && streams.likes >= 0) {
double rating;
try {
rating = futureDislikeRating.get(3, TimeUnit.SECONDS);
} catch (Exception e) {
rating = -1;
}
if (rating > 1 && rating <= 5) {
streams.dislikes = Math.round(streams.likes * ((5 - rating) / (rating - 1)));
}
}
return mapper.writeValueAsBytes(streams);
} else if (Constants.GEO_RESTRICTION_CHECKER_URL == null) {
throw new GeographicRestrictionException("This instance does not have a geo restriction checker set in its configuration");
}
throw new GeographicRestrictionException("Geo restricted content, this instance is not part of the Matrix Federation protocol");
}
Streams streams = CollectionUtils.collectStreamInfo(info);
String lbryURL = null; String lbryURL = null;
@ -123,40 +201,20 @@ public class StreamHandlers {
} }
if (lbryURL != null) if (lbryURL != null)
videoStreams.add(new PipedStream(lbryURL, "MP4", "LBRY", "video/mp4", false)); streams.videoStreams.add(0, new PipedStream(lbryURL, "MP4", "LBRY", "video/mp4", false));
boolean livestream = info.getStreamType() == StreamType.LIVE_STREAM;
if (!livestream) {
info.getVideoOnlyStreams().forEach(stream -> videoStreams.add(new PipedStream(rewriteVideoURL(stream.getContent()),
String.valueOf(stream.getFormat()), stream.getResolution(), stream.getFormat().getMimeType(), true,
stream.getBitrate(), stream.getInitStart(), stream.getInitEnd(), stream.getIndexStart(),
stream.getIndexEnd(), stream.getCodec(), stream.getWidth(), stream.getHeight(), 30)));
info.getVideoStreams()
.forEach(stream -> videoStreams
.add(new PipedStream(rewriteVideoURL(stream.getContent()), String.valueOf(stream.getFormat()),
stream.getResolution(), stream.getFormat().getMimeType(), false)));
info.getAudioStreams()
.forEach(stream -> audioStreams.add(new PipedStream(rewriteVideoURL(stream.getContent()),
String.valueOf(stream.getFormat()), stream.getAverageBitrate() + " kbps",
stream.getFormat().getMimeType(), false, stream.getBitrate(), stream.getInitStart(),
stream.getInitEnd(), stream.getIndexStart(), stream.getIndexEnd(), stream.getCodec(), stream.getAudioTrackId(), stream.getAudioTrackName())));
}
final List<ContentItem> relatedStreams = collectRelatedItems(info.getRelatedItems());
long time = info.getUploadDate() != null ? info.getUploadDate().offsetDateTime().toInstant().toEpochMilli() long time = info.getUploadDate() != null ? info.getUploadDate().offsetDateTime().toInstant().toEpochMilli()
: System.currentTimeMillis(); : System.currentTimeMillis();
if (info.getUploadDate() != null && System.currentTimeMillis() - time < TimeUnit.DAYS.toMillis(Constants.FEED_RETENTION)) { if (info.getUploadDate() != null && System.currentTimeMillis() - time < TimeUnit.DAYS.toMillis(Constants.FEED_RETENTION)) {
VideoHelpers.updateVideo(info.getId(), info, time); VideoHelpers.updateVideo(info.getId(), info, time);
StreamInfo finalInfo = info;
Multithreading.runAsync(() -> { Multithreading.runAsync(() -> {
try { try {
MatrixHelper.sendEvent("video.piped.stream.info", new FederatedVideoInfo( MatrixHelper.sendEvent("video.piped.stream.info", new FederatedVideoInfo(
info.getId(), StringUtils.substring(info.getUploaderUrl(), -24), finalInfo.getId(), StringUtils.substring(finalInfo.getUploaderUrl(), -24),
info.getName(), finalInfo.getName(),
info.getDuration(), info.getViewCount()) finalInfo.getDuration(), finalInfo.getViewCount())
); );
} catch (IOException e) { } catch (IOException e) {
throw new RuntimeException(e); throw new RuntimeException(e);
@ -172,8 +230,10 @@ public class StreamHandlers {
lbryId = null; lbryId = null;
} }
streams.lbryId = lbryId;
// Attempt to get dislikes calculating with the RYD API rating // Attempt to get dislikes calculating with the RYD API rating
if (info.getDislikeCount() < 0 && info.getLikeCount() >= 0) { if (streams.dislikes < 0 && streams.likes >= 0) {
double rating; double rating;
try { try {
rating = futureDislikeRating.get(3, TimeUnit.SECONDS); rating = futureDislikeRating.get(3, TimeUnit.SECONDS);
@ -182,17 +242,10 @@ public class StreamHandlers {
} }
if (rating > 1 && rating <= 5) { if (rating > 1 && rating <= 5) {
info.setDislikeCount(Math.round(info.getLikeCount() * ((5 - rating) / (rating - 1)))); streams.dislikes = Math.round(streams.likes * ((5 - rating) / (rating - 1)));
} }
} }
final Streams streams = new Streams(info.getName(), info.getDescription().getContent(),
info.getTextualUploadDate(), info.getUploaderName(), substringYouTube(info.getUploaderUrl()),
rewriteURL(info.getUploaderAvatarUrl()), rewriteURL(info.getThumbnailUrl()), info.getDuration(),
info.getViewCount(), info.getLikeCount(), info.getDislikeCount(), info.getUploaderSubscriberCount(), info.isUploaderVerified(),
audioStreams, videoStreams, relatedStreams, subtitles, livestream, rewriteVideoURL(info.getHlsUrl()),
rewriteVideoURL(info.getDashMpdUrl()), lbryId, chapters);
return mapper.writeValueAsBytes(streams); return mapper.writeValueAsBytes(streams);
} }

View file

@ -1,21 +1,66 @@
package me.kavin.piped.utils; package me.kavin.piped.utils;
import me.kavin.piped.utils.obj.ChannelItem; import it.unimi.dsi.fastutil.objects.ObjectArrayList;
import me.kavin.piped.utils.obj.ContentItem; import me.kavin.piped.utils.obj.*;
import me.kavin.piped.utils.obj.PlaylistItem;
import me.kavin.piped.utils.obj.StreamItem;
import org.schabi.newpipe.extractor.InfoItem; import org.schabi.newpipe.extractor.InfoItem;
import org.schabi.newpipe.extractor.channel.ChannelInfoItem; import org.schabi.newpipe.extractor.channel.ChannelInfoItem;
import org.schabi.newpipe.extractor.playlist.PlaylistInfoItem; import org.schabi.newpipe.extractor.playlist.PlaylistInfoItem;
import org.schabi.newpipe.extractor.stream.StreamInfo;
import org.schabi.newpipe.extractor.stream.StreamInfoItem; import org.schabi.newpipe.extractor.stream.StreamInfoItem;
import org.schabi.newpipe.extractor.stream.StreamType;
import java.util.List; import java.util.List;
import static me.kavin.piped.utils.URLUtils.rewriteURL; import static me.kavin.piped.utils.URLUtils.*;
import static me.kavin.piped.utils.URLUtils.substringYouTube;
public class CollectionUtils { public class CollectionUtils {
public static Streams collectStreamInfo(StreamInfo info) {
final List<Subtitle> subtitles = new ObjectArrayList<>();
final List<ChapterSegment> chapters = new ObjectArrayList<>();
info.getStreamSegments().forEach(segment -> chapters.add(new ChapterSegment(segment.getTitle(), rewriteURL(segment.getPreviewUrl()),
segment.getStartTimeSeconds())));
info.getSubtitles()
.forEach(subtitle -> subtitles.add(new Subtitle(rewriteURL(subtitle.getContent()),
subtitle.getFormat().getMimeType(), subtitle.getDisplayLanguageName(),
subtitle.getLanguageTag(), subtitle.isAutoGenerated())));
final List<PipedStream> videoStreams = new ObjectArrayList<>();
final List<PipedStream> audioStreams = new ObjectArrayList<>();
boolean livestream = info.getStreamType() == StreamType.LIVE_STREAM;
if (!livestream) {
info.getVideoOnlyStreams().forEach(stream -> videoStreams.add(new PipedStream(rewriteVideoURL(stream.getContent()),
String.valueOf(stream.getFormat()), stream.getResolution(), stream.getFormat().getMimeType(), true,
stream.getBitrate(), stream.getInitStart(), stream.getInitEnd(), stream.getIndexStart(),
stream.getIndexEnd(), stream.getCodec(), stream.getWidth(), stream.getHeight(), 30)));
info.getVideoStreams()
.forEach(stream -> videoStreams
.add(new PipedStream(rewriteVideoURL(stream.getContent()), String.valueOf(stream.getFormat()),
stream.getResolution(), stream.getFormat().getMimeType(), false)));
info.getAudioStreams()
.forEach(stream -> audioStreams.add(new PipedStream(rewriteVideoURL(stream.getContent()),
String.valueOf(stream.getFormat()), stream.getAverageBitrate() + " kbps",
stream.getFormat().getMimeType(), false, stream.getBitrate(), stream.getInitStart(),
stream.getInitEnd(), stream.getIndexStart(), stream.getIndexEnd(), stream.getCodec(), stream.getAudioTrackId(), stream.getAudioTrackName())));
}
final List<ContentItem> relatedStreams = collectRelatedItems(info.getRelatedItems());
final Streams streams = new Streams(info.getName(), info.getDescription().getContent(),
info.getTextualUploadDate(), info.getUploaderName(), substringYouTube(info.getUploaderUrl()),
rewriteURL(info.getUploaderAvatarUrl()), rewriteURL(info.getThumbnailUrl()), info.getDuration(),
info.getViewCount(), info.getLikeCount(), info.getDislikeCount(), info.getUploaderSubscriberCount(), info.isUploaderVerified(),
audioStreams, videoStreams, relatedStreams, subtitles, livestream, rewriteVideoURL(info.getHlsUrl()),
rewriteVideoURL(info.getDashMpdUrl()), null, chapters);
return streams;
}
public static List<ContentItem> collectRelatedItems(List<? extends InfoItem> items) { public static List<ContentItem> collectRelatedItems(List<? extends InfoItem> items) {
return items return items
.stream() .stream()

View file

@ -0,0 +1,104 @@
package me.kavin.piped.utils;
import it.unimi.dsi.fastutil.objects.Object2LongOpenHashMap;
import it.unimi.dsi.fastutil.objects.Object2ObjectOpenHashMap;
import it.unimi.dsi.fastutil.objects.ObjectArrayList;
import me.kavin.piped.utils.obj.federation.FederatedGeoBypassResponse;
import java.util.List;
import java.util.Map;
import java.util.Timer;
import java.util.TimerTask;
import java.util.concurrent.TimeUnit;
public class GeoRestrictionBypassHelper {
private static final Map<String, Long> requestsMap = new Object2LongOpenHashMap<>();
private static final Map<String, Response> responsesMap = new Object2ObjectOpenHashMap<>();
private static final List<ListenerRequest> waitingListeners = new ObjectArrayList<>();
public static void makeRequest(String id, WaitingListener listener) {
synchronized (requestsMap) {
if (!requestsMap.containsKey(id))
requestsMap.put(id, System.currentTimeMillis());
else {
synchronized (responsesMap) {
if (responsesMap.containsKey(id)) {
listener.done();
return;
}
}
}
}
synchronized (waitingListeners) {
waitingListeners.add(new ListenerRequest(id, listener));
}
}
public static void addResponse(FederatedGeoBypassResponse response) {
String id = response.getVideoId();
synchronized (requestsMap) {
if (requestsMap.containsKey(id)) {
synchronized (responsesMap) {
responsesMap.put(id, new Response(response));
}
synchronized (waitingListeners) {
for (ListenerRequest waitingListener : waitingListeners) {
if (waitingListener.id.equals(id)) {
waitingListener.listener.done();
}
}
}
}
}
}
public static FederatedGeoBypassResponse getResponse(String id) {
synchronized (responsesMap) {
if (responsesMap.containsKey(id)) {
return responsesMap.get(id).response;
}
}
return null;
}
private static final class ListenerRequest {
private final String id;
private final long creationTime = System.currentTimeMillis();
private final WaitingListener listener;
public ListenerRequest(String id, WaitingListener listener) {
this.id = id;
this.listener = listener;
}
}
private static final class Response {
private final FederatedGeoBypassResponse response;
private final long time = System.currentTimeMillis();
public Response(FederatedGeoBypassResponse response) {
this.response = response;
}
}
static {
long time = TimeUnit.SECONDS.toMillis(60);
// Start evictor timer to remove old requests, responses and listeners
new Timer().scheduleAtFixedRate(new TimerTask() {
@Override
public void run() {
synchronized (requestsMap) {
requestsMap.entrySet().removeIf(e -> e.getValue() < System.currentTimeMillis() - time);
}
synchronized (responsesMap) {
responsesMap.entrySet().removeIf(e -> e.getValue().time < System.currentTimeMillis() - time);
}
synchronized (waitingListeners) {
waitingListeners.removeIf(e -> e.creationTime < System.currentTimeMillis() - time);
}
}
}, time, time);
}
}

View file

@ -43,4 +43,12 @@ public class RequestUtils {
} }
} }
} }
public static JsonNode sendGetJson(String url, String ua) throws IOException {
return getJsonNode(Constants.h2client, new Request.Builder().header("User-Agent", ua).url(url).build());
}
public static JsonNode sendGetJson(String url) throws IOException {
return sendGetJson(url, Constants.USER_AGENT);
}
} }

View file

@ -0,0 +1,21 @@
package me.kavin.piped.utils;
import lombok.AllArgsConstructor;
@AllArgsConstructor
public class WaitingListener {
private final long maxWaitTime;
public void waitFor() throws InterruptedException {
synchronized (this) {
this.wait(maxWaitTime);
}
}
public void done() {
synchronized (this) {
this.notifyAll();
}
}
}

View file

@ -1,14 +1,20 @@
package me.kavin.piped.utils.matrix; package me.kavin.piped.utils.matrix;
import com.fasterxml.jackson.databind.JsonNode; import com.fasterxml.jackson.databind.JsonNode;
import me.kavin.piped.consts.Constants;
import me.kavin.piped.utils.*; import me.kavin.piped.utils.*;
import me.kavin.piped.utils.obj.MatrixHelper;
import me.kavin.piped.utils.obj.Streams;
import me.kavin.piped.utils.obj.federation.FederatedChannelInfo; import me.kavin.piped.utils.obj.federation.FederatedChannelInfo;
import me.kavin.piped.utils.obj.federation.FederatedGeoBypassRequest;
import me.kavin.piped.utils.obj.federation.FederatedGeoBypassResponse;
import me.kavin.piped.utils.obj.federation.FederatedVideoInfo; import me.kavin.piped.utils.obj.federation.FederatedVideoInfo;
import okhttp3.MediaType; import okhttp3.MediaType;
import okhttp3.OkHttpClient; import okhttp3.OkHttpClient;
import okhttp3.Request; import okhttp3.Request;
import okhttp3.RequestBody; import okhttp3.RequestBody;
import org.hibernate.StatelessSession; import org.hibernate.StatelessSession;
import org.schabi.newpipe.extractor.stream.StreamInfo;
import java.io.IOException; import java.io.IOException;
import java.util.Set; import java.util.Set;
@ -121,17 +127,40 @@ public class SyncRunner implements Runnable {
for (var event : events) { for (var event : events) {
var type = event.get("type").asText(); var type = event.get("type").asText();
var content = event.at("/content/content");
if (type.startsWith("video.piped.stream.bypass.")) {
switch (type) {
case "video.piped.stream.bypass.request" -> {
FederatedGeoBypassRequest bypassRequest = mapper.treeToValue(content, FederatedGeoBypassRequest.class);
if (bypassRequest.getAllowedCountries().contains(Constants.YOUTUBE_COUNTRY)) {
// We're capable of helping another instance!
Multithreading.runAsync(() -> {
try {
StreamInfo info = StreamInfo.getInfo("https://www.youtube.com/watch?v=" + bypassRequest.getVideoId());
Streams streams = CollectionUtils.collectStreamInfo(info);
FederatedGeoBypassResponse bypassResponse = new FederatedGeoBypassResponse(bypassRequest.getVideoId(), Constants.YOUTUBE_COUNTRY, streams);
MatrixHelper.sendEvent("video.piped.stream.bypass.response", bypassResponse);
} catch (Exception ignored) {
}
});
}
}
case "video.piped.stream.bypass.response" -> {
FederatedGeoBypassResponse bypassResponse = mapper.treeToValue(content, FederatedGeoBypassResponse.class);
GeoRestrictionBypassHelper.addResponse(bypassResponse);
}
}
}
if (event.get("sender").asText().equals(user_id)) { if (event.get("sender").asText().equals(user_id)) {
if (type.startsWith("video.piped.stream.bypass.")) {
// TODO: Implement geo-restriction bypassing
}
continue; continue;
} }
var content = event.at("/content/content");
switch (type) { switch (type) {
case "video.piped.stream.info" -> { case "video.piped.stream.info" -> {

View file

@ -1,5 +1,8 @@
package me.kavin.piped.utils.obj; package me.kavin.piped.utils.obj;
import lombok.NoArgsConstructor;
@NoArgsConstructor
public class ChannelItem extends ContentItem { public class ChannelItem extends ContentItem {
public final String type = "channel"; public final String type = "channel";

View file

@ -1,5 +1,8 @@
package me.kavin.piped.utils.obj; package me.kavin.piped.utils.obj;
import lombok.NoArgsConstructor;
@NoArgsConstructor
public class ChapterSegment { public class ChapterSegment {
public String title, image; public String title, image;

View file

@ -1,6 +1,17 @@
package me.kavin.piped.utils.obj; package me.kavin.piped.utils.obj;
public class ContentItem { import com.fasterxml.jackson.annotation.JsonSubTypes;
import com.fasterxml.jackson.annotation.JsonTypeInfo;
import lombok.NoArgsConstructor;
@NoArgsConstructor
@JsonTypeInfo(use = JsonTypeInfo.Id.DEDUCTION, property = "type")
@JsonSubTypes({
@JsonSubTypes.Type(name = "stream", value = StreamItem.class),
@JsonSubTypes.Type(name = "channel", value = ChannelItem.class),
@JsonSubTypes.Type(name = "playlist", value = PlaylistItem.class),
})
public abstract class ContentItem {
public String url; public String url;

View file

@ -1,5 +1,8 @@
package me.kavin.piped.utils.obj; package me.kavin.piped.utils.obj;
import lombok.NoArgsConstructor;
@NoArgsConstructor
public class PipedStream { public class PipedStream {
public String url, format, quality, mimeType, codec, audioTrackId, audioTrackName; public String url, format, quality, mimeType, codec, audioTrackId, audioTrackName;

View file

@ -1,5 +1,8 @@
package me.kavin.piped.utils.obj; package me.kavin.piped.utils.obj;
import lombok.NoArgsConstructor;
@NoArgsConstructor
public class PlaylistItem extends ContentItem { public class PlaylistItem extends ContentItem {
public final String type = "playlist"; public final String type = "playlist";

View file

@ -1,5 +1,8 @@
package me.kavin.piped.utils.obj; package me.kavin.piped.utils.obj;
import lombok.NoArgsConstructor;
@NoArgsConstructor
public class StreamItem extends ContentItem { public class StreamItem extends ContentItem {
public final String type = "stream"; public final String type = "stream";

View file

@ -1,9 +1,11 @@
package me.kavin.piped.utils.obj; package me.kavin.piped.utils.obj;
import lombok.NoArgsConstructor;
import me.kavin.piped.consts.Constants; import me.kavin.piped.consts.Constants;
import java.util.List; import java.util.List;
@NoArgsConstructor
public class Streams { public class Streams {
public String title, description, uploadDate, uploader, uploaderUrl, uploaderAvatar, thumbnailUrl, hls, dash, public String title, description, uploadDate, uploader, uploaderUrl, uploaderAvatar, thumbnailUrl, hls, dash,

View file

@ -1,9 +1,12 @@
package me.kavin.piped.utils.obj; package me.kavin.piped.utils.obj;
import lombok.NoArgsConstructor;
@NoArgsConstructor
public class Subtitle { public class Subtitle {
public final String url, mimeType, name, code; public String url, mimeType, name, code;
public final boolean autoGenerated; public boolean autoGenerated;
public Subtitle(String url, String mimeType, String name, String code, boolean autoGenerated) { public Subtitle(String url, String mimeType, String name, String code, boolean autoGenerated) {
this.url = url; this.url = url;

View file

@ -0,0 +1,15 @@
package me.kavin.piped.utils.obj.federation;
import lombok.AllArgsConstructor;
import lombok.Getter;
import lombok.NoArgsConstructor;
import java.util.List;
@NoArgsConstructor
@AllArgsConstructor
@Getter
public class FederatedGeoBypassRequest {
private String videoId;
private List<String> allowedCountries;
}

View file

@ -0,0 +1,22 @@
package me.kavin.piped.utils.obj.federation;
import lombok.Getter;
import lombok.NoArgsConstructor;
import me.kavin.piped.consts.Constants;
import me.kavin.piped.utils.obj.Streams;
@NoArgsConstructor
@Getter
public class FederatedGeoBypassResponse {
private String videoId;
private String country;
private String videoProxyUrl;
private Streams data;
public FederatedGeoBypassResponse(String videoId, String country, Streams data) {
this.videoId = videoId;
this.country = country;
this.data = data;
this.videoProxyUrl = Constants.PROXY_PART;
}
}