2020-01-24 17:02:28 -05:00
|
|
|
alias SigProc = Proc(Array(String), Int32, Array(String))
|
|
|
|
|
2020-09-27 10:19:44 -07:00
|
|
|
struct DecryptFunction
|
|
|
|
@decrypt_function = [] of {SigProc, Int32}
|
|
|
|
@decrypt_time = Time.monotonic
|
|
|
|
|
|
|
|
def initialize(@use_polling = true)
|
2018-08-04 15:30:44 -05:00
|
|
|
end
|
|
|
|
|
2020-09-27 10:19:44 -07:00
|
|
|
def update_decrypt_function
|
|
|
|
@decrypt_function = fetch_decrypt_function
|
|
|
|
end
|
2018-08-04 15:30:44 -05:00
|
|
|
|
2020-09-27 10:19:44 -07:00
|
|
|
private def fetch_decrypt_function(id = "CvFH_6DNRCY")
|
|
|
|
document = YT_POOL.client &.get("/watch?v=#{id}&gl=US&hl=en").body
|
|
|
|
url = document.match(/src="(?<url>\/s\/player\/[^\/]+\/player_ias[^\/]+\/en_US\/base.js)"/).not_nil!["url"]
|
|
|
|
player = YT_POOL.client &.get(url).body
|
2018-08-04 15:30:44 -05:00
|
|
|
|
2020-09-27 10:19:44 -07:00
|
|
|
function_name = player.match(/^(?<name>[^=]+)=function\(\w\){\w=\w\.split\(""\);[^\. ]+\.[^( ]+/m).not_nil!["name"]
|
|
|
|
function_body = player.match(/^#{Regex.escape(function_name)}=function\(\w\){(?<body>[^}]+)}/m).not_nil!["body"]
|
|
|
|
function_body = function_body.split(";")[1..-2]
|
2018-08-04 15:30:44 -05:00
|
|
|
|
2020-09-27 10:19:44 -07:00
|
|
|
var_name = function_body[0][0, 2]
|
|
|
|
var_body = player.delete("\n").match(/var #{Regex.escape(var_name)}={(?<body>(.*?))};/).not_nil!["body"]
|
|
|
|
|
|
|
|
operations = {} of String => SigProc
|
|
|
|
var_body.split("},").each do |operation|
|
|
|
|
op_name = operation.match(/^[^:]+/).not_nil![0]
|
|
|
|
op_body = operation.match(/\{[^}]+/).not_nil![0]
|
|
|
|
|
|
|
|
case op_body
|
|
|
|
when "{a.reverse()"
|
|
|
|
operations[op_name] = ->(a : Array(String), b : Int32) { a.reverse }
|
|
|
|
when "{a.splice(0,b)"
|
|
|
|
operations[op_name] = ->(a : Array(String), b : Int32) { a.delete_at(0..(b - 1)); a }
|
|
|
|
else
|
|
|
|
operations[op_name] = ->(a : Array(String), b : Int32) { c = a[0]; a[0] = a[b % a.size]; a[b % a.size] = c; a }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
decrypt_function = [] of {SigProc, Int32}
|
|
|
|
function_body.each do |function|
|
|
|
|
function = function.lchop(var_name).delete("[].")
|
2018-08-04 15:30:44 -05:00
|
|
|
|
2020-09-27 10:19:44 -07:00
|
|
|
op_name = function.match(/[^\(]+/).not_nil![0]
|
|
|
|
value = function.match(/\(\w,(?<value>[\d]+)\)/).not_nil!["value"].to_i
|
2018-08-04 15:30:44 -05:00
|
|
|
|
2020-09-27 10:19:44 -07:00
|
|
|
decrypt_function << {operations[op_name], value}
|
|
|
|
end
|
|
|
|
|
|
|
|
return decrypt_function
|
2018-08-04 15:30:44 -05:00
|
|
|
end
|
|
|
|
|
2020-09-27 10:19:44 -07:00
|
|
|
def decrypt_signature(fmt : Hash(String, JSON::Any))
|
|
|
|
return "" if !fmt["s"]? || !fmt["sp"]?
|
|
|
|
|
|
|
|
sp = fmt["sp"].as_s
|
|
|
|
sig = fmt["s"].as_s.split("")
|
|
|
|
if !@use_polling
|
|
|
|
now = Time.monotonic
|
|
|
|
if now - @decrypt_time > 60.seconds || @decrypt_function.size == 0
|
|
|
|
@decrypt_function = fetch_decrypt_function
|
|
|
|
@decrypt_time = Time.monotonic
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
@decrypt_function.each do |proc, value|
|
|
|
|
sig = proc.call(sig, value)
|
|
|
|
end
|
|
|
|
|
|
|
|
return "&#{sp}=#{sig.join("")}"
|
|
|
|
end
|
2018-08-04 15:30:44 -05:00
|
|
|
end
|