use crate::definitions::SiteDefinition;
use anyhow::Result;
use fantoccini::ClientBuilder;
use scraper::{Html, Selector};
use tokio::runtime;
use crate::VIDEO;
fn get_video_info(video: &mut VIDEO, url: &str, webdriver_port: u16) -> Result<bool> {
if video.info.is_empty() {
let local_url = url.to_owned();
let rt = runtime::Builder::new_current_thread()
.enable_time()
.enable_io()
.build()
.unwrap();
rt.block_on(async move {
let webdriver_url = format!("http://localhost:{}", webdriver_port);
let c = ClientBuilder::native()
.connect(&webdriver_url)
.await
.expect("failed to connect to web driver");
c.goto(&local_url).await.expect("could not go to the URL");
let mut body = c.source().await.expect("could not read the site source");
let html = Html::parse_document(body.as_str());
let iframe_selector = Selector::parse(r#"iframe"#).unwrap();
let iframe_elem = html.select(&iframe_selector).next();
match iframe_elem {
Some(iframe) => {
c.goto(iframe.value().attr("src").unwrap())
.await
.expect("could not go to the URL");
body = c.source().await.expect("could not read the iframe source");
}
None => (),
};
video.info.push_str(body.as_str());
c.close_window().await.expect("could not close the window");
});
}
Ok(true)
}
struct ZZKTPlayerHandler;
impl SiteDefinition for ZZKTPlayerHandler {
fn can_handle_url<'a>(
&'a self,
video: &mut VIDEO,
url: &'a str,
webdriver_port: u16,
) -> Result<bool> {
let _ = get_video_info(video, url, webdriver_port)?;
let video_info_html = Html::parse_document(video.info.as_str());
let div_selector = Selector::parse(r#"div[id="kt_player"]"#).unwrap();
let div_elem = video_info_html.select(&div_selector).next();
match div_elem {
Some(_div) => Ok(true),
None => Ok(false),
}
}
fn is_playlist<'a>(&'a self, _url: &'a str, _webdriver_port: u16) -> Result<bool> {
Ok(false)
}
fn find_video_title<'a>(
&'a self,
video: &'a mut VIDEO,
url: &'a str,
webdriver_port: u16,
) -> Result<String> {
let _ = get_video_info(video, url, webdriver_port)?;
let video_info_html = Html::parse_document(video.info.as_str());
let title_selector = Selector::parse(r#"meta[property="og:title"]"#).unwrap();
let title_elem = video_info_html.select(&title_selector).next().unwrap();
let title_contents = title_elem.value().attr("content").unwrap();
Ok(title_contents.to_string())
}
fn find_video_direct_url<'a>(
&'a self,
video: &'a mut VIDEO,
url: &'a str,
_webdriver_port: u16,
_onlyaudio: bool,
) -> Result<String> {
let _ = get_video_info(video, url, _webdriver_port)?;
let video_info_html = Html::parse_document(video.info.as_str());
let url_selector = Selector::parse("video").unwrap();
let url_elem = video_info_html.select(&url_selector).next().unwrap();
let url_contents = url_elem.value().attr("src").unwrap();
Ok(url_contents.to_string())
}
fn does_video_exist<'a>(
&'a self,
video: &'a mut VIDEO,
url: &'a str,
webdriver_port: u16,
) -> Result<bool> {
let _ = get_video_info(video, url, webdriver_port);
Ok(!video.info.is_empty())
}
fn display_name<'a>(&'a self) -> String {
"Generic KT player".to_string()
}
fn find_video_file_extension<'a>(
&'a self,
_video: &'a mut VIDEO,
_url: &'a str,
_webdriver_port: u16,
_onlyaudio: bool,
) -> Result<String> {
Ok("mp4".to_string())
}
fn web_driver_required<'a>(&'a self) -> bool {
true
}
}
inventory::submit! {
&ZZKTPlayerHandler as &dyn SiteDefinition
}