2018-02-10 10:26:26 +00:00
|
|
|
use std::fs;
|
2017-01-29 15:36:39 +00:00
|
|
|
use std::fs::File;
|
2018-02-10 10:26:26 +00:00
|
|
|
use std::io;
|
|
|
|
use std::io::Read;
|
|
|
|
use std::path::Path;
|
|
|
|
use std::path::PathBuf;
|
2017-01-29 15:36:39 +00:00
|
|
|
|
2019-10-08 09:31:18 +00:00
|
|
|
use crate::authentication::Credentials;
|
|
|
|
use crate::spotify_id::FileId;
|
|
|
|
use crate::volume::Volume;
|
2017-01-29 15:36:39 +00:00
|
|
|
|
2017-01-31 08:21:30 +00:00
|
|
|
#[derive(Clone)]
|
2017-01-29 15:36:39 +00:00
|
|
|
pub struct Cache {
|
|
|
|
root: PathBuf,
|
2017-06-30 06:43:11 +00:00
|
|
|
use_audio_cache: bool,
|
2017-01-29 15:36:39 +00:00
|
|
|
}
|
|
|
|
|
2018-02-10 10:26:26 +00:00
|
|
|
fn mkdir_existing(path: &Path) -> io::Result<()> {
|
|
|
|
fs::create_dir(path).or_else(|err| {
|
|
|
|
if err.kind() == io::ErrorKind::AlreadyExists {
|
|
|
|
Ok(())
|
|
|
|
} else {
|
|
|
|
Err(err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-01-29 15:36:39 +00:00
|
|
|
impl Cache {
|
2017-06-30 06:43:11 +00:00
|
|
|
pub fn new(location: PathBuf, use_audio_cache: bool) -> Cache {
|
2017-01-29 15:36:39 +00:00
|
|
|
mkdir_existing(&location).unwrap();
|
|
|
|
mkdir_existing(&location.join("files")).unwrap();
|
|
|
|
|
|
|
|
Cache {
|
2017-06-30 06:43:11 +00:00
|
|
|
root: location,
|
2018-02-11 11:37:08 +00:00
|
|
|
use_audio_cache: use_audio_cache,
|
2017-01-29 15:36:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-03-16 04:07:04 +00:00
|
|
|
|
2017-01-29 15:36:39 +00:00
|
|
|
impl Cache {
|
|
|
|
fn credentials_path(&self) -> PathBuf {
|
|
|
|
self.root.join("credentials.json")
|
2016-03-16 04:07:04 +00:00
|
|
|
}
|
|
|
|
|
2017-01-29 15:36:39 +00:00
|
|
|
pub fn credentials(&self) -> Option<Credentials> {
|
|
|
|
let path = self.credentials_path();
|
|
|
|
Credentials::from_file(path)
|
2016-03-16 04:07:04 +00:00
|
|
|
}
|
|
|
|
|
2017-01-29 15:36:39 +00:00
|
|
|
pub fn save_credentials(&self, cred: &Credentials) {
|
|
|
|
let path = self.credentials_path();
|
|
|
|
cred.save_to_file(&path);
|
2016-03-16 04:07:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-17 01:15:17 +00:00
|
|
|
// cache volume to root/volume
|
|
|
|
impl Cache {
|
|
|
|
fn volume_path(&self) -> PathBuf {
|
|
|
|
self.root.join("volume")
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn volume(&self) -> Option<u16> {
|
|
|
|
let path = self.volume_path();
|
|
|
|
Volume::from_file(path)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn save_volume(&self, volume: Volume) {
|
|
|
|
let path = self.volume_path();
|
|
|
|
volume.save_to_file(&path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-29 15:36:39 +00:00
|
|
|
impl Cache {
|
|
|
|
fn file_path(&self, file: FileId) -> PathBuf {
|
|
|
|
let name = file.to_base16();
|
|
|
|
self.root.join("files").join(&name[0..2]).join(&name[2..])
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn file(&self, file: FileId) -> Option<File> {
|
|
|
|
File::open(self.file_path(file)).ok()
|
|
|
|
}
|
|
|
|
|
2019-10-08 09:31:18 +00:00
|
|
|
pub fn save_file(&self, file: FileId, contents: &mut dyn Read) {
|
2017-06-30 06:43:11 +00:00
|
|
|
if self.use_audio_cache {
|
|
|
|
let path = self.file_path(file);
|
2017-01-29 15:36:39 +00:00
|
|
|
|
2017-06-30 06:43:11 +00:00
|
|
|
mkdir_existing(path.parent().unwrap()).unwrap();
|
2016-03-16 04:07:04 +00:00
|
|
|
|
2020-07-24 23:20:08 +00:00
|
|
|
let mut cache_file = File::create(path)
|
|
|
|
match cache_file {
|
|
|
|
Ok(file) => ::std::io::copy(contents, &mut file).unwrap(),
|
|
|
|
Err(error) => {
|
|
|
|
self.use_audio_cache = false
|
|
|
|
},
|
|
|
|
};
|
2017-06-30 06:43:11 +00:00
|
|
|
}
|
2017-01-29 15:36:39 +00:00
|
|
|
}
|
|
|
|
}
|