1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
pub mod db;
mod fingerprint;
use db::Repository;
use fingerprint::FingerprintHandle;
use std::error::Error;
use std::path::Path;
pub struct MusicLibrary<T>
where
T: Repository,
{
repo: T,
fp_handle: FingerprintHandle,
}
impl<T> MusicLibrary<T>
where
T: Repository,
{
pub fn new(repo: T) -> MusicLibrary<T> {
MusicLibrary {
repo,
fp_handle: FingerprintHandle::new(),
}
}
pub fn add(&self, filename: &str) -> Result<(), Box<Error>> {
check_extension(filename)?;
let song = get_songname(filename)?;
let hash_array = self.fp_handle.calc_fingerprint(filename)?;
self.repo.index(&song, &hash_array)
}
pub fn recognize(&self, filename: &str) -> Result<String, Box<Error>> {
check_extension(filename)?;
let hash_array = self.fp_handle.calc_fingerprint(filename)?;
match self.repo.find(&hash_array)? {
Some(res) => Ok(res),
None => Ok("No matchings".to_string()),
}
}
pub fn delete(&self, songname: &str) -> Result<String, Box<Error>> {
match self.repo.delete(songname)? {
x if x > 0 => Ok("Successfully deleted".to_string()),
_ => Ok("Song not found".to_string()),
}
}
}
fn check_extension(filename: &str) -> Result<(), Box<Error>> {
let path = Path::new(filename);
let ext = match path.extension() {
Some(e_osstr) => match e_osstr.to_str() {
Some(e) => e,
None => return Err(Box::from("Invalid extension")),
},
None => return Err(Box::from("Invalid extension")),
};
if ext != "mp3" {
return Err(Box::from("Invalid extension"));
}
Ok(())
}
fn get_songname(filename: &str) -> Result<String, Box<Error>> {
match Path::new(filename).file_stem() {
Some(stem) => match stem.to_str() {
Some(stem_str) => Ok(stem_str.to_string()),
None => Err(Box::from(format!("can't convert {:?} to str", stem))),
},
None => Err(Box::from("filename is empty")),
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_check_extension() {
assert!(check_extension("good.mp3").is_ok());
assert!(check_extension("bad.pdf").is_err());
}
#[test]
fn test_get_songname() {
assert_eq!(get_songname("some_name.mp3").unwrap(), "some_name");
}
}