mirror of
				https://gitea.invidious.io/iv-org/invidious.git
				synced 2024-08-15 00:53:41 +00:00 
			
		
		
		
	Split helpers.cr into multiple files
This commit is contained in:
		
							parent
							
								
									0685d3c7c2
								
							
						
					
					
						commit
						90d661e4a7
					
				
					 12 changed files with 2798 additions and 2779 deletions
				
			
		
							
								
								
									
										2906
									
								
								src/invidious.cr
									
										
									
									
									
								
							
							
						
						
									
										2906
									
								
								src/invidious.cr
									
										
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										132
									
								
								src/invidious/channels.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										132
									
								
								src/invidious/channels.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,132 @@ | ||||||
|  | class InvidiousChannel | ||||||
|  |   add_mapping({ | ||||||
|  |     id:      String, | ||||||
|  |     author:  String, | ||||||
|  |     updated: Time, | ||||||
|  |   }) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | class ChannelVideo | ||||||
|  |   add_mapping({ | ||||||
|  |     id:        String, | ||||||
|  |     title:     String, | ||||||
|  |     published: Time, | ||||||
|  |     updated:   Time, | ||||||
|  |     ucid:      String, | ||||||
|  |     author:    String, | ||||||
|  |   }) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def get_channel(id, client, db, refresh = true, pull_all_videos = true) | ||||||
|  |   if db.query_one?("SELECT EXISTS (SELECT true FROM channels WHERE id = $1)", id, as: Bool) | ||||||
|  |     channel = db.query_one("SELECT * FROM channels WHERE id = $1", id, as: InvidiousChannel) | ||||||
|  | 
 | ||||||
|  |     if refresh && Time.now - channel.updated > 10.minutes | ||||||
|  |       channel = fetch_channel(id, client, db, pull_all_videos) | ||||||
|  |       channel_array = channel.to_a | ||||||
|  |       args = arg_array(channel_array) | ||||||
|  | 
 | ||||||
