2021-10-11 13:22:11 -07:00
|
|
|
module Invidious::Routes::Images
|
|
|
|
# Avatars, banners and other large image assets.
|
|
|
|
def self.ggpht(env)
|
|
|
|
url = env.request.path.lchop("/ggpht")
|
|
|
|
|
2021-10-08 05:55:34 -07:00
|
|
|
headers = (
|
|
|
|
{% unless flag?(:disable_quic) %}
|
|
|
|
if CONFIG.use_quic
|
|
|
|
HTTP::Headers{":authority" => "yt3.ggpht.com"}
|
|
|
|
else
|
|
|
|
HTTP::Headers.new
|
|
|
|
end
|
|
|
|
{% else %}
|
|
|
|
HTTP::Headers.new
|
|
|
|
{% end %}
|
|
|
|
)
|
|
|
|
|
2021-10-11 13:22:11 -07:00
|
|
|
REQUEST_HEADERS_WHITELIST.each do |header|
|
|
|
|
if env.request.headers[header]?
|
|
|
|
headers[header] = env.request.headers[header]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-10-08 05:55:34 -07:00
|
|
|
# We're encapsulating this into a proc in order to easily reuse this
|
|
|
|
# portion of the code for each request block below.
|
|
|
|
request_proc = ->(response : HTTP::Client::Response) {
|
|
|
|
env.response.status_code = response.status_code
|
|
|
|
response.headers.each do |key, value|
|
|
|
|
if !RESPONSE_HEADERS_BLACKLIST.includes?(key.downcase)
|
|
|
|
env.response.headers[key] = value
|
2021-10-11 13:22:11 -07:00
|
|
|
end
|
2021-10-08 05:55:34 -07:00
|
|
|
end
|
2021-10-11 13:22:11 -07:00
|
|
|
|
2021-10-08 05:55:34 -07:00
|
|
|
env.response.headers["Access-Control-Allow-Origin"] = "*"
|
2021-10-11 13:22:11 -07:00
|
|
|
|
2021-10-08 05:55:34 -07:00
|
|
|
if response.status_code >= 300
|
|
|
|
env.response.headers.delete("Transfer-Encoding")
|
|
|
|
return
|
2021-10-11 13:22:11 -07:00
|
|
|
end
|
2021-10-08 05:55:34 -07:00
|
|
|
|
|
|
|
proxy_file(response, env)
|
|
|
|
}
|
|
|
|
|
|
|
|
begin
|
|
|
|
{% unless flag?(:disable_quic) %}
|
|
|
|
if CONFIG.use_quic
|
|
|
|
YT_POOL.client &.get(url, headers) do |resp|
|
|
|
|
return request_proc.call(resp)
|
|
|
|
end
|
|
|
|
else
|
2021-10-08 06:11:10 -07:00
|
|
|
HTTP::Client.get("https://yt3.ggpht.com#{url}") do |resp|
|
2021-10-08 05:55:34 -07:00
|
|
|
return request_proc.call(resp)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
{% else %}
|
|
|
|
# This can likely be optimized into a (small) pool sometime in the future.
|
2021-10-08 06:11:10 -07:00
|
|
|
HTTP::Client.get("https://yt3.ggpht.com#{url}") do |resp|
|
2021-10-08 05:55:34 -07:00
|
|
|
return request_proc.call(resp)
|
|
|
|
end
|
|
|
|
{% end %}
|
2021-10-11 13:22:11 -07:00
|
|
|
rescue ex
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.options_storyboard(env)
|
|
|
|
env.response.headers["Access-Control-Allow-Origin"] = "*"
|
|
|
|
env.response.headers["Access-Control-Allow-Methods"] = "GET, OPTIONS"
|
|
|
|
env.response.headers["Access-Control-Allow-Headers"] = "Content-Type, Range"
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.get_storyboard(env)
|
|
|
|
authority = env.params.url["authority"]
|
|
|
|
id = env.params.url["id"]
|
|
|
|
storyboard = env.params.url["storyboard"]
|
|
|
|
index = env.params.url["index"]
|
|
|
|
|
|
|
|
url = "/sb/#{id}/#{storyboard}/#{index}?#{env.params.query}"
|
|
|
|
|
|
|
|
headers = HTTP::Headers.new
|
|
|
|
|
2021-10-08 06:03:33 -07:00
|
|
|
{% unless flag?(:disable_quic) %}
|
|
|
|
headers[":authority"] = "#{authority}.ytimg.com"
|
|
|
|
{% end %}
|
2021-10-11 13:22:11 -07:00
|
|
|
|
|
|
|
REQUEST_HEADERS_WHITELIST.each do |header|
|
|
|
|
if env.request.headers[header]?
|
|
|
|
headers[header] = env.request.headers[header]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-10-08 06:03:33 -07:00
|
|
|
request_proc = ->(response : HTTP::Client::Response) {
|
|
|
|
env.response.status_code = response.status_code
|
|
|
|
response.headers.each do |key, value|
|
|
|
|
if !RESPONSE_HEADERS_BLACKLIST.includes?(key.downcase)
|
|
|
|
env.response.headers[key] = value
|
2021-10-11 13:22:11 -07:00
|
|
|
end
|
2021-10-08 06:03:33 -07:00
|
|
|
end
|
2021-10-11 13:22:11 -07:00
|
|
|
|
2021-10-08 06:03:33 -07:00
|
|
|
env.response.headers["Connection"] = "close"
|
|
|
|
env.response.headers["Access-Control-Allow-Origin"] = "*"
|
2021-10-11 13:22:11 -07:00
|
|
|
|
2021-10-08 06:03:33 -07:00
|
|
|
if response.status_code >= 300
|
|
|
|
return env.response.headers.delete("Transfer-Encoding")
|
2021-10-11 13:22:11 -07:00
|
|
|
end
|
2021-10-08 06:03:33 -07:00
|
|
|
|
|
|
|
proxy_file(response, env)
|
|
|
|
}
|
|
|
|
|
|
|
|
begin
|
|
|
|
{% unless flag?(:disable_quic) %}
|
|
|
|
if CONFIG.use_quic
|
|
|
|
YT_POOL.client &.get(url, headers) do |resp|
|
|
|
|
return request_proc.call(resp)
|
|
|
|
end
|
|
|
|
else
|
2021-10-08 06:14:37 -07:00
|
|
|
HTTP::Client.get("https://#{authority}.ytimg.com#{url}") do |resp|
|
2021-10-08 06:03:33 -07:00
|
|
|
return request_proc.call(resp)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
{% else %}
|
|
|
|
# This can likely be optimized into a (small) pool sometime in the future.
|
2021-10-08 06:14:37 -07:00
|
|
|
HTTP::Client.get("https://#{authority}.ytimg.com#{url}") do |resp|
|
2021-10-08 06:03:33 -07:00
|
|
|
return request_proc.call(resp)
|
|
|
|
end
|
|
|
|
{% end %}
|
2021-10-11 13:22:11 -07:00
|
|
|
rescue ex
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# ??? maybe also for storyboards?
|
|
|
|
def self.s_p_image(env)
|
|
|
|
id = env.params.url["id"]
|
|
|
|
name = env.params.url["name"]
|
|
|
|
url = env.request.resource
|
|
|
|
|
2021-10-08 06:03:33 -07:00
|
|
|
headers = (
|
|
|
|
{% unless flag?(:disable_quic) %}
|
|
|
|
if CONFIG.use_quic
|
|
|
|
HTTP::Headers{":authority" => "i9.ytimg.com"}
|
|
|
|
else
|
|
|
|
HTTP::Headers.new
|
|
|
|
end
|
|
|
|
{% else %}
|
|
|
|
HTTP::Headers.new
|
|
|
|
{% end %}
|
|
|
|
)
|
|
|
|
|
2021-10-11 13:22:11 -07:00
|
|
|
REQUEST_HEADERS_WHITELIST.each do |header|
|
|
|
|
if env.request.headers[header]?
|
|
|
|
headers[header] = env.request.headers[header]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-10-08 06:03:33 -07:00
|
|
|
request_proc = ->(response : HTTP::Client::Response) {
|
|
|
|
env.response.status_code = response.status_code
|
|
|
|
response.headers.each do |key, value|
|
|
|
|
if !RESPONSE_HEADERS_BLACKLIST.includes?(key.downcase)
|
|
|
|
env.response.headers[key] = value
|
2021-10-11 13:22:11 -07:00
|
|
|
end
|
2021-10-08 06:03:33 -07:00
|
|
|
end
|
2021-10-11 13:22:11 -07:00
|
|
|
|
2021-10-08 06:03:33 -07:00
|
|
|
env.response.headers["Access-Control-Allow-Origin"] = "*"
|
2021-10-11 13:22:11 -07:00
|
|
|
|
2021-10-08 06:03:33 -07:00
|
|
|
if response.status_code >= 300 && response.status_code != 404
|
|
|
|
return env.response.headers.delete("Transfer-Encoding")
|
2021-10-11 13:22:11 -07:00
|
|
|
end
|
2021-10-08 06:03:33 -07:00
|
|
|
|
|
|
|
proxy_file(response, env)
|
|
|
|
}
|
|
|
|
|
|
|
|
begin
|
|
|
|
{% unless flag?(:disable_quic) %}
|
|
|
|
if CONFIG.use_quic
|
|
|
|
YT_POOL.client &.get(url, headers) do |resp|
|
|
|
|
return request_proc.call(resp)
|
|
|
|
end
|
|
|
|
else
|
2021-10-08 06:14:37 -07:00
|
|
|
HTTP::Client.get("https://i9.ytimg.com#{url}") do |resp|
|
2021-10-08 06:03:33 -07:00
|
|
|
return request_proc.call(resp)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
{% else %}
|
|
|
|
# This can likely be optimized into a (small) pool sometime in the future.
|
2021-10-08 06:14:37 -07:00
|
|
|
HTTP::Client.get("https://i9.ytimg.com#{url}") do |resp|
|
2021-10-08 06:03:33 -07:00
|
|
|
return request_proc.call(resp)
|
|
|
|
end
|
|
|
|
{% end %}
|
2021-10-11 13:22:11 -07:00
|
|
|
rescue ex
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.yts_image(env)
|
|
|
|
headers = HTTP::Headers.new
|
|
|
|
REQUEST_HEADERS_WHITELIST.each do |header|
|
|
|
|
if env.request.headers[header]?
|
|
|
|
headers[header] = env.request.headers[header]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
begin
|
|
|
|
YT_POOL.client &.get(env.request.resource, headers) do |response|
|
|
|
|
env.response.status_code = response.status_code
|
|
|
|
response.headers.each do |key, value|
|
|
|
|
if !RESPONSE_HEADERS_BLACKLIST.includes?(key.downcase)
|
|
|
|
env.response.headers[key] = value
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
env.response.headers["Access-Control-Allow-Origin"] = "*"
|
|
|
|
|
|
|
|
if response.status_code >= 300 && response.status_code != 404
|
|
|
|
env.response.headers.delete("Transfer-Encoding")
|
|
|
|
break
|
|
|
|
end
|
|
|
|
|
|
|
|
proxy_file(response, env)
|
|
|
|
end
|
|
|
|
rescue ex
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.thumbnails(env)
|
|
|
|
id = env.params.url["id"]
|
|
|
|
name = env.params.url["name"]
|
|
|
|
|
|
|
|
headers = HTTP::Headers{":authority" => "i.ytimg.com"}
|
|
|
|
|
|
|
|
if name == "maxres.jpg"
|
|
|
|
build_thumbnails(id).each do |thumb|
|
|
|
|
if YT_POOL.client &.head("/vi/#{id}/#{thumb[:url]}.jpg", headers).status_code == 200
|
|
|
|
name = thumb[:url] + ".jpg"
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
url = "/vi/#{id}/#{name}"
|
|
|
|
|
|
|
|
REQUEST_HEADERS_WHITELIST.each do |header|
|
|
|
|
if env.request.headers[header]?
|
|
|
|
headers[header] = env.request.headers[header]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
begin
|
|
|
|
YT_POOL.client &.get(url, headers) do |response|
|
|
|
|
env.response.status_code = response.status_code
|
|
|
|
response.headers.each do |key, value|
|
|
|
|
if !RESPONSE_HEADERS_BLACKLIST.includes?(key.downcase)
|
|
|
|
env.response.headers[key] = value
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
env.response.headers["Access-Control-Allow-Origin"] = "*"
|
|
|
|
|
|
|
|
if response.status_code >= 300 && response.status_code != 404
|
|
|
|
env.response.headers.delete("Transfer-Encoding")
|
|
|
|
break
|
|
|
|
end
|
|
|
|
|
|
|
|
proxy_file(response, env)
|
|
|
|
end
|
|
|
|
rescue ex
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|