extern crate reqwest; extern crate serde_json; extern crate cursive; extern crate regex; extern crate urlencoding; use cursive::theme::Effect; use cursive::utils::markup::StyledString; use cursive::Cursive; use cursive::views::Dialog; use serde_json::Value; use self::regex::Regex; pub fn query_url_gen(title: &str) -> String { // query config let mut url = String::from("https://en.wikipedia.org"); url.push_str("/w/api.php?"); url.push_str("action=query&"); url.push_str("format=json&"); url.push_str("prop=extracts%7Clinks&"); url.push_str("indexpageids=1&"); url.push_str("titles="); url.push_str(&urlencoding::encode(title)); url.push_str("&"); url.push_str("redirects=1&"); url.push_str("pllimit=40&"); url.push_str("explaintext=1"); url } pub fn search_url_gen(search: &str) -> String { // search config search.replace(" ", "%20"); let mut url = String::from("https://en.wikipedia.org"); url.push_str("/w/api.php?"); url.push_str("action=opensearch&"); url.push_str("format=json&"); url.push_str("search="); url.push_str(&urlencoding::encode(search)); url.push_str("&"); url.push_str("limit=20"); url } pub fn get_extract(v: &Value) -> Result { let pageid = &v["query"]["pageids"][0]; let pageid_str = match pageid { Value::String(id) => id, _ => panic!("wut"), }; match &v["query"]["pages"][pageid_str]["extract"] { Value::String(extract) => { // format to plain text extract.replace("\\\\", "\\"); Ok(format!("{}", extract)) } // ignore non strings _ => Ok(format!("This page has been deleted or moved")) } } pub fn extract_formatter(extract: String) -> StyledString { let mut formatted = StyledString::new(); let heading= Regex::new(r"^== (?P.*) ==$").unwrap(); let subheading= Regex::new(r"^=== (?P.*) ===$").unwrap(); let subsubheading= Regex::new(r"^==== (?P.*) ====$").unwrap(); for line in extract.lines() { if heading.is_match(line) { formatted.append( StyledString::styled( heading.replace(line, "$d"), Effect::Bold ) ); } else if subheading.is_match(line) { formatted.append( StyledString::styled( subheading.replace(line, "$d"), Effect::Italic ) ); } else if subsubheading.is_match(line) { formatted.append( StyledString::styled( subsubheading.replace(line, "$d"), Effect::Underline ) ); } else { formatted.append(StyledString::plain(line)); } formatted.append(StyledString::plain("\n")) } formatted } pub fn get_search_results(search: &str) -> Result, reqwest::Error> { let url = search_url_gen(search); let mut res = reqwest::get(&url[..])?; let v: Value = serde_json::from_str(&res.text().unwrap()) .unwrap_or_else( |e| { panic!("Recieved error {:?}", e); } ); let mut results: Vec = vec![]; for item in v[1].as_array().unwrap() { match item { Value::String(x) => results.push(x.to_string()), // ignore non strings _ => (), } } Ok(results) } pub fn get_links(v: &Value) -> Result, reqwest::Error> { let pageid = &v["query"]["pageids"][0]; let pageid_str = match pageid { Value::String(id) => id, _ => panic!("wut"), }; let mut links = vec![]; match &v["query"]["pages"][pageid_str]["links"] { Value::Array(arr) => { for item in arr { match item["title"] { Value::String(ref title) => links.push(title.to_string()), _ => links.push(String::from("lol")) } } }, _ => links.push(String::from("lol")) }; Ok(links) } pub fn pop_error(s: &mut Cursive, msg: String) { s.add_layer(Dialog::text(format!("{}", msg)) .button("Ok", |s| s.quit())); } pub fn handler(e: reqwest::Error) -> String { let mut msg: String = String::new(); if e.is_http() { match e.url() { None => msg.push_str(&format!("No URL given")), Some(url) => msg.push_str(&format!("Problem making request to: {}", url)), } } if e.is_redirect() { msg.push_str(&format!("server redirecting too many times or making loop")); } msg }