|  |       db.exec("INSERT INTO channels VALUES (#{args}) \ | ||||||
|  |         ON CONFLICT (id) DO UPDATE SET updated = $3", channel_array) | ||||||
|  |     end | ||||||
|  |   else | ||||||
|  |     channel = fetch_channel(id, client, db, pull_all_videos) | ||||||
|  |     args = arg_array(channel.to_a) | ||||||
|  |     db.exec("INSERT INTO channels VALUES (#{args})", channel.to_a) | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return channel | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def fetch_channel(ucid, client, db, pull_all_videos = true) | ||||||
|  |   rss = client.get("/feeds/videos.xml?channel_id=#{ucid}").body | ||||||
|  |   rss = XML.parse_html(rss) | ||||||
|  | 
 | ||||||
|  |   author = rss.xpath_node(%q(//feed/title)) | ||||||
|  |   if !author | ||||||
|  |     raise "Deleted or invalid channel" | ||||||
|  |   end | ||||||
|  |   author = author.content | ||||||
|  | 
 | ||||||
|  |   if !pull_all_videos | ||||||
|  |     rss.xpath_nodes("//feed/entry").each do |entry| | ||||||
|  |       video_id = entry.xpath_node("videoid").not_nil!.content | ||||||
|  |       title = entry.xpath_node("title").not_nil!.content | ||||||
|  |       published = Time.parse(entry.xpath_node("published").not_nil!.content, "%FT%X%z", Time::Location.local) | ||||||
|  |       updated = Time.parse(entry.xpath_node("updated").not_nil!.content, "%FT%X%z", Time::Location.local) | ||||||
|  |       author = entry.xpath_node("author/name").not_nil!.content | ||||||
|  |       ucid = entry.xpath_node("channelid").not_nil!.content | ||||||
|  | 
 | ||||||
|  |       video = ChannelVideo.new(video_id, title, published, Time.now, ucid, author) | ||||||
|  | 
 | ||||||
|  |       db.exec("UPDATE users SET notifications = notifications || $1 \ | ||||||
|  |         WHERE updated < $2 AND $3 = ANY(subscriptions) AND $1 <> ALL(notifications)", video.id, video.published, ucid) | ||||||
|  | 
 | ||||||
|  |       video_array = video.to_a | ||||||
|  |       args = arg_array(video_array) | ||||||
|  |       db.exec("INSERT INTO channel_videos VALUES (#{args}) \ | ||||||
|  |         ON CONFLICT (id) DO UPDATE SET title = $2, published = $3, \ | ||||||
|  |         updated = $4, ucid = $5, author = $6", video_array) | ||||||
|  |     end | ||||||
|  |   else | ||||||
|  |     videos = [] of ChannelVideo | ||||||
|  |     page = 1 | ||||||
|  | 
 | ||||||
|  |     loop do | ||||||
|  |       url = produce_videos_url(ucid, page) | ||||||
|  |       response = client.get(url) | ||||||
|  | 
 | ||||||
|  |       json = JSON.parse(response.body) | ||||||
|  |       content_html = json["content_html"].as_s | ||||||
|  |       if content_html.empty? | ||||||
|  |         # If we don't get anything, move on | ||||||
|  |         break | ||||||
|  |       end | ||||||
|  |       document = XML.parse_html(content_html) | ||||||
|  | 
 | ||||||
|  |       document.xpath_nodes(%q(//li[contains(@class, "feed-item-container")])).each do |item| | ||||||
|  |         anchor = item.xpath_node(%q(.//h3[contains(@class,"yt-lockup-title")]/a)) | ||||||
|  |         if !anchor | ||||||
|  |           raise "could not find anchor" | ||||||
|  |         end | ||||||
|  | 
 | ||||||
|  |         title = anchor.content.strip | ||||||
|  |         video_id = anchor["href"].lchop("/watch?v=") | ||||||
|  | 
 | ||||||
|  |         published = item.xpath_node(%q(.//div[@class="yt-lockup-meta"]/ul/li[1])) | ||||||
|  |         if !published | ||||||
|  |           # This happens on Youtube red videos, here we just skip them | ||||||
|  |           next | ||||||
|  |         end | ||||||
|  |         published = published.content | ||||||
|  |         published = decode_date(published) | ||||||
|  | 
 | ||||||
|  |         videos << ChannelVideo.new(video_id, title, published, Time.now, ucid, author) | ||||||
|  |       end | ||||||
|  | 
 | ||||||
|  |       if document.xpath_nodes(%q(//li[contains(@class, "channels-content-item")])).size < 30 | ||||||
|  |         break | ||||||
|  |       end | ||||||
|  | 
 | ||||||
|  |       page += 1 | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     video_ids = [] of String | ||||||
|  |     videos.each do |video| | ||||||
|  |       db.exec("UPDATE users SET notifications = notifications || $1 \ | ||||||
|  |         WHERE updated < $2 AND $3 = ANY(subscriptions) AND $1 <> ALL(notifications)", video.id, video.published, ucid) | ||||||
|  |       video_ids << video.id | ||||||
|  | 
 | ||||||
|  |       video_array = video.to_a | ||||||
|  |       args = arg_array(video_array) | ||||||
|  |       db.exec("INSERT INTO channel_videos VALUES (#{args}) ON CONFLICT (id) DO NOTHING", video_array) | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     # When a video is deleted from a channel, we find and remove it here | ||||||
|  |     db.exec("DELETE FROM channel_videos * WHERE NOT id = ANY ('{#{video_ids.map { |a| %("#{a}") }.join(",")}}') AND ucid = $1", ucid) | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   channel = InvidiousChannel.new(ucid, author, Time.now) | ||||||
|  | 
 | ||||||
|  |   return channel | ||||||
|  | end | ||||||
							
								
								
									
										247
									
								
								src/invidious/comments.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										247
									
								
								src/invidious/comments.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,247 @@ | ||||||
|  | class RedditThing | ||||||
|  |   JSON.mapping({ | ||||||
|  |     kind: String, | ||||||
|  |     data: RedditComment | RedditLink | RedditMore | RedditListing, | ||||||
|  |   }) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | class RedditComment | ||||||
|  |   JSON.mapping({ | ||||||
|  |     author:    String, | ||||||
|  |     body_html: String, | ||||||
|  |     replies:   RedditThing | String, | ||||||
|  |     score:     Int32, | ||||||
|  |     depth:     Int32, | ||||||
|  |   }) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | class RedditLink | ||||||
|  |   JSON.mapping({ | ||||||
|  |     author:       String, | ||||||
|  |     score:        Int32, | ||||||
|  |     subreddit:    String, | ||||||
|  |     num_comments: Int32, | ||||||
|  |     id:           String, | ||||||
|  |     permalink:    String, | ||||||
|  |     title:        String, | ||||||
|  |   }) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | class RedditMore | ||||||
|  |   JSON.mapping({ | ||||||
|  |     children: Array(String), | ||||||
|  |     count:    Int32, | ||||||
|  |     depth:    Int32, | ||||||
|  |   }) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | class RedditListing | ||||||
|  |   JSON.mapping({ | ||||||
|  |     children: Array(RedditThing), | ||||||
|  |     modhash:  String, | ||||||
|  |   }) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def get_reddit_comments(id, client, headers) | ||||||
|  |   query = "(url:3D#{id}%20OR%20url:#{id})%20(site:youtube.com%20OR%20site:youtu.be)" | ||||||
|  |   search_results = client.get("/search.json?q=#{query}", headers) | ||||||
|  | 
 | ||||||
|  |   if search_results.status_code == 200 | ||||||
|  |     search_results = RedditThing.from_json(search_results.body) | ||||||
|  | 
 | ||||||
|  |     thread = search_results.data.as(RedditListing).children.sort_by { |child| child.data.as(RedditLink).score }[-1] | ||||||
|  |     thread = thread.data.as(RedditLink) | ||||||
|  | 
 | ||||||
|  |     result = client.get("/r/#{thread.subreddit}/comments/#{thread.id}.json?limit=100&sort=top", headers).body | ||||||
|  |     result = Array(RedditThing).from_json(result) | ||||||
|  |   elsif search_results.status_code == 302 | ||||||
|  |     result = client.get(search_results.headers["Location"], headers).body | ||||||
|  |     result = Array(RedditThing).from_json(result) | ||||||
|  | 
 | ||||||
|  |     thread = result[0].data.as(RedditListing).children[0].data.as(RedditLink) | ||||||
|  |   else | ||||||
|  |     raise "Got error code #{search_results.status_code}" | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   comments = result[1].data.as(RedditListing).children | ||||||
|  |   return comments, thread | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def template_youtube_comments(comments) | ||||||
|  |   html = "" | ||||||
|  | 
 | ||||||
|  |   root = comments["comments"].as_a | ||||||
|  |   root.each do |child| | ||||||
|  |     if child["replies"]? | ||||||
|  |       replies_html = <<-END_HTML | ||||||
|  |       <div id="replies" class="pure-g"> | ||||||
|  |         <div class="pure-u-md-1-24"></div> | ||||||
|  |         <div class="pure-u-md-23-24"> | ||||||
|  |           <p> | ||||||
|  |             <a href="javascript:void(0)" data-continuation="#{child["replies"]["continuation"]}" | ||||||
|  |               onclick="load_comments(this)">View #{child["replies"]["replyCount"]} replies</a> | ||||||
|  |           </p> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |       END_HTML | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     html += <<-END_HTML | ||||||
|  |     <div class="pure-g"> | ||||||
|  |       <div class="pure-u-1"> | ||||||
|  |         <p> | ||||||
|  |           <a href="javascript:void(0)" onclick="toggle(this)">[ - ]</a>  | ||||||
|  |           <i class="icon ion-ios-thumbs-up"></i> #{child["likeCount"]}  | ||||||
|  |           <b><a href="#{child["authorUrl"]}">#{child["author"]}</a></b> | ||||||
|  |         </p> | ||||||
|  |         <div> | ||||||
|  |         #{child["content"]} | ||||||
|  |         #{replies_html} | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |     END_HTML | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   if comments["continuation"]? | ||||||
|  |     html += <<-END_HTML | ||||||
|  |     <div class="pure-g"> | ||||||
|  |       <div class="pure-u-1"> | ||||||
|  |         <p> | ||||||
|  |           <a href="javascript:void(0)" data-continuation="#{comments["continuation"]}" | ||||||
|  |             onclick="load_comments(this)">Load more</a> | ||||||
|  |         </p> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |     END_HTML | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return html | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def template_reddit_comments(root) | ||||||
|  |   html = "" | ||||||
|  |   root.each do |child| | ||||||
|  |     if child.data.is_a?(RedditComment) | ||||||
|  |       child = child.data.as(RedditComment) | ||||||
|  |       author = child.author | ||||||
|  |       score = child.score | ||||||
|  |       body_html = HTML.unescape(child.body_html) | ||||||
|  | 
 | ||||||
|  |       replies_html = "" | ||||||
|  |       if child.replies.is_a?(RedditThing) | ||||||
|  |         replies = child.replies.as(RedditThing) | ||||||
|  |         replies_html = template_reddit_comments(replies.data.as(RedditListing).children) | ||||||
|  |       end | ||||||
|  | 
 | ||||||
|  |       content = <<-END_HTML | ||||||
|  |       <p> | ||||||
|  |         <a href="javascript:void(0)" onclick="toggle(this)">[ - ]</a>  | ||||||
|  |         <i class="icon ion-ios-thumbs-up"></i> #{score}  | ||||||
|  |         <b><a href="https://www.reddit.com/user/#{author}">#{author}</a></b>  | ||||||
|  |       </p> | ||||||
|  |       <div> | ||||||
|  |       #{body_html} | ||||||
|  |       #{replies_html} | ||||||
|  |       </div> | ||||||
|  |       END_HTML | ||||||
|  | 
 | ||||||
|  |       if child.depth > 0 | ||||||
|  |         html += <<-END_HTML | ||||||
|  |           <div class="pure-g"> | ||||||
|  |           <div class="pure-u-1-24"> | ||||||
|  |           </div> | ||||||
|  |           <div class="pure-u-23-24"> | ||||||
|  |           #{content} | ||||||
|  |           </div> | ||||||
|  |           </div> | ||||||
|  |         END_HTML | ||||||
|  |       else | ||||||
|  |         html += <<-END_HTML | ||||||
|  |           <div class="pure-g"> | ||||||
|  |           <div class="pure-u-1"> | ||||||
|  |           #{content} | ||||||
|  |           </div> | ||||||
|  |           </div> | ||||||
|  |         END_HTML | ||||||
|  |       end | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return html | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def add_alt_links(html) | ||||||
|  |   alt_links = [] of {String, String} | ||||||
|  | 
 | ||||||
|  |   # This is painful but likely the only way to accomplish this in Crystal, | ||||||
|  |   # as Crystigiri and others are not able to insert XML Nodes into a document. | ||||||
|  |   # The goal here is to use as little regex as possible | ||||||
|  |   html.scan(/<a[^>]*>([^<]+)<\/a>/) do |match| | ||||||
|  |     anchor = XML.parse_html(match[0]) | ||||||
|  |     anchor = anchor.xpath_node("//a").not_nil! | ||||||
|  |     url = URI.parse(anchor["href"]) | ||||||
|  | 
 | ||||||
|  |     if ["www.youtube.com", "m.youtube.com"].includes?(url.host) | ||||||
|  |       if url.path == "/redirect" | ||||||
|  |         params = HTTP::Params.parse(url.query.not_nil!) | ||||||
|  |         alt_url = params["q"]? | ||||||
|  |         alt_url ||= "/" | ||||||
|  |       else | ||||||
|  |         alt_url = url.full_path | ||||||
|  |       end | ||||||
|  | 
 | ||||||
|  |       alt_link = <<-END_HTML | ||||||
|  |       <a href="#{alt_url}"> | ||||||
|  |         <i class="icon ion-ios-link"></i> | ||||||
|  |       </a> | ||||||
|  |       END_HTML | ||||||
|  |     elsif url.host == "youtu.be" | ||||||
|  |       alt_link = <<-END_HTML | ||||||
|  |       <a href="/watch?v=#{url.path.try &.lchop("/")}&#{url.query}"> | ||||||
|  |         <i class="icon ion-ios-link"></i> | ||||||
|  |       </a> | ||||||
|  |       END_HTML | ||||||
|  |     elsif url.to_s == "#" | ||||||
|  |       length_seconds = decode_length_seconds(anchor.content) | ||||||
|  |       alt_anchor = <<-END_HTML | ||||||
|  |       <a href="javascript:void(0)" onclick="player.currentTime(#{length_seconds})">#{anchor.content}</a> | ||||||
|  |       END_HTML | ||||||
|  | 
 | ||||||
|  |       html = html.sub(anchor.to_s, alt_anchor) | ||||||
|  |       next | ||||||
|  |     else | ||||||
|  |       alt_link = "" | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     alt_links << {anchor.to_s, alt_link} | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   alt_links.each do |original, alternate| | ||||||
|  |     html = html.sub(original, original + alternate) | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return html | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def fill_links(html, scheme, host) | ||||||
|  |   html = XML.parse_html(html) | ||||||
|  | 
 | ||||||
|  |   html.xpath_nodes("//a").each do |match| | ||||||
|  |     url = URI.parse(match["href"]) | ||||||
|  |     # Reddit links don't have host | ||||||
|  |     if !url.host && !match["href"].starts_with?("javascript") && !url.to_s.ends_with? "#" | ||||||
|  |       url.scheme = scheme | ||||||
|  |       url.host = host | ||||||
|  |       match["href"] = url | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   if host == "www.youtube.com" | ||||||
|  |     html = html.xpath_node(%q(//p[@id="eow-description"])).not_nil!.to_xml | ||||||
|  |   else | ||||||
|  |     html = html.to_xml(options: XML::SaveOptions::NO_DECL) | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   html | ||||||
|  | end | ||||||
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										273
									
								
								src/invidious/helpers/helpers.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										273
									
								
								src/invidious/helpers/helpers.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,273 @@ | ||||||
|  | class Config | ||||||
|  |   YAML.mapping({ | ||||||
|  |     crawl_threads:   Int32, | ||||||
|  |     channel_threads: Int32, | ||||||
|  |     video_threads:   Int32, | ||||||
|  |     db:              NamedTuple( | ||||||
|  |       user: String, | ||||||
|  |       password: String, | ||||||
|  |       host: String, | ||||||
|  |       port: Int32, | ||||||
|  |       dbname: String, | ||||||
|  |     ), | ||||||
|  |     dl_api_key: String?, | ||||||
|  |     https_only: Bool?, | ||||||
|  |     hmac_key:   String?, | ||||||
|  |   }) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | class FilteredCompressHandler < Kemal::Handler | ||||||
|  |   exclude ["/videoplayback", "/api/*"] | ||||||
|  | 
 | ||||||
|  |   def call(env) | ||||||
|  |     return call_next env if exclude_match? env | ||||||
|  | 
 | ||||||
|  |     {% if flag?(:without_zlib) %} | ||||||
|  |       call_next env | ||||||
|  |     {% else %} | ||||||
|  |       request_headers = env.request.headers | ||||||
|  | 
 | ||||||
|  |       if request_headers.includes_word?("Accept-Encoding", "gzip") | ||||||
|  |         env.response.headers["Content-Encoding"] = "gzip" | ||||||
|  |         env.response.output = Gzip::Writer.new(env.response.output, sync_close: true) | ||||||
|  |       elsif request_headers.includes_word?("Accept-Encoding", "deflate") | ||||||
|  |         env.response.headers["Content-Encoding"] = "deflate" | ||||||
|  |         env.response.output = Flate::Writer.new(env.response.output, sync_close: true) | ||||||
|  |       end | ||||||
|  | 
 | ||||||
|  |       call_next env | ||||||
|  |     {% end %} | ||||||
|  |   end | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def rank_videos(db, n, filter, url) | ||||||
|  |   top = [] of {Float64, String} | ||||||
|  | 
 | ||||||
|  |   db.query("SELECT id, wilson_score, published FROM videos WHERE views > 5000 ORDER BY published DESC LIMIT 1000") do |rs| | ||||||
|  |     rs.each do | ||||||
|  |       id = rs.read(String) | ||||||
|  |       wilson_score = rs.read(Float64) | ||||||
|  |       published = rs.read(Time) | ||||||
|  | 
 | ||||||
|  |       # Exponential decay, older videos tend to rank lower | ||||||
|  |       temperature = wilson_score * Math.exp(-0.000005*((Time.now - published).total_minutes)) | ||||||
|  |       top << {temperature, id} | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   top.sort! | ||||||
|  | 
 | ||||||
|  |   # Make hottest come first | ||||||
|  |   top.reverse! | ||||||
|  |   top = top.map { |a, b| b } | ||||||
|  | 
 | ||||||
|  |   if filter | ||||||
|  |     language_list = [] of String | ||||||
|  |     top.each do |id| | ||||||
|  |       if language_list.size == n | ||||||
|  |         break | ||||||
|  |       else | ||||||
|  |         client = make_client(url) | ||||||
|  |         begin | ||||||
|  |           video = get_video(id, db) | ||||||
|  |         rescue ex | ||||||
|  |           next | ||||||
|  |         end | ||||||
|  | 
 | ||||||
|  |         if video.language | ||||||
|  |           language = video.language | ||||||
|  |         else | ||||||
|  |           description = XML.parse(video.description) | ||||||
|  |           content = [video.title, description.content].join(" ") | ||||||
|  |           content = content[0, 10000] | ||||||
|  | 
 | ||||||
|  |           results = DetectLanguage.detect(content) | ||||||
|  |           language = results[0].language | ||||||
|  | 
 | ||||||
|  |           db.exec("UPDATE videos SET language = $1 WHERE id = $2", language, id) | ||||||
|  |         end | ||||||
|  | 
 | ||||||
|  |         if language == "en" | ||||||
|  |           language_list << id | ||||||
|  |         end | ||||||
|  |       end | ||||||
|  |     end | ||||||
|  |     return language_list | ||||||
|  |   else | ||||||
|  |     return top[0..n - 1] | ||||||
|  |   end | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def login_req(login_form, f_req) | ||||||
|  |   data = { | ||||||
|  |     "pstMsg"          => "1", | ||||||
|  |     "checkConnection" => "youtube", | ||||||
|  |     "checkedDomains"  => "youtube", | ||||||
|  |     "hl"              => "en", | ||||||
|  |     "deviceinfo"      => %q([null,null,null,[],null,"US",null,null,[],"GlifWebSignIn",null,[null,null,[]]]), | ||||||
|  |     "f.req"           => f_req, | ||||||
|  |     "flowName"        => "GlifWebSignIn", | ||||||
|  |     "flowEntry"       => "ServiceLogin", | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   data = login_form.merge(data) | ||||||
|  | 
 | ||||||
|  |   return HTTP::Params.encode(data) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def produce_playlist_url(ucid, index) | ||||||
|  |   ucid = ucid.lchop("UC") | ||||||
|  |   ucid = "VLUU" + ucid | ||||||
|  | 
 | ||||||
|  |   continuation = write_var_int(index) | ||||||
|  |   continuation.unshift(0x08_u8) | ||||||
|  |   slice = continuation.to_unsafe.to_slice(continuation.size) | ||||||
|  | 
 | ||||||
|  |   continuation = Base64.urlsafe_encode(slice, false) | ||||||
|  |   continuation = "PT:" + continuation | ||||||
|  |   continuation = continuation.bytes | ||||||
|  |   continuation.unshift(0x7a_u8, continuation.size.to_u8) | ||||||
|  | 
 | ||||||
|  |   slice = continuation.to_unsafe.to_slice(continuation.size) | ||||||
|  |   continuation = Base64.urlsafe_encode(slice) | ||||||
|  |   continuation = URI.escape(continuation) | ||||||
|  |   continuation = continuation.bytes | ||||||
|  |   continuation.unshift(continuation.size.to_u8) | ||||||
|  | 
 | ||||||
|  |   continuation.unshift(ucid.size.to_u8) | ||||||
|  |   continuation = ucid.bytes + continuation | ||||||
|  |   continuation.unshift(0x12.to_u8, ucid.size.to_u8) | ||||||
|  |   continuation.unshift(0xe2_u8, 0xa9_u8, 0x85_u8, 0xb2_u8, 2_u8, continuation.size.to_u8) | ||||||
|  | 
 | ||||||
|  |   slice = continuation.to_unsafe.to_slice(continuation.size) | ||||||
|  |   continuation = Base64.urlsafe_encode(slice) | ||||||
|  |   continuation = URI.escape(continuation) | ||||||
|  | 
 | ||||||
|  |   url = "/browse_ajax?action_continuation=1&continuation=#{continuation}" | ||||||
|  | 
 | ||||||
|  |   return url | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def produce_videos_url(ucid, page = 1) | ||||||
|  |   page = "#{page}" | ||||||
|  | 
 | ||||||
|  |   meta = "\x12\x06videos \x00\x30\x02\x38\x01\x60\x01\x6a\x00\x7a" | ||||||
|  |   meta += page.size.to_u8.unsafe_chr | ||||||
|  |   meta += page | ||||||
|  |   meta += "\xb8\x01\x00" | ||||||
|  | 
 | ||||||
|  |   meta = Base64.urlsafe_encode(meta) | ||||||
|  |   meta = URI.escape(meta) | ||||||
|  | 
 | ||||||
|  |   continuation = "\x12" | ||||||
|  |   continuation += ucid.size.to_u8.unsafe_chr | ||||||
|  |   continuation += ucid | ||||||
|  |   continuation += "\x1a" | ||||||
|  |   continuation += meta.size.to_u8.unsafe_chr | ||||||
|  |   continuation += meta | ||||||
|  | 
 | ||||||
|  |   continuation = continuation.size.to_u8.unsafe_chr + continuation | ||||||
|  |   continuation = "\xe2\xa9\x85\xb2\x02" + continuation | ||||||
|  | 
 | ||||||
|  |   continuation = Base64.urlsafe_encode(continuation) | ||||||
|  |   continuation = URI.escape(continuation) | ||||||
|  | 
 | ||||||
|  |   url = "/browse_ajax?continuation=#{continuation}" | ||||||
|  | 
 | ||||||
|  |   return url | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def read_var_int(bytes) | ||||||
|  |   numRead = 0 | ||||||
|  |   result = 0 | ||||||
|  | 
 | ||||||
|  |   read = bytes[numRead] | ||||||
|  | 
 | ||||||
|  |   if bytes.size == 1 | ||||||
|  |     result = bytes[0].to_i32 | ||||||
|  |   else | ||||||
|  |     while ((read & 0b10000000) != 0) | ||||||
|  |       read = bytes[numRead].to_u64 | ||||||
|  |       value = (read & 0b01111111) | ||||||
|  |       result |= (value << (7 * numRead)) | ||||||
|  | 
 | ||||||
|  |       numRead += 1 | ||||||
|  |       if numRead > 5 | ||||||
|  |         raise "VarInt is too big" | ||||||
|  |       end | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return result | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def write_var_int(value : Int) | ||||||
|  |   bytes = [] of UInt8 | ||||||
|  |   value = value.to_u32 | ||||||
|  | 
 | ||||||
|  |   if value == 0 | ||||||
|  |     bytes = [0_u8] | ||||||
|  |   else | ||||||
|  |     while value != 0 | ||||||
|  |       temp = (value & 0b01111111).to_u8 | ||||||
|  |       value = value >> 7 | ||||||
|  | 
 | ||||||
|  |       if value != 0 | ||||||
|  |         temp |= 0b10000000 | ||||||
|  |       end | ||||||
|  | 
 | ||||||
|  |       bytes << temp | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return bytes | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def generate_captcha(key) | ||||||
|  |   minute = Random::Secure.rand(12) | ||||||
|  |   minute_angle = minute * 30 | ||||||
|  |   minute = minute * 5 | ||||||
|  | 
 | ||||||
|  |   hour = Random::Secure.rand(12) | ||||||
|  |   hour_angle = hour * 30 + minute_angle.to_f / 12 | ||||||
|  |   if hour == 0 | ||||||
|  |     hour = 12 | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   clock_svg = <<-END_SVG | ||||||
|  |   <svg viewBox="0 0 100 100" width="200px"> | ||||||
|  |   <circle cx="50" cy="50" r="45" fill="#eee" stroke="black" stroke-width="2"></circle> | ||||||
|  |    | ||||||
|  |   <text x="69"     y="20.091" text-anchor="middle" fill="black" font-family="Arial" font-size="10px"> 1</text> | ||||||
|  |   <text x="82.909" y="34"     text-anchor="middle" fill="black" font-family="Arial" font-size="10px"> 2</text> | ||||||
|  |   <text x="88"     y="53"     text-anchor="middle" fill="black" font-family="Arial" font-size="10px"> 3</text> | ||||||
|  |   <text x="82.909" y="72"     text-anchor="middle" fill="black" font-family="Arial" font-size="10px"> 4</text> | ||||||
|  |   <text x="69"     y="85.909" text-anchor="middle" fill="black" font-family="Arial" font-size="10px"> 5</text> | ||||||
|  |   <text x="50"     y="91"     text-anchor="middle" fill="black" font-family="Arial" font-size="10px"> 6</text> | ||||||
|  |   <text x="31"     y="85.909" text-anchor="middle" fill="black" font-family="Arial" font-size="10px"> 7</text> | ||||||
|  |   <text x="17.091" y="72"     text-anchor="middle" fill="black" font-family="Arial" font-size="10px"> 8</text> | ||||||
|  |   <text x="12"     y="53"     text-anchor="middle" fill="black" font-family="Arial" font-size="10px"> 9</text> | ||||||
|  |   <text x="17.091" y="34"     text-anchor="middle" fill="black" font-family="Arial" font-size="10px">10</text> | ||||||
|  |   <text x="31"     y="20.091" text-anchor="middle" fill="black" font-family="Arial" font-size="10px">11</text> | ||||||
|  |   <text x="50"     y="15"     text-anchor="middle" fill="black" font-family="Arial" font-size="10px">12</text> | ||||||
|  | 
 | ||||||
|  |   <circle cx="50" cy="50" r="3" fill="black"></circle> | ||||||
|  |   <line id="minute" transform="rotate(#{minute_angle}, 50, 50)" x1="50" y1="50" x2="50" y2="16" fill="black" stroke="black" stroke-width="2"></line> | ||||||
|  |   <line id="hour"   transform="rotate(#{hour_angle}, 50, 50)" x1="50" y1="50" x2="50" y2="24" fill="black" stroke="black" stroke-width="2"></line> | ||||||
|  |   </svg> | ||||||
|  |   END_SVG | ||||||
|  | 
 | ||||||
|  |   challenge = "" | ||||||
|  |   convert = Process.run(%(convert -density 1200 -resize 400x400 -background none svg:- png:-), shell: true, | ||||||
|  |     input: IO::Memory.new(clock_svg), output: Process::Redirect::Pipe) do |proc| | ||||||
|  |     challenge = proc.output.gets_to_end | ||||||
|  |     challenge = Base64.strict_encode(challenge) | ||||||
|  |     challenge = "data:image/png;base64,#{challenge}" | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   answer = "#{hour}:#{minute.to_s.rjust(2, '0')}" | ||||||
|  |   token = OpenSSL::HMAC.digest(:sha256, key, answer) | ||||||
|  |   token = Base64.encode(token) | ||||||
|  | 
 | ||||||
|  |   return {challenge: challenge, token: token} | ||||||
|  | end | ||||||
							
								
								
									
										18
									
								
								src/invidious/helpers/macros.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/invidious/helpers/macros.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,18 @@ | ||||||
|  | macro add_mapping(mapping) | ||||||
|  |     def initialize({{*mapping.keys.map { |id| "@#{id}".id }}}) | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     def to_a | ||||||
|  |         return [{{*mapping.keys.map { |id| "@#{id}".id }}}] | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     DB.mapping({{mapping}}) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | macro templated(filename) | ||||||
|  |     render "src/invidious/views/#{{{filename}}}.ecr", "src/invidious/views/layout.ecr" | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | macro rendered(filename) | ||||||
|  |     render "src/invidious/views/#{{{filename}}}.ecr" | ||||||
|  | end | ||||||
							
								
								
									
										129
									
								
								src/invidious/helpers/utils.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										129
									
								
								src/invidious/helpers/utils.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,129 @@ | ||||||
|  | # See http://www.evanmiller.org/how-not-to-sort-by-average-rating.html | ||||||
|  | def ci_lower_bound(pos, n) | ||||||
|  |   if n == 0 | ||||||
|  |     return 0.0 | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   # z value here represents a confidence level of 0.95 | ||||||
|  |   z = 1.96 | ||||||
|  |   phat = 1.0*pos/n | ||||||
|  | 
 | ||||||
|  |   return (phat + z*z/(2*n) - z * Math.sqrt((phat*(1 - phat) + z*z/(4*n))/n))/(1 + z*z/n) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def elapsed_text(elapsed) | ||||||
|  |   millis = elapsed.total_milliseconds | ||||||
|  |   return "#{millis.round(2)}ms" if millis >= 1 | ||||||
|  | 
 | ||||||
|  |   "#{(millis * 1000).round(2)}µs" | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def make_client(url) | ||||||
|  |   context = OpenSSL::SSL::Context::Client.new | ||||||
|  |   context.add_options( | ||||||
|  |     OpenSSL::SSL::Options::ALL | | ||||||
|  |     OpenSSL::SSL::Options::NO_SSL_V2 | | ||||||
|  |     OpenSSL::SSL::Options::NO_SSL_V3 | ||||||
|  |   ) | ||||||
|  |   client = HTTP::Client.new(url, context) | ||||||
|  |   client.read_timeout = 10.seconds | ||||||
|  |   client.connect_timeout = 10.seconds | ||||||
|  |   return client | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def decode_length_seconds(string) | ||||||
|  |   length_seconds = string.split(":").map { |a| a.to_i } | ||||||
|  |   length_seconds = [0] * (3 - length_seconds.size) + length_seconds | ||||||
|  |   length_seconds = Time::Span.new(length_seconds[0], length_seconds[1], length_seconds[2]) | ||||||
|  |   length_seconds = length_seconds.total_seconds.to_i | ||||||
|  | 
 | ||||||
|  |   return length_seconds | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def decode_time(string) | ||||||
|  |   time = string.try &.to_f? | ||||||
|  | 
 | ||||||
|  |   if !time | ||||||
|  |     hours = /(?<hours>\d+)h/.match(string).try &.["hours"].try &.to_f | ||||||
|  |     hours ||= 0 | ||||||
|  | 
 | ||||||
|  |     minutes = /(?<minutes>\d+)m(?!s)/.match(string).try &.["minutes"].try &.to_f | ||||||
|  |     minutes ||= 0 | ||||||
|  | 
 | ||||||
|  |     seconds = /(?<seconds>\d+)s/.match(string).try &.["seconds"].try &.to_f | ||||||
|  |     seconds ||= 0 | ||||||
|  | 
 | ||||||
|  |     millis = /(?<millis>\d+)ms/.match(string).try &.["millis"].try &.to_f | ||||||
|  |     millis ||= 0 | ||||||
|  | 
 | ||||||
|  |     time = hours * 3600 + minutes * 60 + seconds + millis / 1000 | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return time | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def decode_date(string : String) | ||||||
|  |   # Time matches format "20 hours ago", "40 minutes ago"... | ||||||
|  |   date = string.split(" ")[-3, 3] | ||||||
|  |   delta = date[0].to_i | ||||||
|  | 
 | ||||||
|  |   case date[1] | ||||||
|  |   when .includes? "minute" | ||||||
|  |     delta = delta.minutes | ||||||
|  |   when .includes? "hour" | ||||||
|  |     delta = delta.hours | ||||||
|  |   when .includes? "day" | ||||||
|  |     delta = delta.days | ||||||
|  |   when .includes? "week" | ||||||
|  |     delta = delta.weeks | ||||||
|  |   when .includes? "month" | ||||||
|  |     delta = delta.months | ||||||
|  |   when .includes? "year" | ||||||
|  |     delta = delta.years | ||||||
|  |   else | ||||||
|  |     raise "Could not parse #{string}" | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return Time.now - delta | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def recode_date(time : Time) | ||||||
|  |   span = Time.now - time | ||||||
|  | 
 | ||||||
|  |   if span.total_days > 365.0 | ||||||
|  |     span = {span.total_days / 365, "year"} | ||||||
|  |   elsif span.total_days > 30.0 | ||||||
|  |     span = {span.total_days / 30, "month"} | ||||||
|  |   elsif span.total_days > 7.0 | ||||||
|  |     span = {span.total_days / 7, "week"} | ||||||
|  |   elsif span.total_hours > 24.0 | ||||||
|  |     span = {span.total_days, "day"} | ||||||
|  |   elsif span.total_minutes > 60.0 | ||||||
|  |     span = {span.total_hours, "hour"} | ||||||
|  |   else | ||||||
|  |     span = {0, "units"} | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   span = {span[0].to_i, span[1]} | ||||||
|  |   if span[0] > 1 | ||||||
|  |     span = {span[0], span[1] + "s"} | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return span.join(" ") | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def number_with_separator(number) | ||||||
|  |   number.to_s.reverse.gsub(/(\d{3})(?=\d)/, "\\1,").reverse | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def arg_array(array, start = 1) | ||||||
|  |   if array.size == 0 | ||||||
|  |     args = "NULL" | ||||||
|  |   else | ||||||
|  |     args = [] of String | ||||||
|  |     (start..array.size + start - 1).each { |i| args << "($#{i})" } | ||||||
|  |     args = args.join(",") | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return args | ||||||
|  | end | ||||||
							
								
								
									
										136
									
								
								src/invidious/jobs.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										136
									
								
								src/invidious/jobs.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,136 @@ | ||||||
|  | def crawl_videos(db) | ||||||
|  |   ids = Deque(String).new | ||||||
|  |   random = Random.new | ||||||
|  | 
 | ||||||
|  |   search(random.base64(3)).each do |video| | ||||||
|  |     ids << video.id | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   loop do | ||||||
|  |     if ids.empty? | ||||||
|  |       search(random.base64(3)).each do |video| | ||||||
|  |         ids << video.id | ||||||
|  |       end | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     begin | ||||||
|  |       id = ids[0] | ||||||
|  |       video = get_video(id, db) | ||||||
|  |     rescue ex | ||||||
|  |       STDOUT << id << " : " << ex.message << "\n" | ||||||
|  |       next | ||||||
|  |     ensure | ||||||
|  |       ids.delete(id) | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     rvs = [] of Hash(String, String) | ||||||
|  |     if video.info.has_key?("rvs") | ||||||
|  |       video.info["rvs"].split(",").each do |rv| | ||||||
|  |         rvs << HTTP::Params.parse(rv).to_h | ||||||
|  |       end | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     rvs.each do |rv| | ||||||
|  |       if rv.has_key?("id") && !db.query_one?("SELECT EXISTS (SELECT true FROM videos WHERE id = $1)", rv["id"], as: Bool) | ||||||
|  |         ids.delete(id) | ||||||
|  |         ids << rv["id"] | ||||||
|  |         if ids.size == 150 | ||||||
|  |           ids.shift | ||||||
|  |         end | ||||||
|  |       end | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     Fiber.yield | ||||||
|  |   end | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def refresh_channels(db) | ||||||
|  |   loop do | ||||||
|  |     db.query("SELECT id FROM channels ORDER BY updated") do |rs| | ||||||
|  |       rs.each do | ||||||
|  |         client = make_client(YT_URL) | ||||||
|  | 
 | ||||||
|  |         begin | ||||||
|  |           id = rs.read(String) | ||||||
|  |           channel = fetch_channel(id, client, db, false) | ||||||
|  |           db.exec("UPDATE channels SET updated = $1 WHERE id = $2", Time.now, id) | ||||||
|  |         rescue ex | ||||||
|  |           STDOUT << id << " : " << ex.message << "\n" | ||||||
|  |           next | ||||||
|  |         end | ||||||
|  |       end | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     Fiber.yield | ||||||
|  |   end | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def refresh_videos(db) | ||||||
|  |   loop do | ||||||
|  |     db.query("SELECT id FROM videos ORDER BY updated") do |rs| | ||||||
|  |       rs.each do | ||||||
|  |         begin | ||||||
|  |           id = rs.read(String) | ||||||
|  |           video = get_video(id, db) | ||||||
|  |         rescue ex | ||||||
|  |           STDOUT << id << " : " << ex.message << "\n" | ||||||
|  |           next | ||||||
|  |         end | ||||||
|  |       end | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     Fiber.yield | ||||||
|  |   end | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def pull_top_videos(config, db) | ||||||
|  |   if config.dl_api_key | ||||||
|  |     DetectLanguage.configure do |dl_config| | ||||||
|  |       dl_config.api_key = config.dl_api_key.not_nil! | ||||||
|  |     end | ||||||
|  |     filter = true | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   filter ||= false | ||||||
|  | 
 | ||||||
|  |   loop do | ||||||
|  |     begin | ||||||
|  |       top = rank_videos(db, 40, filter, YT_URL) | ||||||
|  |     rescue ex | ||||||
|  |       next | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     if top.size > 0 | ||||||
|  |       args = arg_array(top) | ||||||
|  |     else | ||||||
|  |       next | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     videos = [] of Video | ||||||
|  | 
 | ||||||
|  |     top.each do |id| | ||||||
|  |       begin | ||||||
|  |         videos << get_video(id, db) | ||||||
|  |       rescue ex | ||||||
|  |         next | ||||||
|  |       end | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     yield videos | ||||||
|  |     Fiber.yield | ||||||
|  |   end | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def update_decrypt_function | ||||||
|  |   loop do | ||||||
|  |     begin | ||||||
|  |       client = make_client(YT_URL) | ||||||
|  |       decrypt_function = fetch_decrypt_function(client) | ||||||
|  |     rescue ex | ||||||
|  |       next | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     yield decrypt_function | ||||||
|  |     Fiber.yield | ||||||
|  |   end | ||||||
|  | end | ||||||
							
								
								
									
										30
									
								
								src/invidious/search.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								src/invidious/search.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,30 @@ | ||||||
|  | def search(query, page = 1) | ||||||
|  |   client = make_client(YT_URL) | ||||||
|  |   html = client.get("/results?q=#{URI.escape(query)}&page=#{page}&sp=EgIQAVAU").body | ||||||
|  |   html = XML.parse_html(html) | ||||||
|  | 
 | ||||||
|  |   videos = [] of ChannelVideo | ||||||
|  | 
 | ||||||
|  |   html.xpath_nodes(%q(//ol[@class="item-section"]/li)).each do |item| | ||||||
|  |     root = item.xpath_node(%q(div[contains(@class,"yt-lockup-video")]/div)) | ||||||
|  |     if !root | ||||||
|  |       next | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     id = root.xpath_node(%q(.//div[contains(@class,"yt-lockup-thumbnail")]/a/@href)).not_nil!.content.lchop("/watch?v=") | ||||||
|  | 
 | ||||||
|  |     title = root.xpath_node(%q(.//div[@class="yt-lockup-content"]/h3/a)).not_nil!.content | ||||||
|  | 
 | ||||||
|  |     author = root.xpath_node(%q(.//div[@class="yt-lockup-content"]/div/a)).not_nil! | ||||||
|  |     ucid = author["href"].rpartition("/")[-1] | ||||||
|  |     author = author.content | ||||||
|  | 
 | ||||||
|  |     published = root.xpath_node(%q(.//ul[@class="yt-lockup-meta-info"]/li[1])).not_nil!.content | ||||||
|  |     published = decode_date(published) | ||||||
|  | 
 | ||||||
|  |     video = ChannelVideo.new(id, title, published, Time.now, ucid, author) | ||||||
|  |     videos << video | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return videos | ||||||
|  | end | ||||||
							
								
								
									
										65
									
								
								src/invidious/signatures.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								src/invidious/signatures.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,65 @@ | ||||||
|  | def fetch_decrypt_function(client, id = "CvFH_6DNRCY") | ||||||
|  |   document = client.get("/watch?v=#{id}").body | ||||||
|  |   url = document.match(/src="(?<url>\/yts\/jsbin\/player-.{9}\/en_US\/base.js)"/).not_nil!["url"] | ||||||
|  |   player = client.get(url).body | ||||||
|  | 
 | ||||||
|  |   function_name = player.match(/\(b\|\|\(b="signature"\),d.set\(b,(?<name>[a-zA-Z0-9]{2})\(c\)\)\)/).not_nil!["name"] | ||||||
|  |   function_body = player.match(/#{function_name}=function\(a\){(?<body>[^}]+)}/).not_nil!["body"] | ||||||
|  |   function_body = function_body.split(";")[1..-2] | ||||||
|  | 
 | ||||||
|  |   var_name = function_body[0][0, 2] | ||||||
|  | 
 | ||||||
|  |   operations = {} of String => String | ||||||
|  |   matches = player.delete("\n").match(/var #{var_name}={(?<op1>[a-zA-Z0-9]{2}:[^}]+}),(?<op2>[a-zA-Z0-9]{2}:[^}]+}),(?<op3>[a-zA-Z0-9]{2}:[^}]+})};/).not_nil! | ||||||
|  |   3.times do |i| | ||||||
|  |     operation = matches["op#{i + 1}"] | ||||||
|  |     op_name = operation[0, 2] | ||||||
|  | 
 | ||||||
|  |     op_body = operation.match(/\{[^}]+\}/).not_nil![0] | ||||||
|  |     case op_body | ||||||
|  |     when "{a.reverse()}" | ||||||
|  |       operations[op_name] = "a" | ||||||
|  |     when "{a.splice(0,b)}" | ||||||
|  |       operations[op_name] = "b" | ||||||
|  |     else | ||||||
|  |       operations[op_name] = "c" | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   decrypt_function = [] of {name: String, value: Int32} | ||||||
|  |   function_body.each do |function| | ||||||
|  |     function = function.lchop(var_name + ".") | ||||||
|  |     op_name = function[0, 2] | ||||||
|  | 
 | ||||||
|  |     function = function.lchop(op_name + "(a,") | ||||||
|  |     value = function.rchop(")").to_i | ||||||
|  | 
 | ||||||
|  |     decrypt_function << {name: operations[op_name], value: value} | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return decrypt_function | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def decrypt_signature(a, code) | ||||||
|  |   a = a.split("") | ||||||
|  | 
 | ||||||
|  |   code.each do |item| | ||||||
|  |     case item[:name] | ||||||
|  |     when "a" | ||||||
|  |       a.reverse! | ||||||
|  |     when "b" | ||||||
|  |       a.delete_at(0..(item[:value] - 1)) | ||||||
|  |     when "c" | ||||||
|  |       a = splice(a, item[:value]) | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return a.join("") | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def splice(a, b) | ||||||
|  |   c = a[0] | ||||||
|  |   a[0] = a[b % a.size] | ||||||
|  |   a[b % a.size] = c | ||||||
|  |   return a | ||||||
|  | end | ||||||
							
								
								
									
										146
									
								
								src/invidious/users.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										146
									
								
								src/invidious/users.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,146 @@ | ||||||
|  | class User | ||||||
|  |   module PreferencesConverter | ||||||
|  |     def self.from_rs(rs) | ||||||
|  |       begin | ||||||
|  |         Preferences.from_json(rs.read(String)) | ||||||
|  |       rescue ex | ||||||
|  |         DEFAULT_USER_PREFERENCES | ||||||
|  |       end | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   add_mapping({ | ||||||
|  |     id:            String, | ||||||
|  |     updated:       Time, | ||||||
|  |     notifications: Array(String), | ||||||
|  |     subscriptions: Array(String), | ||||||
|  |     email:         String, | ||||||
|  |     preferences:   { | ||||||
|  |       type:      Preferences, | ||||||
|  |       default:   DEFAULT_USER_PREFERENCES, | ||||||
|  |       converter: PreferencesConverter, | ||||||
|  |     }, | ||||||
|  |     password: String?, | ||||||
|  |     token:    String, | ||||||
|  |     watched:  Array(String), | ||||||
|  |   }) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | DEFAULT_USER_PREFERENCES = Preferences.from_json({ | ||||||
|  |   "video_loop"  => false, | ||||||
|  |   "autoplay"    => false, | ||||||
|  |   "speed"       => 1.0, | ||||||
|  |   "quality"     => "hd720", | ||||||
|  |   "volume"      => 100, | ||||||
|  |   "comments"    => "youtube", | ||||||
|  |   "dark_mode"   => false, | ||||||
|  |   "thin_mode "  => false, | ||||||
|  |   "max_results" => 40, | ||||||
|  |   "sort"        => "published", | ||||||
|  |   "latest_only" => false, | ||||||
|  |   "unseen_only" => false, | ||||||
|  | }.to_json) | ||||||
|  | 
 | ||||||
|  | # TODO: Migrate preferences so fields will not be nilable | ||||||
|  | class Preferences | ||||||
|  |   JSON.mapping({ | ||||||
|  |     video_loop: Bool, | ||||||
|  |     autoplay:   Bool, | ||||||
|  |     speed:      Float32, | ||||||
|  |     quality:    String, | ||||||
|  |     volume:     Int32, | ||||||
|  |     comments:   { | ||||||
|  |       type:    String, | ||||||
|  |       nilable: true, | ||||||
|  |       default: "youtube", | ||||||
|  |     }, | ||||||
|  |     redirect_feed: { | ||||||
|  |       type:    Bool, | ||||||
|  |       nilable: true, | ||||||
|  |       default: false, | ||||||
|  |     }, | ||||||
|  |     dark_mode: Bool, | ||||||
|  |     thin_mode: { | ||||||
|  |       type:    Bool, | ||||||
|  |       nilable: true, | ||||||
|  |       default: false, | ||||||
|  |     }, | ||||||
|  |     max_results:        Int32, | ||||||
|  |     sort:               String, | ||||||
|  |     latest_only:        Bool, | ||||||
|  |     unseen_only:        Bool, | ||||||
|  |     notifications_only: { | ||||||
|  |       type:    Bool, | ||||||
|  |       nilable: true, | ||||||
|  |       default: false, | ||||||
|  |     }, | ||||||
|  |   }) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def get_user(sid, client, headers, db, refresh = true) | ||||||
|  |   if db.query_one?("SELECT EXISTS (SELECT true FROM users WHERE id = $1)", sid, as: Bool) | ||||||
|  |     user = db.query_one("SELECT * FROM users WHERE id = $1", sid, as: User) | ||||||
|  | 
 | ||||||
|  |     if refresh && Time.now - user.updated > 1.minute | ||||||
|  |       user = fetch_user(sid, client, headers, db) | ||||||
|  |       user_array = user.to_a | ||||||
|  | 
 | ||||||
|  |       user_array[5] = user_array[5].to_json | ||||||
|  |       args = arg_array(user_array) | ||||||
|  | 
 | ||||||
|  |       db.exec("INSERT INTO users VALUES (#{args}) \ | ||||||
|  |       ON CONFLICT (email) DO UPDATE SET id = $1, updated = $2, subscriptions = $4", user_array) | ||||||
|  |     end | ||||||
|  |   else | ||||||
|  |     user = fetch_user(sid, client, headers, db) | ||||||
|  |     user_array = user.to_a | ||||||
|  | 
 | ||||||
|  |     user_array[5] = user_array[5].to_json | ||||||
|  |     args = arg_array(user.to_a) | ||||||
|  | 
 | ||||||
|  |     db.exec("INSERT INTO users VALUES (#{args}) \ | ||||||
|  |     ON CONFLICT (email) DO UPDATE SET id = $1, updated = $2, subscriptions = $4", user_array) | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return user | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def fetch_user(sid, client, headers, db) | ||||||
|  |   feed = client.get("/subscription_manager?disable_polymer=1", headers) | ||||||
|  |   feed = XML.parse_html(feed.body) | ||||||
|  | 
 | ||||||
|  |   channels = [] of String | ||||||
|  |   feed.xpath_nodes(%q(//ul[@id="guide-channels"]/li/a)).each do |channel| | ||||||
|  |     if !["Popular on YouTube", "Music", "Sports", "Gaming"].includes? channel["title"] | ||||||
|  |       channel_id = channel["href"].lstrip("/channel/") | ||||||
|  | 
 | ||||||
|  |       begin | ||||||
|  |         channel = get_channel(channel_id, client, db, false, false) | ||||||
|  |         channels << channel.id | ||||||
|  |       rescue ex | ||||||
|  |         next | ||||||
|  |       end | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   email = feed.xpath_node(%q(//a[@class="yt-masthead-picker-header yt-masthead-picker-active-account"])) | ||||||
|  |   if email | ||||||
|  |     email = email.content.strip | ||||||
|  |   else | ||||||
|  |     email = "" | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   token = Base64.urlsafe_encode(Random::Secure.random_bytes(32)) | ||||||
|  | 
 | ||||||
|  |   user = User.new(sid, Time.now, [] of String, channels, email, DEFAULT_USER_PREFERENCES, nil, token, [] of String) | ||||||
|  |   return user | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def create_user(sid, email, password) | ||||||
|  |   password = Crypto::Bcrypt::Password.create(password, cost: 10) | ||||||
|  |   token = Base64.urlsafe_encode(Random::Secure.random_bytes(32)) | ||||||
|  | 
 | ||||||
|  |   user = User.new(sid, Time.now, [] of String, [] of String, email, DEFAULT_USER_PREFERENCES, password.to_s, token, [] of String) | ||||||
|  | 
 | ||||||
|  |   return user | ||||||
|  | end | ||||||
							
								
								
									
										223
									
								
								src/invidious/videos.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										223
									
								
								src/invidious/videos.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,223 @@ | ||||||
|  | class Video | ||||||
|  |   module HTTPParamConverter | ||||||
|  |     def self.from_rs(rs) | ||||||
|  |       HTTP::Params.parse(rs.read(String)) | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   add_mapping({ | ||||||
|  |     id:   String, | ||||||
|  |     info: { | ||||||
|  |       type:      HTTP::Params, | ||||||
|  |       default:   HTTP::Params.parse(""), | ||||||
|  |       converter: Video::HTTPParamConverter, | ||||||
|  |     }, | ||||||
|  |     updated:            Time, | ||||||
|  |     title:              String, | ||||||
|  |     views:              Int64, | ||||||
|  |     likes:              Int32, | ||||||
|  |     dislikes:           Int32, | ||||||
|  |     wilson_score:       Float64, | ||||||
|  |     published:          Time, | ||||||
|  |     description:        String, | ||||||
|  |     language:           String?, | ||||||
|  |     author:             String, | ||||||
|  |     ucid:               String, | ||||||
|  |     allowed_regions:    Array(String), | ||||||
|  |     is_family_friendly: Bool, | ||||||
|  |     genre:              String, | ||||||
|  |   }) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def get_video(id, db, refresh = true) | ||||||
|  |   if db.query_one?("SELECT EXISTS (SELECT true FROM videos WHERE id = $1)", id, as: Bool) | ||||||
|  |     video = db.query_one("SELECT * FROM videos WHERE id = $1", id, as: Video) | ||||||
|  | 
 | ||||||
|  |     # If record was last updated over an hour ago, refresh (expire param in response lasts for 6 hours) | ||||||
|  |     if refresh && Time.now - video.updated > 1.hour | ||||||
|  |       begin | ||||||
|  |         video = fetch_video(id) | ||||||
|  |         video_array = video.to_a | ||||||
|  |         args = arg_array(video_array[1..-1], 2) | ||||||
|  | 
 | ||||||
|  |         db.exec("UPDATE videos SET (info,updated,title,views,likes,dislikes,wilson_score,\ | ||||||
|  |           published,description,language,author,ucid, allowed_regions, is_family_friendly, genre)\ | ||||||
|  |           = (#{args}) WHERE id = $1", video_array) | ||||||
|  |       rescue ex | ||||||
|  |         db.exec("DELETE FROM videos * WHERE id = $1", id) | ||||||
|  |         raise ex | ||||||
|  |       end | ||||||
|  |     end | ||||||
|  |   else | ||||||
|  |     video = fetch_video(id) | ||||||
|  |     video_array = video.to_a | ||||||
|  |     args = arg_array(video_array) | ||||||
|  | 
 | ||||||
|  |     db.exec("INSERT INTO videos VALUES (#{args}) ON CONFLICT (id) DO NOTHING", video_array) | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   return video | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def fetch_video(id) | ||||||
|  |   html_channel = Channel(XML::Node).new | ||||||
|  |   info_channel = Channel(HTTP::Params).new | ||||||
|  | 
 | ||||||
|  |   spawn do | ||||||
|  |     client = make_client(YT_URL) | ||||||
|  |     html = client.get("/watch?v=#{id}&bpctr=#{Time.new.epoch + 2000}&disable_polymer=1") | ||||||
|  |     html = XML.parse_html(html.body) | ||||||
|  | 
 | ||||||
|  |     html_channel.send(html) | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   spawn do | ||||||
|  |     client = make_client(YT_URL) | ||||||
|  |     info = client.get("/get_video_info?video_id=#{id}&el=detailpage&ps=default&eurl=&gl=US&hl=en&disable_polymer=1") | ||||||
|  |     info = HTTP::Params.parse(info.body) | ||||||
|  | 
 | ||||||
|  |     if info["reason"]? | ||||||
|  |       info = client.get("/get_video_info?video_id=#{id}&ps=default&eurl=&gl=US&hl=en&disable_polymer=1") | ||||||
|  |       info = HTTP::Params.parse(info.body) | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     info_channel.send(info) | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   html = html_channel.receive | ||||||
|  |   info = info_channel.receive | ||||||
|  | 
 | ||||||
|  |   if info["reason"]? | ||||||
|  |     raise info["reason"] | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   title = info["title"] | ||||||
|  |   views = info["view_count"].to_i64 | ||||||
|  |   author = info["author"] | ||||||
|  |   ucid = info["ucid"] | ||||||
|  | 
 | ||||||
|  |   likes = html.xpath_node(%q(//button[@title="I like this"]/span)) | ||||||
|  |   likes = likes.try &.content.delete(",").try &.to_i | ||||||
|  |   likes ||= 0 | ||||||
|  | 
 | ||||||
|  |   dislikes = html.xpath_node(%q(//button[@title="I dislike this"]/span)) | ||||||
|  |   dislikes = dislikes.try &.content.delete(",").try &.to_i | ||||||
|  |   dislikes ||= 0 | ||||||
|  | 
 | ||||||
|  |   description = html.xpath_node(%q(//p[@id="eow-description"])) | ||||||
|  |   description = description ? description.to_xml : "" | ||||||
|  | 
 | ||||||
|  |   wilson_score = ci_lower_bound(likes, likes + dislikes) | ||||||
|  | 
 | ||||||
|  |   published = html.xpath_node(%q(//meta[@itemprop="datePublished"])).not_nil!["content"] | ||||||
|  |   published = Time.parse(published, "%Y-%m-%d", Time::Location.local) | ||||||
|  | 
 | ||||||
|  |   allowed_regions = html.xpath_node(%q(//meta[@itemprop="regionsAllowed"])).not_nil!["content"].split(",") | ||||||
|  |   is_family_friendly = html.xpath_node(%q(//meta[@itemprop="isFamilyFriendly"])).not_nil!["content"] == "True" | ||||||
|  |   genre = html.xpath_node(%q(//meta[@itemprop="genre"])).not_nil!["content"] | ||||||
|  | 
 | ||||||
|  |   video = Video.new(id, info, Time.now, title, views, likes, dislikes, wilson_score, published, description, | ||||||
|  |     nil, author, ucid, allowed_regions, is_family_friendly, genre) | ||||||
|  | 
 | ||||||
|  |   return video | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | def itag_to_metadata(itag : String) | ||||||
|  |   # See https://github.com/rg3/youtube-dl/blob/master/youtube_dl/extractor/youtube.py#L380-#L476 | ||||||
|  |   formats = {"5"  => {"ext" => "flv", "width" => 400, "height" => 240, "acodec" => "mp3", "abr" => 64, "vcodec" => "h263"}, | ||||||
|  |              "6"  => {"ext" => "flv", "width" => 450, "height" => 270, "acodec" => "mp3", "abr" => 64, "vcodec" => "h263"}, | ||||||
|  |              "13" => {"ext" => "3gp", "acodec" => "aac", "vcodec" => "mp4v"}, | ||||||
|  |              "17" => {"ext" => "3gp", "width" => 176, "height" => 144, "acodec" => "aac", "abr" => 24, "vcodec" => "mp4v"}, | ||||||
|  |              "18" => {"ext" => "mp4", "width" => 640, "height" => 360, "acodec" => "aac", "abr" => 96, "vcodec" => "h264"}, | ||||||
|  |              "22" => {"ext" => "mp4", "width" => 1280, "height" => 720, "acodec" => "aac", "abr" => 192, "vcodec" => "h264"}, | ||||||
|  |              "34" => {"ext" => "flv", "width" => 640, "height" => 360, "acodec" => "aac", "abr" => 128, "vcodec" => "h264"}, | ||||||
|  |              "35" => {"ext" => "flv", "width" => 854, "height" => 480, "acodec" => "aac", "abr" => 128, "vcodec" => "h264"}, | ||||||
|  | 
 | ||||||
|  |              "36" => {"ext" => "3gp", "width" => 320, "acodec" => "aac", "vcodec" => "mp4v"}, | ||||||
|  |              "37" => {"ext" => "mp4", "width" => 1920, "height" => 1080, "acodec" => "aac", "abr" => 192, "vcodec" => "h264"}, | ||||||
|  |              "38" => {"ext" => "mp4", "width" => 4096, "height" => 3072, "acodec" => "aac", "abr" => 192, "vcodec" => "h264"}, | ||||||
|  |              "43" => {"ext" => "webm", "width" => 640, "height" => 360, "acodec" => "vorbis", "abr" => 128, "vcodec" => "vp8"}, | ||||||
|  |              "44" => {"ext" => "webm", "width" => 854, "height" => 480, "acodec" => "vorbis", "abr" => 128, "vcodec" => "vp8"}, | ||||||
|  |              "45" => {"ext" => "webm", "width" => 1280, "height" => 720, "acodec" => "vorbis", "abr" => 192, "vcodec" => "vp8"}, | ||||||
|  |              "46" => {"ext" => "webm", "width" => 1920, "height" => 1080, "acodec" => "vorbis", "abr" => 192, "vcodec" => "vp8"}, | ||||||
|  |              "59" => {"ext" => "mp4", "width" => 854, "height" => 480, "acodec" => "aac", "abr" => 128, "vcodec" => "h264"}, | ||||||
|  |              "78" => {"ext" => "mp4", "width" => 854, "height" => 480, "acodec" => "aac", "abr" => 128, "vcodec" => "h264"}, | ||||||
|  | 
 | ||||||
|  |              # 3D videos | ||||||
|  |              "82"  => {"ext" => "mp4", "height" => 360, "format" => "3D", "acodec" => "aac", "abr" => 128, "vcodec" => "h264"}, | ||||||
|  |              "83"  => {"ext" => "mp4", "height" => 480, "format" => "3D", "acodec" => "aac", "abr" => 128, "vcodec" => "h264"}, | ||||||
|  |              "84"  => {"ext" => "mp4", "height" => 720, "format" => "3D", "acodec" => "aac", "abr" => 192, "vcodec" => "h264"}, | ||||||
|  |              "85"  => {"ext" => "mp4", "height" => 1080, "format" => "3D", "acodec" => "aac", "abr" => 192, "vcodec" => "h264"}, | ||||||
|  |              "100" => {"ext" => "webm", "height" => 360, "format" => "3D", "acodec" => "vorbis", "abr" => 128, "vcodec" => "vp8"}, | ||||||
|  |              "101" => {"ext" => "webm", "height" => 480, "format" => "3D", "acodec" => "vorbis", "abr" => 192, "vcodec" => "vp8"}, | ||||||
|  |              "102" => {"ext" => "webm", "height" => 720, "format" => "3D", "acodec" => "vorbis", "abr" => 192, "vcodec" => "vp8"}, | ||||||
|  | 
 | ||||||
|  |              # Apple HTTP Live Streaming | ||||||
|  |              "91"  => {"ext" => "mp4", "height" => 144, "format" => "HLS", "acodec" => "aac", "abr" => 48, "vcodec" => "h264"}, | ||||||
|  |              "92"  => {"ext" => "mp4", "height" => 240, "format" => "HLS", "acodec" => "aac", "abr" => 48, "vcodec" => "h264"}, | ||||||
|  |              "93"  => {"ext" => "mp4", "height" => 360, "format" => "HLS", "acodec" => "aac", "abr" => 128, "vcodec" => "h264"}, | ||||||
|  |              "94"  => {"ext" => "mp4", "height" => 480, "format" => "HLS", "acodec" => "aac", "abr" => 128, "vcodec" => "h264"}, | ||||||
|  |              "95"  => {"ext" => "mp4", "height" => 720, "format" => "HLS", "acodec" => "aac", "abr" => 256, "vcodec" => "h264"}, | ||||||
|  |              "96"  => {"ext" => "mp4", "height" => 1080, "format" => "HLS", "acodec" => "aac", "abr" => 256, "vcodec" => "h264"}, | ||||||
|  |              "132" => {"ext" => "mp4", "height" => 240, "format" => "HLS", "acodec" => "aac", "abr" => 48, "vcodec" => "h264"}, | ||||||
|  |              "151" => {"ext" => "mp4", "height" => 72, "format" => "HLS", "acodec" => "aac", "abr" => 24, "vcodec" => "h264"}, | ||||||
|  | 
 | ||||||
|  |              # DASH mp4 video | ||||||
|  |              "133" => {"ext" => "mp4", "height" => 240, "format" => "DASH video", "vcodec" => "h264"}, | ||||||
|  |              "134" => {"ext" => "mp4", "height" => 360, "format" => "DASH video", "vcodec" => "h264"}, | ||||||
|  |              "135" => {"ext" => "mp4", "height" => 480, "format" => "DASH video", "vcodec" => "h264"}, | ||||||
|  |              "136" => {"ext" => "mp4", "height" => 720, "format" => "DASH video", "vcodec" => "h264"}, | ||||||
|  |              "137" => {"ext" => "mp4", "height" => 1080, "format" => "DASH video", "vcodec" => "h264"}, | ||||||
|  |              "138" => {"ext" => "mp4", "format" => "DASH video", "vcodec" => "h264"}, # Height can vary (https=>//github.com/rg3/youtube-dl/issues/4559) | ||||||
|  |              "160" => {"ext" => "mp4", "height" => 144, "format" => "DASH video", "vcodec" => "h264"}, | ||||||
|  |              "212" => {"ext" => "mp4", "height" => 480, "format" => "DASH video", "vcodec" => "h264"}, | ||||||
|  |              "264" => {"ext" => "mp4", "height" => 1440, "format" => "DASH video", "vcodec" => "h264"}, | ||||||
|  |              "298" => {"ext" => "mp4", "height" => 720, "format" => "DASH video", "vcodec" => "h264", "fps" => 60}, | ||||||
|  |              "299" => {"ext" => "mp4", "height" => 1080, "format" => "DASH video", "vcodec" => "h264", "fps" => 60}, | ||||||
|  |              "266" => {"ext" => "mp4", "height" => 2160, "format" => "DASH video", "vcodec" => "h264"}, | ||||||
|  | 
 | ||||||
|  |              # Dash mp4 audio | ||||||
|  |              "139" => {"ext" => "m4a", "format" => "DASH audio", "acodec" => "aac", "abr" => 48, "container" => "m4a_dash"}, | ||||||
|  |              "140" => {"ext" => "m4a", "format" => "DASH audio", "acodec" => "aac", "abr" => 128, "container" => "m4a_dash"}, | ||||||
|  |              "141" => {"ext" => "m4a", "format" => "DASH audio", "acodec" => "aac", "abr" => 256, "container" => "m4a_dash"}, | ||||||
|  |              "256" => {"ext" => "m4a", "format" => "DASH audio", "acodec" => "aac", "container" => "m4a_dash"}, | ||||||
|  |              "258" => {"ext" => "m4a", "format" => "DASH audio", "acodec" => "aac", "container" => "m4a_dash"}, | ||||||
|  |              "325" => {"ext" => "m4a", "format" => "DASH audio", "acodec" => "dtse", "container" => "m4a_dash"}, | ||||||
|  |              "328" => {"ext" => "m4a", "format" => "DASH audio", "acodec" => "ec-3", "container" => "m4a_dash"}, | ||||||
|  | 
 | ||||||
|  |              # Dash webm | ||||||
|  |              "167" => {"ext" => "webm", "height" => 360, "width" => 640, "format" => "DASH video", "container" => "webm", "vcodec" => "vp8"}, | ||||||
|  |              "168" => {"ext" => "webm", "height" => 480, "width" => 854, "format" => "DASH video", "container" => "webm", "vcodec" => "vp8"}, | ||||||
|  |              "169" => {"ext" => "webm", "height" => 720, "width" => 1280, "format" => "DASH video", "container" => "webm", "vcodec" => "vp8"}, | ||||||
|  |              "170" => {"ext" => "webm", "height" => 1080, "width" => 1920, "format" => "DASH video", "container" => "webm", "vcodec" => "vp8"}, | ||||||
|  |              "218" => {"ext" => "webm", "height" => 480, "width" => 854, "format" => "DASH video", "container" => "webm", "vcodec" => "vp8"}, | ||||||
|  |              "219" => {"ext" => "webm", "height" => 480, "width" => 854, "format" => "DASH video", "container" => "webm", "vcodec" => "vp8"}, | ||||||
|  |              "278" => {"ext" => "webm", "height" => 144, "format" => "DASH video", "container" => "webm", "vcodec" => "vp9"}, | ||||||
|  |              "242" => {"ext" => "webm", "height" => 240, "format" => "DASH video", "vcodec" => "vp9"}, | ||||||
|  |              "243" => {"ext" => "webm", "height" => 360, "format" => "DASH video", "vcodec" => "vp9"}, | ||||||
|  |              "244" => {"ext" => "webm", "height" => 480, "format" => "DASH video", "vcodec" => "vp9"}, | ||||||
|  |              "245" => {"ext" => "webm", "height" => 480, "format" => "DASH video", "vcodec" => "vp9"}, | ||||||
|  |              "246" => {"ext" => "webm", "height" => 480, "format" => "DASH video", "vcodec" => "vp9"}, | ||||||
|  |              "247" => {"ext" => "webm", "height" => 720, "format" => "DASH video", "vcodec" => "vp9"}, | ||||||
|  |              "248" => {"ext" => "webm", "height" => 1080, "format" => "DASH video", "vcodec" => "vp9"}, | ||||||
|  |              "271" => {"ext" => "webm", "height" => 1440, "format" => "DASH video", "vcodec" => "vp9"}, | ||||||
|  |              # itag 272 videos are either 3840x2160 (e.g. RtoitU2A-3E) or 7680x4320 (sLprVF6d7Ug) | ||||||
|  |              "272" => {"ext" => "webm", "height" => 2160, "format" => "DASH video", "vcodec" => "vp9"}, | ||||||
|  |              "302" => {"ext" => "webm", "height" => 720, "format" => "DASH video", "vcodec" => "vp9", "fps" => 60}, | ||||||
|  |              "303" => {"ext" => "webm", "height" => 1080, "format" => "DASH video", "vcodec" => "vp9", "fps" => 60}, | ||||||
|  |              "308" => {"ext" => "webm", "height" => 1440, "format" => "DASH video", "vcodec" => "vp9", "fps" => 60}, | ||||||
|  |              "313" => {"ext" => "webm", "height" => 2160, "format" => "DASH video", "vcodec" => "vp9"}, | ||||||
|  |              "315" => {"ext" => "webm", "height" => 2160, "format" => "DASH video", "vcodec" => "vp9", "fps" => 60}, | ||||||
|  | 
 | ||||||
|  |              # Dash webm audio | ||||||
|  |              "171" => {"ext" => "webm", "acodec" => "vorbis", "format" => "DASH audio", "abr" => 128}, | ||||||
|  |              "172" => {"ext" => "webm", "acodec" => "vorbis", "format" => "DASH audio", "abr" => 256}, | ||||||
|  | 
 | ||||||
|  |              # Dash webm audio with opus inside | ||||||
|  |              "249" => {"ext" => "webm", "format" => "DASH audio", "acodec" => "opus", "abr" => 50}, | ||||||
|  |              "250" => {"ext" => "webm", "format" => "DASH audio", "acodec" => "opus", "abr" => 70}, | ||||||
|  |              "251" => {"ext" => "webm", "format" => "DASH audio", "acodec" => "opus", "abr" => 160}, | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return formats[itag] | ||||||
|  | end | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue