mirror of
https://github.com/neon-mmd/websurfx.git
synced 2024-11-21 21:48:21 -05:00
Merge pull request #104 from xffxff/error_stack
improve error handling by using `error-stack` crate
This commit is contained in:
commit
42686b92f3
17
Cargo.lock
generated
17
Cargo.lock
generated
@ -268,6 +268,12 @@ dependencies = [
|
|||||||
"alloc-no-stdlib",
|
"alloc-no-stdlib",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "anyhow"
|
||||||
|
version = "1.0.71"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "9c7d0618f0e0b7e8ff11427422b64564d5fb0be1940354bfe2e0529b18a9d9b8"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "askama_escape"
|
name = "askama_escape"
|
||||||
version = "0.10.3"
|
version = "0.10.3"
|
||||||
@ -733,6 +739,16 @@ dependencies = [
|
|||||||
"libc",
|
"libc",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "error-stack"
|
||||||
|
version = "0.3.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "5f00447f331c7f726db5b8532ebc9163519eed03c6d7c8b73c90b3ff5646ac85"
|
||||||
|
dependencies = [
|
||||||
|
"anyhow",
|
||||||
|
"rustc_version 0.4.0",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "failure"
|
name = "failure"
|
||||||
version = "0.1.8"
|
version = "0.1.8"
|
||||||
@ -3373,6 +3389,7 @@ dependencies = [
|
|||||||
"actix-files",
|
"actix-files",
|
||||||
"actix-web",
|
"actix-web",
|
||||||
"env_logger",
|
"env_logger",
|
||||||
|
"error-stack",
|
||||||
"fake-useragent",
|
"fake-useragent",
|
||||||
"handlebars",
|
"handlebars",
|
||||||
"log",
|
"log",
|
||||||
|
@ -23,6 +23,7 @@ redis = {version="*"}
|
|||||||
md5 = {version="*"}
|
md5 = {version="*"}
|
||||||
rand={version="*"}
|
rand={version="*"}
|
||||||
once_cell = {version="*"}
|
once_cell = {version="*"}
|
||||||
|
error-stack = "0.3.1"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
rusty-hook = "^0.11.2"
|
rusty-hook = "^0.11.2"
|
||||||
|
@ -9,7 +9,9 @@ use scraper::{Html, Selector};
|
|||||||
|
|
||||||
use crate::search_results_handler::aggregation_models::RawSearchResult;
|
use crate::search_results_handler::aggregation_models::RawSearchResult;
|
||||||
|
|
||||||
use super::engine_models::EngineErrorKind;
|
use super::engine_models::EngineError;
|
||||||
|
|
||||||
|
use error_stack::{IntoReport, Report, Result, ResultExt};
|
||||||
|
|
||||||
/// This function scrapes results from the upstream engine duckduckgo and puts all the scraped
|
/// This function scrapes results from the upstream engine duckduckgo and puts all the scraped
|
||||||
/// results like title, visiting_url (href in html),engine (from which engine it was fetched from)
|
/// results like title, visiting_url (href in html),engine (from which engine it was fetched from)
|
||||||
@ -32,7 +34,7 @@ pub async fn results(
|
|||||||
query: &str,
|
query: &str,
|
||||||
page: u32,
|
page: u32,
|
||||||
user_agent: &str,
|
user_agent: &str,
|
||||||
) -> Result<HashMap<String, RawSearchResult>, EngineErrorKind> {
|
) -> Result<HashMap<String, RawSearchResult>, EngineError> {
|
||||||
// Page number can be missing or empty string and so appropriate handling is required
|
// Page number can be missing or empty string and so appropriate handling is required
|
||||||
// so that upstream server recieves valid page number.
|
// so that upstream server recieves valid page number.
|
||||||
let url: String = match page {
|
let url: String = match page {
|
||||||
@ -51,33 +53,71 @@ pub async fn results(
|
|||||||
|
|
||||||
// initializing HeaderMap and adding appropriate headers.
|
// initializing HeaderMap and adding appropriate headers.
|
||||||
let mut header_map = HeaderMap::new();
|
let mut header_map = HeaderMap::new();
|
||||||
header_map.insert(USER_AGENT, user_agent.parse()?);
|
header_map.insert(
|
||||||
header_map.insert(REFERER, "https://google.com/".parse()?);
|
USER_AGENT,
|
||||||
header_map.insert(CONTENT_TYPE, "application/x-www-form-urlencoded".parse()?);
|
user_agent
|
||||||
header_map.insert(COOKIE, "kl=wt-wt".parse()?);
|
.parse()
|
||||||
|
.into_report()
|
||||||
|
.change_context(EngineError::UnexpectedError)?,
|
||||||
|
);
|
||||||
|
header_map.insert(
|
||||||
|
REFERER,
|
||||||
|
"https://google.com/"
|
||||||
|
.parse()
|
||||||
|
.into_report()
|
||||||
|
.change_context(EngineError::UnexpectedError)?,
|
||||||
|
);
|
||||||
|
header_map.insert(
|
||||||
|
CONTENT_TYPE,
|
||||||
|
"application/x-www-form-urlencoded"
|
||||||
|
.parse()
|
||||||
|
.into_report()
|
||||||
|
.change_context(EngineError::UnexpectedError)?,
|
||||||
|
);
|
||||||
|
header_map.insert(
|
||||||
|
COOKIE,
|
||||||
|
"kl=wt-wt"
|
||||||
|
.parse()
|
||||||
|
.into_report()
|
||||||
|
.change_context(EngineError::UnexpectedError)?,
|
||||||
|
);
|
||||||
|
|
||||||
// fetch the html from upstream duckduckgo engine
|
// fetch the html from upstream duckduckgo engine
|
||||||
let results: String = reqwest::Client::new()
|
let results: String = reqwest::Client::new()
|
||||||
.get(url)
|
.get(url)
|
||||||
.timeout(Duration::from_secs(30))
|
.timeout(Duration::from_secs(5))
|
||||||
.headers(header_map) // add spoofed headers to emulate human behaviour
|
.headers(header_map) // add spoofed headers to emulate human behaviour
|
||||||
.send()
|
.send()
|
||||||
.await?
|
.await
|
||||||
|
.into_report()
|
||||||
|
.change_context(EngineError::RequestError)?
|
||||||
.text()
|
.text()
|
||||||
.await?;
|
.await
|
||||||
|
.into_report()
|
||||||
|
.change_context(EngineError::RequestError)?;
|
||||||
|
|
||||||
let document: Html = Html::parse_document(&results);
|
let document: Html = Html::parse_document(&results);
|
||||||
|
|
||||||
let no_result: Selector = Selector::parse(".no-results")?;
|
let no_result: Selector = Selector::parse(".no-results")
|
||||||
|
.map_err(|_| Report::new(EngineError::UnexpectedError))
|
||||||
|
.attach_printable_lazy(|| format!("invalid CSS selector: {}", ".no-results"))?;
|
||||||
|
|
||||||
if document.select(&no_result).next().is_some() {
|
if document.select(&no_result).next().is_some() {
|
||||||
return Err(EngineErrorKind::EmptyResultSet);
|
return Err(Report::new(EngineError::EmptyResultSet));
|
||||||
}
|
}
|
||||||
|
|
||||||
let results: Selector = Selector::parse(".result")?;
|
let results: Selector = Selector::parse(".result")
|
||||||
let result_title: Selector = Selector::parse(".result__a")?;
|
.map_err(|_| Report::new(EngineError::UnexpectedError))
|
||||||
let result_url: Selector = Selector::parse(".result__url")?;
|
.attach_printable_lazy(|| format!("invalid CSS selector: {}", ".result"))?;
|
||||||
let result_desc: Selector = Selector::parse(".result__snippet")?;
|
let result_title: Selector = Selector::parse(".result__a")
|
||||||
|
.map_err(|_| Report::new(EngineError::UnexpectedError))
|
||||||
|
.attach_printable_lazy(|| format!("invalid CSS selector: {}", ".result__a"))?;
|
||||||
|
let result_url: Selector = Selector::parse(".result__url")
|
||||||
|
.map_err(|_| Report::new(EngineError::UnexpectedError))
|
||||||
|
.attach_printable_lazy(|| format!("invalid CSS selector: {}", ".result__url"))?;
|
||||||
|
let result_desc: Selector = Selector::parse(".result__snippet")
|
||||||
|
.map_err(|_| Report::new(EngineError::UnexpectedError))
|
||||||
|
.attach_printable_lazy(|| format!("invalid CSS selector: {}", ".result__snippet"))?;
|
||||||
|
|
||||||
// scrape all the results from the html
|
// scrape all the results from the html
|
||||||
Ok(document
|
Ok(document
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
//! This module provides the error enum to handle different errors associated while requesting data from
|
//! This module provides the error enum to handle different errors associated while requesting data from
|
||||||
//! the upstream search engines with the search query provided by the user.
|
//! the upstream search engines with the search query provided by the user.
|
||||||
|
|
||||||
use reqwest::header::InvalidHeaderValue;
|
use error_stack::Context;
|
||||||
use scraper::error::SelectorErrorKind;
|
use std::fmt;
|
||||||
|
|
||||||
/// A custom error type used for handle engine associated errors.
|
/// A custom error type used for handle engine associated errors.
|
||||||
///
|
///
|
||||||
@ -15,73 +15,29 @@ use scraper::error::SelectorErrorKind;
|
|||||||
/// and are errors mostly related to failure in initialization of HeaderMap, Selector errors and
|
/// and are errors mostly related to failure in initialization of HeaderMap, Selector errors and
|
||||||
/// all other errors occuring within the code handling the `upstream search engines`.
|
/// all other errors occuring within the code handling the `upstream search engines`.
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum EngineErrorKind {
|
pub enum EngineError {
|
||||||
RequestError(reqwest::Error),
|
|
||||||
EmptyResultSet,
|
EmptyResultSet,
|
||||||
UnexpectedError {
|
RequestError,
|
||||||
message: String,
|
UnexpectedError,
|
||||||
source: Option<Box<dyn std::error::Error>>,
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Implementing `Display` trait to make errors writable on the stdout and also providing/passing the
|
impl fmt::Display for EngineError {
|
||||||
/// appropriate errors that should be written to the stdout when this error is raised/encountered.
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
impl std::fmt::Display for EngineErrorKind {
|
|
||||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
||||||
match self {
|
match self {
|
||||||
EngineErrorKind::RequestError(request_error) => {
|
EngineError::EmptyResultSet => {
|
||||||
write!(f, "Request error: {}", request_error)
|
|
||||||
}
|
|
||||||
EngineErrorKind::EmptyResultSet => {
|
|
||||||
write!(f, "The upstream search engine returned an empty result set")
|
write!(f, "The upstream search engine returned an empty result set")
|
||||||
}
|
}
|
||||||
EngineErrorKind::UnexpectedError { message, source } => {
|
EngineError::RequestError => {
|
||||||
write!(f, "Unexpected error: {}", message)?;
|
write!(
|
||||||
if let Some(source) = source {
|
f,
|
||||||
write!(f, "\nCaused by: {}", source)?;
|
"Error occurred while requesting data from upstream search engine"
|
||||||
}
|
)
|
||||||
Ok(())
|
}
|
||||||
|
EngineError::UnexpectedError => {
|
||||||
|
write!(f, "An unexpected error occurred while processing the data")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Implementing `Error` trait to make the the `EngineErrorKind` enum an error type and
|
impl Context for EngineError {}
|
||||||
/// mapping `ReqwestErrors` to `RequestError` and `UnexpectedError` errors to all other unexpected
|
|
||||||
/// errors ocurring within the code handling the upstream search engines.
|
|
||||||
impl std::error::Error for EngineErrorKind {
|
|
||||||
fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {
|
|
||||||
match self {
|
|
||||||
EngineErrorKind::RequestError(request_error) => Some(request_error),
|
|
||||||
EngineErrorKind::UnexpectedError { source, .. } => source.as_deref().map(|s| s),
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Implementing `From` trait to map the `SelectorErrorKind` to `UnexpectedError` variant.
|
|
||||||
impl From<SelectorErrorKind<'_>> for EngineErrorKind {
|
|
||||||
fn from(err: SelectorErrorKind<'_>) -> Self {
|
|
||||||
Self::UnexpectedError {
|
|
||||||
message: err.to_string(),
|
|
||||||
source: None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Implementing `From` trait to map the `InvalidHeaderValue` to `UnexpectedError` variant.
|
|
||||||
impl From<InvalidHeaderValue> for EngineErrorKind {
|
|
||||||
fn from(err: InvalidHeaderValue) -> Self {
|
|
||||||
Self::UnexpectedError {
|
|
||||||
message: err.to_string(),
|
|
||||||
source: Some(Box::new(err)),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Implementing `From` trait to map all `reqwest::Error` to `UnexpectedError` variant.
|
|
||||||
impl From<reqwest::Error> for EngineErrorKind {
|
|
||||||
fn from(err: reqwest::Error) -> Self {
|
|
||||||
Self::RequestError(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -8,7 +8,8 @@ use std::collections::HashMap;
|
|||||||
|
|
||||||
use crate::search_results_handler::aggregation_models::RawSearchResult;
|
use crate::search_results_handler::aggregation_models::RawSearchResult;
|
||||||
|
|
||||||
use super::engine_models::EngineErrorKind;
|
use super::engine_models::EngineError;
|
||||||
|
use error_stack::{IntoReport, Report, Result, ResultExt};
|
||||||
|
|
||||||
/// This function scrapes results from the upstream engine duckduckgo and puts all the scraped
|
/// This function scrapes results from the upstream engine duckduckgo and puts all the scraped
|
||||||
/// results like title, visiting_url (href in html),engine (from which engine it was fetched from)
|
/// results like title, visiting_url (href in html),engine (from which engine it was fetched from)
|
||||||
@ -31,43 +32,76 @@ pub async fn results(
|
|||||||
query: &str,
|
query: &str,
|
||||||
page: u32,
|
page: u32,
|
||||||
user_agent: &str,
|
user_agent: &str,
|
||||||
) -> Result<HashMap<String, RawSearchResult>, EngineErrorKind> {
|
) -> Result<HashMap<String, RawSearchResult>, EngineError> {
|
||||||
// Page number can be missing or empty string and so appropriate handling is required
|
// Page number can be missing or empty string and so appropriate handling is required
|
||||||
// so that upstream server recieves valid page number.
|
// so that upstream server recieves valid page number.
|
||||||
let url: String = format!("https://searx.work/search?q={query}&pageno={page}");
|
let url: String = format!("https://searx.work/search?q={query}&pageno={page}");
|
||||||
|
|
||||||
// initializing headers and adding appropriate headers.
|
// initializing headers and adding appropriate headers.
|
||||||
let mut header_map = HeaderMap::new();
|
let mut header_map = HeaderMap::new();
|
||||||
header_map.insert(USER_AGENT, user_agent.parse()?);
|
header_map.insert(
|
||||||
header_map.insert(REFERER, "https://google.com/".parse()?);
|
USER_AGENT,
|
||||||
header_map.insert(CONTENT_TYPE, "application/x-www-form-urlencoded".parse()?);
|
user_agent
|
||||||
header_map.insert(COOKIE, "categories=general; language=auto; locale=en; autocomplete=duckduckgo; image_proxy=1; method=POST; safesearch=2; theme=simple; results_on_new_tab=1; doi_resolver=oadoi.org; simple_style=auto; center_alignment=1; query_in_title=1; infinite_scroll=0; disabled_engines=; enabled_engines=\"archive is__general\\054yep__general\\054curlie__general\\054currency__general\\054ddg definitions__general\\054wikidata__general\\054duckduckgo__general\\054tineye__general\\054lingva__general\\054startpage__general\\054yahoo__general\\054wiby__general\\054marginalia__general\\054alexandria__general\\054wikibooks__general\\054wikiquote__general\\054wikisource__general\\054wikiversity__general\\054wikivoyage__general\\054dictzone__general\\054seznam__general\\054mojeek__general\\054naver__general\\054wikimini__general\\054brave__general\\054petalsearch__general\\054goo__general\"; disabled_plugins=; enabled_plugins=\"searx.plugins.hostname_replace\\054searx.plugins.oa_doi_rewrite\\054searx.plugins.vim_hotkeys\"; tokens=; maintab=on; enginetab=on".parse()?);
|
.parse()
|
||||||
|
.into_report()
|
||||||
|
.change_context(EngineError::UnexpectedError)?,
|
||||||
|
);
|
||||||
|
header_map.insert(
|
||||||
|
REFERER,
|
||||||
|
"https://google.com/"
|
||||||
|
.parse()
|
||||||
|
.into_report()
|
||||||
|
.change_context(EngineError::UnexpectedError)?,
|
||||||
|
);
|
||||||
|
header_map.insert(
|
||||||
|
CONTENT_TYPE,
|
||||||
|
"application/x-www-form-urlencoded"
|
||||||
|
.parse()
|
||||||
|
.into_report()
|
||||||
|
.change_context(EngineError::UnexpectedError)?,
|
||||||
|
);
|
||||||
|
header_map.insert(COOKIE, "categories=general; language=auto; locale=en; autocomplete=duckduckgo; image_proxy=1; method=POST; safesearch=2; theme=simple; results_on_new_tab=1; doi_resolver=oadoi.org; simple_style=auto; center_alignment=1; query_in_title=1; infinite_scroll=0; disabled_engines=; enabled_engines=\"archive is__general\\054yep__general\\054curlie__general\\054currency__general\\054ddg definitions__general\\054wikidata__general\\054duckduckgo__general\\054tineye__general\\054lingva__general\\054startpage__general\\054yahoo__general\\054wiby__general\\054marginalia__general\\054alexandria__general\\054wikibooks__general\\054wikiquote__general\\054wikisource__general\\054wikiversity__general\\054wikivoyage__general\\054dictzone__general\\054seznam__general\\054mojeek__general\\054naver__general\\054wikimini__general\\054brave__general\\054petalsearch__general\\054goo__general\"; disabled_plugins=; enabled_plugins=\"searx.plugins.hostname_replace\\054searx.plugins.oa_doi_rewrite\\054searx.plugins.vim_hotkeys\"; tokens=; maintab=on; enginetab=on".parse().into_report().change_context(EngineError::UnexpectedError)?);
|
||||||
|
|
||||||
// fetch the html from upstream searx instance engine
|
// fetch the html from upstream searx instance engine
|
||||||
let results: String = reqwest::Client::new()
|
let results: String = reqwest::Client::new()
|
||||||
.get(url)
|
.get(url)
|
||||||
.headers(header_map) // add spoofed headers to emulate human behaviours.
|
.headers(header_map) // add spoofed headers to emulate human behaviours.
|
||||||
.send()
|
.send()
|
||||||
.await?
|
.await
|
||||||
|
.into_report()
|
||||||
|
.change_context(EngineError::RequestError)?
|
||||||
.text()
|
.text()
|
||||||
.await?;
|
.await
|
||||||
|
.into_report()
|
||||||
|
.change_context(EngineError::RequestError)?;
|
||||||
|
|
||||||
let document: Html = Html::parse_document(&results);
|
let document: Html = Html::parse_document(&results);
|
||||||
|
|
||||||
let no_result: Selector = Selector::parse("#urls>.dialog-error>p")?;
|
let no_result: Selector = Selector::parse("#urls>.dialog-error>p")
|
||||||
|
.map_err(|_| Report::new(EngineError::UnexpectedError))
|
||||||
|
.attach_printable_lazy(|| format!("invalid CSS selector: {}", "#urls>.dialog-error>p"))?;
|
||||||
|
|
||||||
if let Some(no_result_msg) = document.select(&no_result).nth(1) {
|
if let Some(no_result_msg) = document.select(&no_result).nth(1) {
|
||||||
if no_result_msg.inner_html()
|
if no_result_msg.inner_html()
|
||||||
== "we didn't find any results. Please use another query or search in more categories"
|
== "we didn't find any results. Please use another query or search in more categories"
|
||||||
{
|
{
|
||||||
return Err(EngineErrorKind::EmptyResultSet);
|
return Err(Report::new(EngineError::EmptyResultSet));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let results: Selector = Selector::parse(".result")?;
|
let results: Selector = Selector::parse(".result")
|
||||||
let result_title: Selector = Selector::parse("h3>a")?;
|
.map_err(|_| Report::new(EngineError::UnexpectedError))
|
||||||
let result_url: Selector = Selector::parse("h3>a")?;
|
.attach_printable_lazy(|| format!("invalid CSS selector: {}", ".result"))?;
|
||||||
let result_desc: Selector = Selector::parse(".content")?;
|
let result_title: Selector = Selector::parse("h3>a")
|
||||||
|
.map_err(|_| Report::new(EngineError::UnexpectedError))
|
||||||
|
.attach_printable_lazy(|| format!("invalid CSS selector: {}", "h3>a"))?;
|
||||||
|
let result_url: Selector = Selector::parse("h3>a")
|
||||||
|
.map_err(|_| Report::new(EngineError::UnexpectedError))
|
||||||
|
.attach_printable_lazy(|| format!("invalid CSS selector: {}", "h3>a"))?;
|
||||||
|
|
||||||
|
let result_desc: Selector = Selector::parse(".content")
|
||||||
|
.map_err(|_| Report::new(EngineError::UnexpectedError))
|
||||||
|
.attach_printable_lazy(|| format!("invalid CSS selector: {}", ".content"))?;
|
||||||
|
|
||||||
// scrape all the results from the html
|
// scrape all the results from the html
|
||||||
Ok(document
|
Ok(document
|
||||||
|
@ -58,8 +58,19 @@ pub async fn aggregate(
|
|||||||
searx::results(query, page, &user_agent)
|
searx::results(query, page, &user_agent)
|
||||||
);
|
);
|
||||||
|
|
||||||
let ddg_map_results: HashMap<String, RawSearchResult> = ddg_map_results?;
|
let ddg_map_results = ddg_map_results.unwrap_or_else(|e| {
|
||||||
let searx_map_results: HashMap<String, RawSearchResult> = searx_map_results?;
|
if debug {
|
||||||
|
log::error!("Error fetching results from DuckDuckGo: {:?}", e);
|
||||||
|
}
|
||||||
|
HashMap::new()
|
||||||
|
});
|
||||||
|
|
||||||
|
let searx_map_results = searx_map_results.unwrap_or_else(|e| {
|
||||||
|
if debug {
|
||||||
|
log::error!("Error fetching results from Searx: {:?}", e);
|
||||||
|
}
|
||||||
|
HashMap::new()
|
||||||
|
});
|
||||||
|
|
||||||
result_map.extend(ddg_map_results);
|
result_map.extend(ddg_map_results);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user