2023-08-06 15:07:56 +01:00
|
|
|
use params::Args;
|
2023-07-29 20:36:03 +01:00
|
|
|
use prelude::*;
|
|
|
|
|
2023-07-25 10:36:08 +01:00
|
|
|
mod errors;
|
2023-07-25 14:47:33 +01:00
|
|
|
pub mod feed;
|
2023-07-29 20:36:03 +01:00
|
|
|
pub mod file;
|
2023-07-25 14:56:59 +01:00
|
|
|
pub mod history;
|
2023-07-29 20:36:03 +01:00
|
|
|
pub mod network;
|
2023-08-06 12:14:02 +01:00
|
|
|
pub mod params;
|
2023-07-25 10:46:47 +01:00
|
|
|
pub mod prelude;
|
2023-07-25 10:33:41 +01:00
|
|
|
|
2023-07-28 18:35:41 +01:00
|
|
|
#[cfg(test)]
|
|
|
|
mod test_utils;
|
|
|
|
|
2023-07-29 20:36:03 +01:00
|
|
|
use file::FileEnv;
|
|
|
|
use network::NetworkEnv;
|
2023-07-25 10:33:41 +01:00
|
|
|
|
2023-07-28 18:35:41 +01:00
|
|
|
pub struct Env {
|
2023-07-29 20:36:03 +01:00
|
|
|
pub network: NetworkEnv,
|
|
|
|
pub file: FileEnv,
|
2023-07-28 18:35:41 +01:00
|
|
|
}
|
|
|
|
|
2023-08-07 09:43:13 +01:00
|
|
|
pub fn run(site: &str, a: &Args, e: Env) -> Result<()> {
|
|
|
|
for channel_name in file::read::lines_from(&a.subscriptions, &e.file)? {
|
2023-07-25 10:33:41 +01:00
|
|
|
println!("Channel: {}", channel_name);
|
2023-07-29 20:36:03 +01:00
|
|
|
let feed_url = feed::find(site, &channel_name, &e.network)?;
|
|
|
|
for entry in feed::get(&feed_url, &e.network)?.entries() {
|
2023-07-25 10:50:31 +01:00
|
|
|
if let Some(link) = entry.links().get(0).cloned() {
|
2023-08-06 15:07:56 +01:00
|
|
|
if !history::find(&link, &a.history, &e.file)? {
|
2023-07-25 10:33:41 +01:00
|
|
|
println!("Downloading {}: {}", &channel_name, entry.title().as_str());
|
2023-07-29 20:36:03 +01:00
|
|
|
(e.network.download_as_mp3)(&link.href)?;
|
2023-08-06 15:07:56 +01:00
|
|
|
history::add(&link, &a.history, &e.file)?;
|
2023-07-25 10:33:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
2023-08-05 06:50:00 +01:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use std::{collections::HashMap, sync::mpsc};
|
|
|
|
|
|
|
|
use atom_syndication::{Entry, EntryBuilder, Feed, FeedBuilder, LinkBuilder, Text};
|
|
|
|
|
|
|
|
use crate::test_utils::{
|
|
|
|
create_text_file, mock_fetch_as_text_with_rss_url, mock_file_append_line, mock_file_open,
|
|
|
|
mock_network_download_as_mp3, mock_network_fetch_as_bytes_with_rss_entries,
|
|
|
|
};
|
|
|
|
use pretty_assertions::assert_eq;
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn downloads_two_items_from_two_feeds_ignoring_existing_items_in_history() -> Result<()> {
|
|
|
|
//given
|
|
|
|
|
|
|
|
let site = "http://example.com/";
|
|
|
|
let (tx, rx) = mpsc::channel::<String>(); // channel to recieve notice of downloaded urls
|
|
|
|
|
|
|
|
// two channels in subscriptions.txt
|
2023-08-06 12:14:02 +01:00
|
|
|
let subs_file_name = "subs";
|
|
|
|
let subs_dir =
|
|
|
|
create_text_file(subs_file_name, "@channel1\nignore me\n@channel2".as_bytes())?;
|
2023-08-07 09:43:13 +01:00
|
|
|
let subs_file_name = format!("{}/{}", subs_dir.path().to_string_lossy(), subs_file_name);
|
|
|
|
|
2023-08-05 06:50:00 +01:00
|
|
|
// one item from each channel is already listed in the downloads.txt file
|
2023-08-06 12:14:02 +01:00
|
|
|
let history_file_name = "history";
|
|
|
|
let history_dir = create_text_file(history_file_name, "c1-f2\nc2-f3".as_bytes())?;
|
2023-08-05 06:50:00 +01:00
|
|
|
|
2023-08-06 15:07:56 +01:00
|
|
|
let history_file_name = format!(
|
|
|
|
"{}/{}",
|
|
|
|
history_dir.path().to_string_lossy(),
|
|
|
|
history_file_name
|
|
|
|
);
|
|
|
|
let args = Args {
|
|
|
|
downloads: subs_dir.path().to_string_lossy().to_string(),
|
|
|
|
history: history_file_name.clone(),
|
2023-08-07 09:43:13 +01:00
|
|
|
subscriptions: subs_file_name.clone(),
|
2023-08-06 15:07:56 +01:00
|
|
|
};
|
2023-08-05 06:50:00 +01:00
|
|
|
let env = Env {
|
|
|
|
network: NetworkEnv {
|
|
|
|
fetch_as_text: mock_fetch_as_text_with_rss_url(HashMap::from([
|
|
|
|
("http://example.com/@channel1", "rss-feed-1"),
|
|
|
|
("http://example.com/@channel2", "rss-feed-2"),
|
|
|
|
])),
|
|
|
|
fetch_as_bytes: mock_network_fetch_as_bytes_with_rss_entries(HashMap::from([
|
|
|
|
(
|
|
|
|
"rss-feed-1".into(),
|
|
|
|
feed_with_three_links("c1-f1", "c1-f2", "c1-f3").to_string(),
|
|
|
|
),
|
|
|
|
(
|
|
|
|
"rss-feed-2".into(),
|
|
|
|
feed_with_three_links("c2-f1", "c2-f2", "c2-f3").to_string(),
|
|
|
|
),
|
|
|
|
])),
|
|
|
|
download_as_mp3: mock_network_download_as_mp3(tx),
|
|
|
|
},
|
|
|
|
file: FileEnv {
|
2023-08-06 12:14:02 +01:00
|
|
|
open: mock_file_open(HashMap::from([
|
2023-08-07 09:43:13 +01:00
|
|
|
(subs_file_name.to_string(), subs_file_name),
|
2023-08-06 15:07:56 +01:00
|
|
|
(history_file_name.to_string(), history_file_name),
|
2023-08-06 12:14:02 +01:00
|
|
|
])),
|
2023-08-05 06:50:00 +01:00
|
|
|
append_line: mock_file_append_line(),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
//when
|
2023-08-07 09:43:13 +01:00
|
|
|
run(site, &args, env)?;
|
2023-08-05 06:50:00 +01:00
|
|
|
//then
|
|
|
|
drop(subs_dir);
|
|
|
|
drop(history_dir);
|
|
|
|
|
|
|
|
let mut downloads: Vec<String> = vec![];
|
|
|
|
for m in rx {
|
|
|
|
downloads.push(m);
|
|
|
|
}
|
|
|
|
|
|
|
|
assert_eq!(downloads, vec!["c1-f1", "c1-f3", "c2-f1", "c2-f2"]);
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn entry_with_link(link: &str, title: &str) -> Entry {
|
|
|
|
EntryBuilder::default()
|
|
|
|
.links(vec![LinkBuilder::default().href(link.to_string()).build()])
|
|
|
|
.title(Text::from(title))
|
|
|
|
.build()
|
|
|
|
}
|
|
|
|
fn feed_with_three_links(l1: &str, l2: &str, l3: &str) -> Feed {
|
|
|
|
FeedBuilder::default()
|
|
|
|
.entries(vec![
|
|
|
|
entry_with_link(l1, "l1"),
|
|
|
|
entry_with_link(l2, "l2"),
|
|
|
|
entry_with_link(l3, "l3"),
|
|
|
|
])
|
|
|
|
.build()
|
|
|
|
}
|
|
|
|
}
|