feat: move ArticleLocation to defined errors
All checks were successful
continuous-integration/drone/push Build is passing
All checks were successful
continuous-integration/drone/push Build is passing
Previously, the functions in article_location.rs where returning generic anyhow::Result. In order to ease error handling when using the library, it have been moved to specific errors.
This commit is contained in:
parent
2e6aed97ef
commit
b1d025a23c
@ -1,6 +1,6 @@
|
||||
---
|
||||
name: "Bug report"
|
||||
about: "This template is for reporting a bug"
|
||||
name: "🐛 Bug report"
|
||||
about: "For reporting bugs"
|
||||
title: ""
|
||||
labels:
|
||||
- "type::bug"
|
||||
|
@ -1,5 +1,5 @@
|
||||
---
|
||||
name: "Design discussion"
|
||||
name: "🗣 Design discussion"
|
||||
about: "For discussion about the design of features in the application, when there are several possibilities for implementation"
|
||||
title: ""
|
||||
labels:
|
||||
|
@ -1,6 +1,6 @@
|
||||
---
|
||||
name: "Feature request"
|
||||
about: "This template is for requesting a new feature"
|
||||
name: "💡 Feature request"
|
||||
about: "For requesting a new feature, with an implementation plan"
|
||||
title: ""
|
||||
labels:
|
||||
- "type::enhancement"
|
||||
@ -11,4 +11,4 @@ labels:
|
||||
|
||||
*describe what you would like to be able to do, or what solution you would like*
|
||||
|
||||
*(optional) additional context, comments or implementation propositions*
|
||||
*(optional) additional context, comments
|
||||
|
@ -1,5 +1,5 @@
|
||||
---
|
||||
name: "Ask a question"
|
||||
name: "❓ Ask a question"
|
||||
about: "If you have a question about the usage of the libraries or the tool"
|
||||
title: ""
|
||||
labels:
|
||||
|
@ -1,5 +1,5 @@
|
||||
---
|
||||
name: "Refactor"
|
||||
name: "🚧 Refactor"
|
||||
about: "For refactoring propositions"
|
||||
title: ""
|
||||
labels:
|
||||
|
@ -46,18 +46,18 @@ impl Builder {
|
||||
user: &impl AsRef<str>,
|
||||
password: &impl AsRef<str>,
|
||||
) -> &mut Self {
|
||||
self.user = String::from(user.as_ref());
|
||||
self.password = String::from(password.as_ref());
|
||||
self.user = user.as_ref().into();
|
||||
self.password = password.as_ref().into();
|
||||
self
|
||||
}
|
||||
|
||||
pub(crate) fn homeserver(&mut self, homeserver: &impl AsRef<str>) -> &mut Self {
|
||||
self.homeserver = String::from(homeserver.as_ref());
|
||||
self.homeserver = homeserver.as_ref().into();
|
||||
self
|
||||
}
|
||||
|
||||
pub(crate) fn room(&mut self, room: &impl AsRef<str>) -> &mut Self {
|
||||
self.room = String::from(room.as_ref());
|
||||
self.room = room.as_ref().into();
|
||||
self
|
||||
}
|
||||
}
|
||||
|
@ -1,8 +1,7 @@
|
||||
use std::convert::TryInto;
|
||||
use std::env;
|
||||
|
||||
use anyhow::Result;
|
||||
use log::info;
|
||||
use log::{error, info};
|
||||
use matrix_sdk::{
|
||||
self, async_trait,
|
||||
events::{
|
||||
@ -13,7 +12,7 @@ use matrix_sdk::{
|
||||
Client, ClientConfig, EventHandler, SyncSettings,
|
||||
};
|
||||
|
||||
use crieur_retrieve::{ArticleLocation, Url};
|
||||
use crieur_retrieve::{article_location::Error, article_location::Result, ArticleLocation, Url};
|
||||
|
||||
pub(crate) struct Html {}
|
||||
|
||||
@ -48,12 +47,43 @@ where
|
||||
//TODO: replace occurences ok() by async and logging block when async block is stable
|
||||
let article_html = match article_html(url).await {
|
||||
Ok(url) => url,
|
||||
Err(_) => {
|
||||
room.send(text_message("Can't download the file"), None)
|
||||
Err(Error::MalformedUrl) => {
|
||||
room.send(text_message("Error: Given url is malformed"), None)
|
||||
.await
|
||||
.ok();
|
||||
return;
|
||||
}
|
||||
Err(Error::UnknownNewspaper) => {
|
||||
room.send(
|
||||
text_message("Error: Given url is do not correspond to a known newspaper"),
|
||||
None,
|
||||
)
|
||||
.await
|
||||
.ok();
|
||||
return;
|
||||
}
|
||||
Err(Error::Misconfiguration(key)) => {
|
||||
error!(
|
||||
"Error in configuration : {} key is missing or malformed",
|
||||
&key
|
||||
);
|
||||
room.send(
|
||||
text_message("Error: configuration error, please contact your admin"),
|
||||
None,
|
||||
)
|
||||
.await
|
||||
.ok();
|
||||
return;
|
||||
}
|
||||
Err(_) => {
|
||||
room.send(
|
||||
text_message("Unknown error =/, can't download the file"),
|
||||
None,
|
||||
)
|
||||
.await
|
||||
.ok();
|
||||
return;
|
||||
}
|
||||
};
|
||||
|
||||
room.send_attachment(
|
||||
|
@ -2,17 +2,42 @@ use std::boxed::Box;
|
||||
use std::convert::TryInto;
|
||||
use std::env;
|
||||
|
||||
use anyhow::{anyhow, Result};
|
||||
use anyhow::anyhow;
|
||||
use log::info;
|
||||
use url::{Host, Url};
|
||||
|
||||
use crate::newspaper::Newspaper;
|
||||
use crate::newspapers::mediapart::{self, Mediapart};
|
||||
|
||||
/// Enumerate all errors that can be encountered when using ArticleLocation
|
||||
#[derive(thiserror::Error, Debug)]
|
||||
pub enum Error {
|
||||
/// The url was not set. Therefore, the article location can't be deduced
|
||||
#[error("No url set")]
|
||||
NoUrl,
|
||||
/// The given URL isn't an accepted Url
|
||||
#[error("Malformed URL")]
|
||||
MalformedUrl,
|
||||
/// The given url doesn't correspond to a newspaper.
|
||||
#[error("The given url doesn't link to a known newspaper")]
|
||||
UnknownNewspaper,
|
||||
/// Error in configuration : used for missing or malformed configuration
|
||||
#[error("Error in configuration (configuration key {0} malformed or missing)")]
|
||||
Misconfiguration(String),
|
||||
/// Other errors
|
||||
#[error(transparent)]
|
||||
Other(#[from] anyhow::Error),
|
||||
}
|
||||
|
||||
type Newspapers = Vec<Box<dyn Newspaper>>;
|
||||
pub type Result<T, E = Error> = core::result::Result<T, E>;
|
||||
|
||||
fn default_newpapers() -> Result<Newspapers> {
|
||||
let mpruiid = env::var("MEDIAPART_COOKIE")?.into();
|
||||
let config_key = "MEDIAPART_COOKIE".to_string();
|
||||
let mpruiid = env::var(&config_key)
|
||||
.map_err(|_| Error::Misconfiguration(config_key))?
|
||||
.into();
|
||||
|
||||
let mediapart = Mediapart::builder()
|
||||
.login(mediapart::Login::MPRUUID(mpruiid))
|
||||
.build()?;
|
||||
@ -36,13 +61,12 @@ impl Builder {
|
||||
/// # Errors
|
||||
///
|
||||
/// An error is returned if the could not be converted into an url
|
||||
// TODO: move this to a defined error, remove anyhow !
|
||||
pub fn url<U, E>(mut self, url: U) -> Result<Self>
|
||||
where
|
||||
U: TryInto<Url, Error = E> + Send,
|
||||
E: std::error::Error + Sync + Send + 'static,
|
||||
{
|
||||
let url = url.try_into()?;
|
||||
let url = url.try_into().map_err(|_| Error::MalformedUrl)?;
|
||||
self.url = Some(url);
|
||||
Ok(self)
|
||||
}
|
||||
@ -60,18 +84,13 @@ impl Builder {
|
||||
}
|
||||
|
||||
/// Adds several newspapers to the list of accepted newspapers
|
||||
//fn newspapers(&mut self, newspapers: Newspapers) -> Result<&mut Self> {
|
||||
// let newspapers = match &self.newspapers {
|
||||
// Some(current_newspapers) => newspapers
|
||||
// .iter()
|
||||
// .chain(current_newspapers.iter())
|
||||
// .map(|s| *(s.clone()))
|
||||
// .collect::<Newspapers>(),
|
||||
// None => newspapers.into_iter().collect::<Vec<_>>(),
|
||||
// };
|
||||
// self.newspapers = Some(newspapers);
|
||||
// Ok(self)
|
||||
//}
|
||||
pub fn newspapers(mut self, newspapers: Newspapers) -> Self {
|
||||
match &mut self.newspapers {
|
||||
Some(current_newspapers) => current_newspapers.extend(newspapers),
|
||||
None => self.newspapers = Some(newspapers.into_iter().collect::<Vec<_>>()),
|
||||
};
|
||||
self
|
||||
}
|
||||
|
||||
/// Builds the ArticleLocation by looking which newspaper
|
||||
///
|
||||
@ -82,19 +101,16 @@ impl Builder {
|
||||
/// - no newpspaper is given
|
||||
/// - the url is not set
|
||||
/// - the given url has no host
|
||||
// TODO: move this to a defined error, remove anyhow !
|
||||
pub fn build(self) -> Result<ArticleLocation> {
|
||||
let url = Clone::clone(self.url.as_ref().ok_or(anyhow!(
|
||||
"No url set. You can set it with the url() function"
|
||||
))?);
|
||||
let host = url.host_str().ok_or(anyhow!("Given url has no host"))?;
|
||||
let host = Host::parse(host)?;
|
||||
let url = Clone::clone(self.url.as_ref().ok_or(Error::NoUrl)?);
|
||||
let host = url.host_str().ok_or(Error::MalformedUrl)?;
|
||||
let host = Host::parse(host).map_err(|_| Error::MalformedUrl)?;
|
||||
let newspaper = self
|
||||
.newspapers
|
||||
.unwrap_or(default_newpapers()?)
|
||||
.into_iter()
|
||||
.find(|c| c.metadata().hosts.contains(&host))
|
||||
.ok_or(anyhow!("Newspaper couldn't be found"))?;
|
||||
.ok_or(Error::UnknownNewspaper)?;
|
||||
Ok(ArticleLocation { newspaper, url })
|
||||
}
|
||||
}
|
||||
@ -111,6 +127,7 @@ impl ArticleLocation {
|
||||
|
||||
pub async fn retrieve_html(&self) -> Result<String> {
|
||||
info!("It will download from {}", self.url);
|
||||
self.newspaper.retrieve_html(&self.url).await
|
||||
// TODO: modify when retrieve_html returns a specific Error type
|
||||
Ok(self.newspaper.retrieve_html(&self.url).await?)
|
||||
}
|
||||
}
|
||||
|
@ -10,7 +10,7 @@ pub mod newspaper;
|
||||
// TODO: move to another crate
|
||||
pub mod newspapers;
|
||||
|
||||
mod article_location;
|
||||
pub mod article_location;
|
||||
pub use article_location::ArticleLocation;
|
||||
|
||||
mod consts;
|
||||
|
@ -4,6 +4,7 @@ use dotenv::dotenv;
|
||||
|
||||
#[tokio::main]
|
||||
async fn main() -> Result<()> {
|
||||
env_logger::init();
|
||||
dotenv().ok();
|
||||
run().await?;
|
||||
Ok(())
|
||||
|
Loading…
Reference in New Issue
Block a user