use std::{collections::BTreeMap, fs::read_to_string, path::PathBuf, process::Command};
use clap::{arg, ArgAction, ArgMatches, Command as ClapCommand};
use serde::{Deserialize, Serialize};
use crate::{
cli::{get_input, input_path, CliCommand},
config::Config,
config_path,
};
#[derive(Deserialize, Serialize, Default)]
pub struct Movies {
pub file_path: Option<String>,
pub public_file_path: Option<String>,
}
pub enum DisplayMode {
Short,
Comment,
Full,
}
pub struct AllMovies {
pub movies: Vec<OneMovie>,
}
impl AllMovies {
pub fn display(&self, mode: DisplayMode) {
for movie in &self.movies {
match mode {
DisplayMode::Short => println!("{}", movie.display()),
DisplayMode::Comment => println!("{}", movie.display_comment()),
DisplayMode::Full => println!("{}\n", movie.display_full()),
}
}
}
}
#[derive(Deserialize, Serialize, Debug)]
pub struct OneMovie {
pub title: String,
pub note: f64,
pub date: u64,
pub comment: String,
pub seen: Option<String>,
pub summary: Option<String>,
}
impl OneMovie {
pub fn display(&self) -> String {
format!("{} - {} ({}) ", self.note, self.title, self.date,)
}
pub fn display_comment(&self) -> String {
format!(
"{} - {} ({}) - {}",
self.note, self.title, self.date, self.comment
)
}
pub fn display_full(&self) -> String {
format!(
"{} - {} ({}) - {} - {}\n{}",
self.note,
self.title,
self.date,
self.seen.as_deref().unwrap_or(""),
self.comment,
self.summary.as_deref().unwrap_or("")
)
}
}
impl CliCommand for Movies {
fn get_subcommand() -> ClapCommand {
ClapCommand::new("movies")
.about("movies subcommand")
.subcommand(ClapCommand::new("add").about("adds a movie"))
.subcommand(
ClapCommand::new("open").about("open movie file").arg(
arg!(
-p --path "Print the path"
)
.action(ArgAction::SetTrue)
.required(false),
),
)
.subcommand(
ClapCommand::new("stats")
.about("show stats about movies")
.arg(
arg!(
-j --json "print as json"
)
.action(ArgAction::SetTrue)
.required(false),
),
)
.subcommand(
ClapCommand::new("show")
.about("show movies list")
.arg(
arg!(
-r --reverse "Reverse order"
)
.action(ArgAction::SetTrue)
.required(false),
)
.arg(
arg!(
-f --full "Full display"
)
.action(ArgAction::SetTrue)
.required(false),
)
.arg(
arg!(
-c --comment "Show comment"
)
.action(ArgAction::SetTrue)
.required(false),
),
)
.subcommand(
ClapCommand::new("sync").about("sync movies file").arg(
arg!(
-j --json "print as json"
)
.action(ArgAction::SetTrue)
.required(false),
),
)
.arg_required_else_help(true)
}
fn invoke(config: &mut Config, args_matches: &ArgMatches) {
if let Some(matches) = args_matches.subcommand_matches("show") {
Movies::print_sorted_movies(config, matches);
} else if let Some(matches) = args_matches.subcommand_matches("stats") {
Movies::print_stats(config, matches);
} else if let Some(matches) = args_matches.subcommand_matches("sync") {
Movies::sync_movies(config, Some(matches));
} else if let Some(matches) = args_matches.subcommand_matches("open") {
Movies::open_movies(config, matches);
} else if let Some(matches) = args_matches.subcommand_matches("add") {
Movies::add_movie(config, matches);
}
}
}
impl Movies {
pub fn get_movie_path(config: &mut Config) -> PathBuf {
config_path!(config, movies, Movies, file_path, "movies file")
}
fn add_movie(config: &mut Config, _matches: &ArgMatches) {
let file_path = Movies::get_movie_path(config);
let title = get_input("Title");
let note = get_input("Note").parse().expect("Unable to parse note");
let date = get_input("Date").parse().expect("Unable to parse note");
let comment = get_input("Comment");
let seen = get_input("Seen");
let summary = get_input("Summary");
let movie = OneMovie {
title,
note,
date,
comment,
seen: Some(seen),
summary: Some(summary),
};
let mut all_movies = Movies::get_all_movies(config);
all_movies.movies.push(movie);
let movies_file_to_str = serde_json::to_string_pretty(&all_movies.movies)
.expect("Unable to serialize movies file");
std::fs::write(&file_path, movies_file_to_str).expect("Unable to write movies file");
println!("Movie added to '{}'", file_path.display());
}
pub fn open_movies(config: &mut Config, matches: &ArgMatches) {
let file_path = Movies::get_movie_path(config);
let only_path = matches.get_flag("path");
if only_path {
println!("{}", file_path.display());
return;
}
println!("Opening movies file at {}", file_path.display());
Command::new("vi")
.arg(&file_path)
.spawn()
.expect("Unable to open config with default editor")
.wait()
.expect("Error: Editor returned a non-zero status");
}
pub fn get_all_movies(config: &mut Config) -> AllMovies {
let file_path = Movies::get_movie_path(config);
if config.debug > 0 {
println!("Reading movies file at {}", file_path.display());
}
let movies_file_to_str = read_to_string(&file_path)
.unwrap_or_else(|_| panic!("Unable to read movies file at {}", file_path.display()));
let all_movies: Vec<OneMovie> =
serde_json::from_str(&movies_file_to_str).expect("Unable to parse movies file");
AllMovies { movies: all_movies }
}
fn print_sorted_movies(config: &mut Config, matches: &ArgMatches) {
let mut all_movies = Movies::get_all_movies(config);
let reverse = matches.get_flag("reverse");
let show_comment = matches.get_flag("comment");
let show_full = matches.get_flag("full");
all_movies.movies.sort_by(|a, b| {
if reverse {
b.note.partial_cmp(&a.note).unwrap()
} else {
a.note.partial_cmp(&b.note).unwrap()
}
});
if show_full {
all_movies.display(DisplayMode::Full);
} else if show_comment {
all_movies.display(DisplayMode::Comment);
} else {
all_movies.display(DisplayMode::Short);
}
}
fn group_movies_by_date(movies: &AllMovies) -> std::collections::HashMap<u64, Vec<&OneMovie>> {
movies
.movies
.iter()
.fold(std::collections::HashMap::new(), |mut acc, movie| {
let entry: &mut Vec<&OneMovie> = acc.entry(movie.date).or_default();
entry.push(movie);
acc
})
}
fn get_stats(movies: &AllMovies) -> (u64, u64, f64, f64) {
let min_date = movies.movies.iter().map(|m| m.date).min().unwrap_or(0);
let max_date = movies.movies.iter().map(|m| m.date).max().unwrap_or(0);
let avg_note =
movies.movies.iter().map(|m| m.note).sum::<f64>() / movies.movies.len() as f64;
let mut notes = movies.movies.iter().map(|m| m.note).collect::<Vec<f64>>();
notes.sort_by(|a, b| a.partial_cmp(b).unwrap_or(std::cmp::Ordering::Equal));
let median_note = notes[notes.len() / 2];
(min_date, max_date, avg_note, median_note)
}
fn print_stats(config: &mut Config, matches: &ArgMatches) {
let movies = Movies::get_all_movies(config);
let (min_date, max_date, avg_note, median_note) = Movies::get_stats(&movies);
let is_json = matches.get_flag("json");
if is_json {
let stats = serde_json::json!({
"movies": movies.movies.len(),
"min_date": min_date,
"max_date": max_date,
"avg_note": avg_note,
"median_note": median_note,
});
println!("{}", stats);
} else {
println!("Number of movies: {}", movies.movies.len());
println!("Min date: {}", min_date);
println!("Max date: {}", max_date);
println!("Average note: {:.3}", avg_note);
println!("Median note: {:.3}", median_note);
}
}
pub fn sync_movies(config: &mut Config, opt_matches: Option<&ArgMatches>) {
let movies = Movies::get_all_movies(config);
let public_movies_path = config_path!(
config,
movies,
Movies,
public_file_path,
"the public file for movies"
);
let is_json = match opt_matches {
Some(matches) => matches.get_flag("json"),
None => false,
};
let movies_by_date = Movies::group_movies_by_date(&movies);
let movie_by_date_count: std::collections::HashMap<u64, u64> = movies_by_date
.iter()
.map(|(date, movies)| (*date, movies.len() as u64))
.collect();
let movie_by_date_count = BTreeMap::from_iter(movie_by_date_count);
let formatter = serde_json::ser::PrettyFormatter::with_indent(b" ");
let mut buf = Vec::new();
let mut ser = serde_json::Serializer::with_formatter(&mut buf, formatter);
movie_by_date_count
.serialize(&mut ser)
.expect("Unable to serialize movies");
if is_json {
println!(
"{}",
String::from_utf8(buf).expect("Unable to convert to string")
);
} else {
std::fs::write(&public_movies_path, buf).expect("Unable to write movies file");
println!("Movies file saved to '{}'", public_movies_path.display());
}
}
}