Allow a ClientConfig to be passed to YoutubeAPI endpoint handlers.

This commit is contained in:
Samantaz Fox 2021-07-26 01:10:11 +02:00
parent a2dc4fdc9a
commit ff3fce7afa
No known key found for this signature in database
GPG key ID: F42821059186176E
3 changed files with 122 additions and 45 deletions

View file

@ -26,32 +26,96 @@ module YoutubeAPI
} }
#################################################################### ####################################################################
# make_context(region) # struct ClientConfig
#
# Data structure used to pass a client configuration to the different
# API endpoints handlers.
#
# Use case examples:
#
# ```
# # Get Norwegian search results
# conf_1 = ClientConfig.new(region: "NO")
# YoutubeAPI::search("Kollektivet", params: "", client_config: conf_1)
#
# # Use the Android client to request video streams URLs
# conf_2 = ClientConfig.new(client_type: ClientType::Android)
# YoutubeAPI::player(video_id: "dQw4w9WgXcQ", client_config: conf_2)
#
# # Proxy request through russian proxies
# conf_3 = ClientConfig.new(proxy_region: "RU")
# YoutubeAPI::next({video_id: "dQw4w9WgXcQ"}, client_config: conf_3)
# ```
#
struct ClientConfig
# Type of client to emulate (Web or Android).
# See `enum ClientType` and `HARDCODED_CLIENTS`.
property client_type : ClientType
# Region to provide to youtube, e.g to alter search results
# (this is passed as the `gl` parmeter).
property region : String | Nil
# ISO code of country where the proxy is located.
# Used in case of geo-restricted videos.
property proxy_region : String | Nil
# Initialization function
def initialize(
*,
@client_type = ClientType::Web,
@region = "US",
@proxy_region = nil
)
end
# Getter functions that provides easy access to hardcoded clients
# parameters (name/version strings and related API key)
def name : String
HARDCODED_CLIENTS[@client_type][:name]
end
# :ditto:
def version : String
HARDCODED_CLIENTS[@client_type][:version]
end
# :ditto:
def api_key : String
HARDCODED_CLIENTS[@client_type][:api_key]
end
end
# Default client config, used if nothing is passed
DEFAULT_CLIENT_CONFIG = ClientConfig.new
####################################################################
# make_context(client_config)
# #
# Return, as a Hash, the "context" data required to request the # Return, as a Hash, the "context" data required to request the
# youtube API endpoints. # youtube API endpoints.
# #
private def make_context(region : String | Nil) : Hash private def make_context(client_config : ClientConfig) : Hash
return { return {
"client" => { "client" => {
"hl" => "en", "hl" => "en",
"gl" => region || "US", # Can't be empty! "gl" => client_config.region || "US", # Can't be empty!
"clientName" => HARDCODED_CLIENTS[0][:name], "clientName" => client_config.name,
"clientVersion" => HARDCODED_CLIENTS[0][:version], "clientVersion" => client_config.version,
}, },
} }
end end
#################################################################### ####################################################################
# browse(continuation) # browse(continuation, client_config?)
# browse(browse_id, params) # browse(browse_id, params, client_config?)
# browse(browse_id, params, region)
# #
# Requests the youtubei/v1/browse endpoint with the required headers # Requests the youtubei/v1/browse endpoint with the required headers
# and POST data in order to get a JSON reply in english that can # and POST data in order to get a JSON reply in english that can
# be easily parsed. # be easily parsed.
# #
# A region can be provided, default is US. # Both forms can take an optional ClientConfig parameter (see
# `struct ClientConfig` above for more details).
# #
# The requested data can either be: # The requested data can either be:
# #
@ -61,22 +125,27 @@ module YoutubeAPI
# #
# - A playlist ID (parameters MUST be an empty string) # - A playlist ID (parameters MUST be an empty string)
# #
def browse(continuation : String) def browse(continuation : String, client_config : ClientConfig | Nil = nil)
# JSON Request data, required by the API # JSON Request data, required by the API
data = { data = {
"context" => self.make_context("US"), "context" => self.make_context(client_config),
"continuation" => continuation, "continuation" => continuation,
} }
return self._post_json("/youtubei/v1/browse", data) return self._post_json("/youtubei/v1/browse", data, client_config)
end end
# :ditto: # :ditto:
def browse(browse_id : String, *, params : String, region : String = "US") def browse(
browse_id : String,
*, # Force the following paramters to be passed by name
params : String,
client_config : ClientConfig | Nil = nil
)
# JSON Request data, required by the API # JSON Request data, required by the API
data = { data = {
"browseId" => browse_id, "browseId" => browse_id,
"context" => self.make_context(region), "context" => self.make_context(client_config),
} }
# Append the additionnal parameters if those were provided # Append the additionnal parameters if those were provided
@ -85,19 +154,20 @@ module YoutubeAPI
data["params"] = params data["params"] = params
end end
return self._post_json("/youtubei/v1/browse", data) return self._post_json("/youtubei/v1/browse", data, client_config)
end end
#################################################################### ####################################################################
# next(continuation) # next(continuation, client_config?)
# next(continuation, region) # next(data, client_config?)
# next(data)
# next(data, region)
# #
# Requests the youtubei/v1/next endpoint with the required headers # Requests the youtubei/v1/next endpoint with the required headers
# and POST data in order to get a JSON reply in english that can # and POST data in order to get a JSON reply in english that can
# be easily parsed. # be easily parsed.
# #
# Both forms can take an optional ClientConfig parameter (see
# `struct ClientConfig` above for more details).
#
# The requested data can be: # The requested data can be:
# #
# - A continuation token (ctoken). Depending on this token's # - A continuation token (ctoken). Depending on this token's
@ -123,42 +193,33 @@ module YoutubeAPI
# }) # })
# ``` # ```
# #
# Both forms can take an optional region parameter, that ay def next(continuation : String, *, client_config : ClientConfig | Nil = nil)
# impact the data returned by youtube (e.g translation of some
# video titles). E.g:
#
# ```
# YoutubeAPI::next("ABCDEFGH_abcdefgh==", region: "FR")
# YoutubeAPI::next({"videoId": "dQw4w9WgXcQ"}, region: "DE")
# ```
#
def next(continuation : String, *, region : String | Nil = nil)
# JSON Request data, required by the API # JSON Request data, required by the API
data = { data = {
"context" => self.make_context(region), "context" => self.make_context(client_config),
"continuation" => continuation, "continuation" => continuation,
} }
return self._post_json("/youtubei/v1/next", data) return self._post_json("/youtubei/v1/next", data, client_config)
end end
# :ditto: # :ditto:
def next(data : Hash, *, region : String | Nil = nil) def next(data : Hash, *, client_config : ClientConfig | Nil = nil)
# JSON Request data, required by the API # JSON Request data, required by the API
data.merge!({ data.merge!({
"context" => self.make_context(region), "context" => self.make_context(client_config),
}) })
return self._post_json("/youtubei/v1/next", data) return self._post_json("/youtubei/v1/next", data, client_config)
end end
# Allow a NamedTuple to be passed, too. # Allow a NamedTuple to be passed, too.
def next(data : NamedTuple, *, region : String | Nil = nil) def next(data : NamedTuple, *, client_config : ClientConfig | Nil = nil)
return self.next(data.to_h, region: region) return self.next(data.to_h, client_config: client_config)
end end
#################################################################### ####################################################################
# search(search_query, params, region) # search(search_query, params, client_config?)
# #
# Requests the youtubei/v1/search endpoint with the required headers # Requests the youtubei/v1/search endpoint with the required headers
# and POST data in order to get a JSON reply. As the search results # and POST data in order to get a JSON reply. As the search results
@ -168,19 +229,26 @@ module YoutubeAPI
# The requested data is a search string, with some additional # The requested data is a search string, with some additional
# paramters, formatted as a base64 string. # paramters, formatted as a base64 string.
# #
def search(search_query : String, params : String, region = nil) # An optional ClientConfig parameter can be passed, too (see
# `struct ClientConfig` above for more details).
#
def search(
search_query : String,
params : String,
client_config : ClientConfig | Nil = nil
)
# JSON Request data, required by the API # JSON Request data, required by the API
data = { data = {
"query" => search_query, "query" => search_query,
"context" => self.make_context(region), "context" => self.make_context(client_config),
"params" => params, "params" => params,
} }
return self._post_json("/youtubei/v1/search", data) return self._post_json("/youtubei/v1/search", data, client_config)
end end
#################################################################### ####################################################################
# _post_json(endpoint, data) # _post_json(endpoint, data, client_config?)
# #
# Internal function that does the actual request to youtube servers # Internal function that does the actual request to youtube servers
# and handles errors. # and handles errors.
@ -188,10 +256,17 @@ module YoutubeAPI
# The requested data is an endpoint (URL without the domain part) # The requested data is an endpoint (URL without the domain part)
# and the data as a Hash object. # and the data as a Hash object.
# #
def _post_json(endpoint, data) : Hash(String, JSON::Any) def _post_json(
endpoint : String,
data : Hash,
client_config : ClientConfig | Nil
) : Hash(String, JSON::Any)
# Use the default client config if nil is passed
client_config ||= DEFAULT_CLIENT_CONFIG
# Send the POST request and parse result # Send the POST request and parse result
response = YT_POOL.client &.post( response = YT_POOL.client &.post(
"#{endpoint}?key=#{HARDCODED_CLIENTS[0][:api_key]}", "#{endpoint}?key=#{client_config.api_key}",
headers: HTTP::Headers{"content-type" => "application/json; charset=UTF-8"}, headers: HTTP::Headers{"content-type" => "application/json; charset=UTF-8"},
body: data.to_json body: data.to_json
) )

View file

@ -263,7 +263,8 @@ end
def search(query, search_params = produce_search_params(content_type: "all"), region = nil) def search(query, search_params = produce_search_params(content_type: "all"), region = nil)
return 0, [] of SearchItem if query.empty? return 0, [] of SearchItem if query.empty?
initial_data = YoutubeAPI.search(query, search_params, region) client_config = YoutubeAPI::ClientConfig.new(region: region)
initial_data = YoutubeAPI.search(query, search_params, client_config: client_config)
items = extract_items(initial_data) items = extract_items(initial_data)
return items.size, items return items.size, items

View file

@ -14,7 +14,8 @@ def fetch_trending(trending_type, region, locale)
params = "" params = ""
end end
initial_data = YoutubeAPI.browse("FEtrending", params: params, region: region) client_config = YoutubeAPI::ClientConfig.new(region: region)
initial_data = YoutubeAPI.browse("FEtrending", params: params, client_config: client_config)
trending = extract_videos(initial_data) trending = extract_videos(initial_data)
return {trending, plid} return {trending, plid}