2019-03-29 16:30:02 -05:00
|
|
|
struct SearchVideo
|
2019-09-07 17:45:37 +02:00
|
|
|
def to_xml(host_url, auto_generated, query_params, xml : XML::Builder)
|
|
|
|
query_params["v"] = self.id
|
|
|
|
|
2019-06-07 12:39:12 -05:00
|
|
|
xml.element("entry") do
|
|
|
|
xml.element("id") { xml.text "yt:video:#{self.id}" }
|
|
|
|
xml.element("yt:videoId") { xml.text self.id }
|
|
|
|
xml.element("yt:channelId") { xml.text self.ucid }
|
|
|
|
xml.element("title") { xml.text self.title }
|
2019-09-07 17:45:37 +02:00
|
|
|
xml.element("link", rel: "alternate", href: "#{host_url}/watch?#{query_params}")
|
2019-06-07 12:39:12 -05:00
|
|
|
|
|
|
|
xml.element("author") do
|
|
|
|
if auto_generated
|
|
|
|
xml.element("name") { xml.text self.author }
|
|
|
|
xml.element("uri") { xml.text "#{host_url}/channel/#{self.ucid}" }
|
|
|
|
else
|
|
|
|
xml.element("name") { xml.text author }
|
|
|
|
xml.element("uri") { xml.text "#{host_url}/channel/#{ucid}" }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
xml.element("content", type: "xhtml") do
|
|
|
|
xml.element("div", xmlns: "http://www.w3.org/1999/xhtml") do
|
2019-09-07 17:45:37 +02:00
|
|
|
xml.element("a", href: "#{host_url}/watch?#{query_params}") do
|
2019-06-07 12:39:12 -05:00
|
|
|
xml.element("img", src: "#{host_url}/vi/#{self.id}/mqdefault.jpg")
|
|
|
|
end
|
2019-10-04 12:49:58 -04:00
|
|
|
|
2019-11-03 08:53:16 -05:00
|
|
|
xml.element("p", style: "word-break:break-word;white-space:pre-wrap") { xml.text html_to_content(self.description_html) }
|
2019-06-07 12:39:12 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
xml.element("published") { xml.text self.published.to_s("%Y-%m-%dT%H:%M:%S%:z") }
|
|
|
|
|
|
|
|
xml.element("media:group") do
|
|
|
|
xml.element("media:title") { xml.text self.title }
|
|
|
|
xml.element("media:thumbnail", url: "#{host_url}/vi/#{self.id}/mqdefault.jpg",
|
|
|
|
width: "320", height: "180")
|
2019-06-08 15:08:27 -05:00
|
|
|
xml.element("media:description") { xml.text html_to_content(self.description_html) }
|
2019-06-07 12:39:12 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
xml.element("media:community") do
|
|
|
|
xml.element("media:statistics", views: self.views)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-09-07 17:45:37 +02:00
|
|
|
def to_xml(host_url, auto_generated, query_params, xml : XML::Builder | Nil = nil)
|
2019-06-07 12:39:12 -05:00
|
|
|
if xml
|
2019-09-07 17:45:37 +02:00
|
|
|
to_xml(host_url, auto_generated, query_params, xml)
|
2019-06-07 12:39:12 -05:00
|
|
|
else
|
|
|
|
XML.build do |json|
|
2019-09-07 17:45:37 +02:00
|
|
|
to_xml(host_url, auto_generated, query_params, xml)
|
2019-06-07 12:39:12 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-06-08 13:31:41 -05:00
|
|
|
def to_json(locale, config, kemal_config, json : JSON::Builder)
|
|
|
|
json.object do
|
|
|
|
json.field "type", "video"
|
|
|
|
json.field "title", self.title
|
|
|
|
json.field "videoId", self.id
|
|
|
|
|
|
|
|
json.field "author", self.author
|
|
|
|
json.field "authorId", self.ucid
|
|
|
|
json.field "authorUrl", "/channel/#{self.ucid}"
|
|
|
|
|
|
|
|
json.field "videoThumbnails" do
|
|
|
|
generate_thumbnails(json, self.id, config, kemal_config)
|
|
|
|
end
|
|
|
|
|
2019-06-08 15:08:27 -05:00
|
|
|
json.field "description", html_to_content(self.description_html)
|
2019-06-08 13:31:41 -05:00
|
|
|
json.field "descriptionHtml", self.description_html
|
|
|
|
|
|
|
|
json.field "viewCount", self.views
|
|
|
|
json.field "published", self.published.to_unix
|
|
|
|
json.field "publishedText", translate(locale, "`x` ago", recode_date(self.published, locale))
|
|
|
|
json.field "lengthSeconds", self.length_seconds
|
|
|
|
json.field "liveNow", self.live_now
|
|
|
|
json.field "paid", self.paid
|
|
|
|
json.field "premium", self.premium
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def to_json(locale, config, kemal_config, json : JSON::Builder | Nil = nil)
|
|
|
|
if json
|
|
|
|
to_json(locale, config, kemal_config, json)
|
|
|
|
else
|
|
|
|
JSON.build do |json|
|
|
|
|
to_json(locale, config, kemal_config, json)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-04-03 11:35:58 -05:00
|
|
|
db_mapping({
|
2019-03-22 12:24:47 -05:00
|
|
|
title: String,
|
|
|
|
id: String,
|
|
|
|
author: String,
|
|
|
|
ucid: String,
|
|
|
|
published: Time,
|
|
|
|
views: Int64,
|
|
|
|
description_html: String,
|
|
|
|
length_seconds: Int32,
|
|
|
|
live_now: Bool,
|
|
|
|
paid: Bool,
|
|
|
|
premium: Bool,
|
|
|
|
premiere_timestamp: Time?,
|
2018-08-04 23:07:38 -05:00
|
|
|
})
|
|
|
|
end
|
|
|
|
|
2019-03-29 16:30:02 -05:00
|
|
|
struct SearchPlaylistVideo
|
2019-04-03 11:35:58 -05:00
|
|
|
db_mapping({
|
2018-09-20 09:36:09 -05:00
|
|
|
title: String,
|
|
|
|
id: String,
|
|
|
|
length_seconds: Int32,
|
|
|
|
})
|
|
|
|
end
|
|
|
|
|
2019-03-29 16:30:02 -05:00
|
|
|
struct SearchPlaylist
|
2019-06-08 13:31:41 -05:00
|
|
|
def to_json(locale, config, kemal_config, json : JSON::Builder)
|
|
|
|
json.object do
|
|
|
|
json.field "type", "playlist"
|
|
|
|
json.field "title", self.title
|
|
|
|
json.field "playlistId", self.id
|
2019-08-21 19:08:11 -05:00
|
|
|
json.field "playlistThumbnail", self.thumbnail
|
2019-06-08 13:31:41 -05:00
|
|
|
|
|
|
|
json.field "author", self.author
|
|
|
|
json.field "authorId", self.ucid
|
|
|
|
json.field "authorUrl", "/channel/#{self.ucid}"
|
|
|
|
|
|
|
|
json.field "videoCount", self.video_count
|
|
|
|
json.field "videos" do
|
|
|
|
json.array do
|
|
|
|
self.videos.each do |video|
|
|
|
|
json.object do
|
|
|
|
json.field "title", video.title
|
|
|
|
json.field "videoId", video.id
|
|
|
|
json.field "lengthSeconds", video.length_seconds
|
|
|
|
|
|
|
|
json.field "videoThumbnails" do
|
|
|
|
generate_thumbnails(json, video.id, config, Kemal.config)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def to_json(locale, config, kemal_config, json : JSON::Builder | Nil = nil)
|
|
|
|
if json
|
|
|
|
to_json(locale, config, kemal_config, json)
|
|
|
|
else
|
|
|
|
JSON.build do |json|
|
|
|
|
to_json(locale, config, kemal_config, json)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-04-03 11:35:58 -05:00
|
|
|
db_mapping({
|
2019-08-16 15:46:37 -05:00
|
|
|
title: String,
|
|
|
|
id: String,
|
|
|
|
author: String,
|
|
|
|
ucid: String,
|
|
|
|
video_count: Int32,
|
|
|
|
videos: Array(SearchPlaylistVideo),
|
|
|
|
thumbnail: String?,
|
2018-09-20 09:36:09 -05:00
|
|
|
})
|
|
|
|
end
|
|
|
|
|
2019-03-29 16:30:02 -05:00
|
|
|
struct SearchChannel
|
2019-06-08 13:31:41 -05:00
|
|
|
def to_json(locale, config, kemal_config, json : JSON::Builder)
|
|
|
|
json.object do
|
|
|
|
json.field "type", "channel"
|
|
|
|
json.field "author", self.author
|
|
|
|
json.field "authorId", self.ucid
|
|
|
|
json.field "authorUrl", "/channel/#{self.ucid}"
|
|
|
|
|
|
|
|
json.field "authorThumbnails" do
|
|
|
|
json.array do
|
|
|
|
qualities = {32, 48, 76, 100, 176, 512}
|
|
|
|
|
|
|
|
qualities.each do |quality|
|
|
|
|
json.object do
|
2019-07-31 19:16:09 -05:00
|
|
|
json.field "url", self.author_thumbnail.gsub(/=\d+/, "=s#{quality}")
|
2019-06-08 13:31:41 -05:00
|
|
|
json.field "width", quality
|
|
|
|
json.field "height", quality
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-09-12 13:11:21 -04:00
|
|
|
json.field "autoGenerated", self.auto_generated
|
2019-06-08 13:31:41 -05:00
|
|
|
json.field "subCount", self.subscriber_count
|
|
|
|
json.field "videoCount", self.video_count
|
2019-09-12 13:11:21 -04:00
|
|
|
|
2019-06-08 15:08:27 -05:00
|
|
|
json.field "description", html_to_content(self.description_html)
|
2019-06-08 13:31:41 -05:00
|
|
|
json.field "descriptionHtml", self.description_html
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def to_json(locale, config, kemal_config, json : JSON::Builder | Nil = nil)
|
|
|
|
if json
|
|
|
|
to_json(locale, config, kemal_config, json)
|
|
|
|
else
|
|
|
|
JSON.build do |json|
|
|
|
|
to_json(locale, config, kemal_config, json)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-04-03 11:35:58 -05:00
|
|
|
db_mapping({
|
2018-09-20 09:36:09 -05:00
|
|
|
author: String,
|
|
|
|
ucid: String,
|
|
|
|
author_thumbnail: String,
|
|
|
|
subscriber_count: Int32,
|
|
|
|
video_count: Int32,
|
|
|
|
description_html: String,
|
2019-09-12 13:11:21 -04:00
|
|
|
auto_generated: Bool,
|
2018-09-20 09:36:09 -05:00
|
|
|
})
|
|
|
|
end
|
|
|
|
|
|
|
|
alias SearchItem = SearchVideo | SearchChannel | SearchPlaylist
|
|
|
|
|
2018-09-13 17:47:31 -05:00
|
|
|
def channel_search(query, page, channel)
|
2019-10-25 12:58:16 -04:00
|
|
|
response = YT_POOL.client &.get("/channel/#{channel}?disable_polymer=1&hl=en&gl=US")
|
2018-09-13 17:47:31 -05:00
|
|
|
document = XML.parse_html(response.body)
|
|
|
|
canonical = document.xpath_node(%q(//link[@rel="canonical"]))
|
|
|
|
|
|
|
|
if !canonical
|
2019-10-25 12:58:16 -04:00
|
|
|
response = YT_POOL.client &.get("/c/#{channel}?disable_polymer=1&hl=en&gl=US")
|
2018-09-13 17:47:31 -05:00
|
|
|
document = XML.parse_html(response.body)
|
|
|
|
canonical = document.xpath_node(%q(//link[@rel="canonical"]))
|
|
|
|
end
|
|
|
|
|
2019-04-22 15:39:57 -05:00
|
|
|
if !canonical
|
2019-10-25 12:58:16 -04:00
|
|
|
response = YT_POOL.client &.get("/user/#{channel}?disable_polymer=1&hl=en&gl=US")
|
2019-04-22 15:39:57 -05:00
|
|
|
document = XML.parse_html(response.body)
|
|
|
|
canonical = document.xpath_node(%q(//link[@rel="canonical"]))
|
|
|
|
end
|
|
|
|
|
2018-09-13 17:47:31 -05:00
|
|
|
if !canonical
|
2018-09-20 09:36:09 -05:00
|
|
|
return 0, [] of SearchItem
|
2018-09-13 17:47:31 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
ucid = canonical["href"].split("/")[-1]
|
|
|
|
|
|
|
|
url = produce_channel_search_url(ucid, query, page)
|
2019-10-25 12:58:16 -04:00
|
|
|
response = YT_POOL.client &.get(url)
|
2018-09-13 17:47:31 -05:00
|
|
|
json = JSON.parse(response.body)
|
|
|
|
|
|
|
|
if json["content_html"]? && !json["content_html"].as_s.empty?
|
|
|
|
document = XML.parse_html(json["content_html"].as_s)
|
|
|
|
nodeset = document.xpath_nodes(%q(//li[contains(@class, "feed-item-container")]))
|
|
|
|
|
|
|
|
count = nodeset.size
|
2018-09-20 09:36:09 -05:00
|
|
|
items = extract_items(nodeset)
|
2018-09-13 17:47:31 -05:00
|
|
|
else
|
|
|
|
count = 0
|
2018-09-20 09:36:09 -05:00
|
|
|
items = [] of SearchItem
|
2018-09-13 17:47:31 -05:00
|
|
|
end
|
|
|
|
|
2018-09-20 09:36:09 -05:00
|
|
|
return count, items
|
2018-09-13 17:47:31 -05:00
|
|
|
end
|
|
|
|
|
2019-06-28 21:17:56 -05:00
|
|
|
def search(query, page = 1, search_params = produce_search_params(content_type: "all"), region = nil)
|
2018-08-27 15:23:25 -05:00
|
|
|
if query.empty?
|
2018-09-20 09:36:09 -05:00
|
|
|
return {0, [] of SearchItem}
|
2018-08-27 15:23:25 -05:00
|
|
|
end
|
|
|
|
|
2019-10-25 12:58:16 -04:00
|
|
|
html = YT_POOL.client(region, &.get("/results?q=#{URI.encode_www_form(query)}&page=#{page}&sp=#{search_params}&hl=en&disable_polymer=1").body)
|
2018-08-04 23:07:38 -05:00
|
|
|
if html.empty?
|
2018-09-20 09:36:09 -05:00
|
|
|
return {0, [] of SearchItem}
|
2018-08-04 23:07:38 -05:00
|
|
|
end
|
|
|
|
|
2018-08-04 15:30:44 -05:00
|
|
|
html = XML.parse_html(html)
|
2018-08-10 09:44:19 -05:00
|
|
|
nodeset = html.xpath_nodes(%q(//ol[@class="item-section"]/li))
|
2018-09-20 09:36:09 -05:00
|
|
|
items = extract_items(nodeset)
|
2018-08-04 15:30:44 -05:00
|
|
|
|
2018-09-20 09:36:09 -05:00
|
|
|
return {nodeset.size, items}
|
2018-08-04 15:30:44 -05:00
|
|
|
end
|
2018-08-04 17:12:58 -05:00
|
|
|
|
2018-09-17 16:38:18 -05:00
|
|
|
def produce_search_params(sort : String = "relevance", date : String = "", content_type : String = "",
|
|
|
|
duration : String = "", features : Array(String) = [] of String)
|
2019-10-27 13:50:42 -04:00
|
|
|
object = {
|
|
|
|
"1:varint" => 0_i64,
|
|
|
|
"2:embedded" => {} of String => Int64,
|
|
|
|
}
|
|
|
|
|
|
|
|
case sort
|
2019-08-27 09:35:15 -05:00
|
|
|
when "relevance"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["1:varint"] = 0_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "rating"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["1:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "upload_date", "date"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["1:varint"] = 2_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "view_count", "views"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["1:varint"] = 3_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
else
|
|
|
|
raise "No sort #{sort}"
|
|
|
|
end
|
2018-08-04 17:12:58 -05:00
|
|
|
|
2019-10-27 13:50:42 -04:00
|
|
|
case date
|
2019-08-27 09:35:15 -05:00
|
|
|
when "hour"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["1:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "today"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["1:varint"] = 2_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "week"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["1:varint"] = 3_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "month"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["1:varint"] = 4_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "year"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["1:varint"] = 5_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
end
|
2018-08-04 17:12:58 -05:00
|
|
|
|
2019-10-27 13:50:42 -04:00
|
|
|
case content_type
|
2019-08-27 09:35:15 -05:00
|
|
|
when "video"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["2:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "channel"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["2:varint"] = 2_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "playlist"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["2:varint"] = 3_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "movie"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["2:varint"] = 4_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "show"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["2:varint"] = 5_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "all"
|
2019-10-27 13:50:42 -04:00
|
|
|
#
|
2019-08-27 09:35:15 -05:00
|
|
|
else
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["2:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
end
|
2018-08-04 17:12:58 -05:00
|
|
|
|
2019-10-27 13:50:42 -04:00
|
|
|
case duration
|
2019-08-27 09:35:15 -05:00
|
|
|
when "short"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["3:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "long"
|
2019-11-08 09:29:33 -05:00
|
|
|
object["2:embedded"].as(Hash)["3:varint"] = 2_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
end
|
2018-08-04 17:12:58 -05:00
|
|
|
|
|
|
|
features.each do |feature|
|
2019-10-27 13:50:42 -04:00
|
|
|
case feature
|
2019-08-27 09:35:15 -05:00
|
|
|
when "hd"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["4:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "subtitles"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["5:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "creative_commons", "cc"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["6:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "3d"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["7:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "live", "livestream"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["8:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "purchased"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["9:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "4k"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["14:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "360"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["15:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "location"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["23:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
when "hdr"
|
2019-10-27 13:50:42 -04:00
|
|
|
object["2:embedded"].as(Hash)["25:varint"] = 1_i64
|
2019-08-27 09:35:15 -05:00
|
|
|
end
|
2018-08-04 17:12:58 -05:00
|
|
|
end
|
|
|
|
|
2019-10-28 06:17:39 -04:00
|
|
|
if object["2:embedded"].as(Hash).empty?
|
|
|
|
object.delete("2:embedded")
|
|
|
|
end
|
|
|
|
|
2019-10-27 13:50:42 -04:00
|
|
|
params = object.try { |i| Protodec::Any.cast_json(object) }
|
|
|
|
.try { |i| Protodec::Any.from_json(i) }
|
2019-10-28 06:17:39 -04:00
|
|
|
.try { |i| Base64.urlsafe_encode(i) }
|
|
|
|
.try { |i| URI.encode_www_form(i) }
|
2018-08-04 17:12:58 -05:00
|
|
|
|
2019-10-27 13:50:42 -04:00
|
|
|
return params
|
2018-08-04 17:12:58 -05:00
|
|
|
end
|
2018-09-13 17:47:31 -05:00
|
|
|
|
|
|
|
def produce_channel_search_url(ucid, query, page)
|
2019-10-27 13:50:42 -04:00
|
|
|
object = {
|
|
|
|
"80226972:embedded" => {
|
|
|
|
"2:string" => ucid,
|
|
|
|
"3:base64" => {
|
|
|
|
"2:string" => "search",
|
|
|
|
"6:varint" => 2_i64,
|
|
|
|
"7:varint" => 1_i64,
|
|
|
|
"12:varint" => 1_i64,
|
|
|
|
"13:string" => "",
|
|
|
|
"23:varint" => 0_i64,
|
|
|
|
"15:string" => "#{page}",
|
|
|
|
},
|
|
|
|
"11:string" => query,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
continuation = object.try { |i| Protodec::Any.cast_json(object) }
|
|
|
|
.try { |i| Protodec::Any.from_json(i) }
|
|
|
|
.try { |i| Base64.urlsafe_encode(i) }
|
|
|
|
.try { |i| URI.encode_www_form(i) }
|
|
|
|
|
|
|
|
return "/browse_ajax?continuation=#{continuation}&gl=US&hl=en"
|
2018-09-13 17:47:31 -05:00
|
|
|
end
|
2019-08-05 18:49:13 -05:00
|
|
|
|
|
|
|
def process_search_query(query, page, user, region)
|
|
|
|
if user
|
|
|
|
user = user.as(User)
|
|
|
|
view_name = "subscriptions_#{sha256(user.email)}"
|
|
|
|
end
|
|
|
|
|
|
|
|
channel = nil
|
|
|
|
content_type = "all"
|
|
|
|
date = ""
|
|
|
|
duration = ""
|
|
|
|
features = [] of String
|
|
|
|
sort = "relevance"
|
|
|
|
subscriptions = nil
|
|
|
|
|
|
|
|
operators = query.split(" ").select { |a| a.match(/\w+:[\w,]+/) }
|
|
|
|
operators.each do |operator|
|
|
|
|
key, value = operator.downcase.split(":")
|
|
|
|
|
|
|
|
case key
|
|
|
|
when "channel", "user"
|
|
|
|
channel = operator.split(":")[-1]
|
|
|
|
when "content_type", "type"
|
|
|
|
content_type = value
|
|
|
|
when "date"
|
|
|
|
date = value
|
|
|
|
when "duration"
|
|
|
|
duration = value
|
|
|
|
when "feature", "features"
|
|
|
|
features = value.split(",")
|
|
|
|
when "sort"
|
|
|
|
sort = value
|
|
|
|
when "subscriptions"
|
|
|
|
subscriptions = value == "true"
|
|
|
|
else
|
|
|
|
operators.delete(operator)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
search_query = (query.split(" ") - operators).join(" ")
|
|
|
|
|
|
|
|
if channel
|
|
|
|
count, items = channel_search(search_query, page, channel)
|
|
|
|
elsif subscriptions
|
|
|
|
if view_name
|
|
|
|
items = PG_DB.query_all("SELECT id,title,published,updated,ucid,author,length_seconds FROM (
|
|
|
|
SELECT *,
|
|
|
|
to_tsvector(#{view_name}.title) ||
|
|
|
|
to_tsvector(#{view_name}.author)
|
|
|
|
as document
|
|
|
|
FROM #{view_name}
|
|
|
|
) v_search WHERE v_search.document @@ plainto_tsquery($1) LIMIT 20 OFFSET $2;", search_query, (page - 1) * 20, as: ChannelVideo)
|
|
|
|
count = items.size
|
|
|
|
else
|
|
|
|
items = [] of ChannelVideo
|
|
|
|
count = 0
|
|
|
|
end
|
|
|
|
else
|
|
|
|
search_params = produce_search_params(sort: sort, date: date, content_type: content_type,
|
|
|
|
duration: duration, features: features)
|
|
|
|
|
|
|
|
count, items = search(search_query, page, search_params, region).as(Tuple)
|
|
|
|
end
|
|
|
|
|
|
|
|
{search_query, count, items}
|
|
|
|
end
|