2017-05-20 13:12:10 +00:00
|
|
|
//
|
|
|
|
// imag - the personal information management suite for the commandline
|
|
|
|
// Copyright (C) 2015, 2016 Matthias Beyer <mail@beyermatthias.de> and contributors
|
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
// License as published by the Free Software Foundation; version
|
|
|
|
// 2.1 of the License.
|
|
|
|
//
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
//
|
|
|
|
|
2017-05-20 13:35:24 +00:00
|
|
|
use std::path::PathBuf;
|
|
|
|
|
2017-07-15 18:14:18 +00:00
|
|
|
use chrono::naive::NaiveDateTime;
|
2017-05-20 13:35:24 +00:00
|
|
|
use chrono::Datelike;
|
|
|
|
use chrono::Timelike;
|
2017-05-20 13:12:10 +00:00
|
|
|
|
|
|
|
use libimagstore::storeid::StoreId;
|
|
|
|
|
|
|
|
use datepath::accuracy::Accuracy;
|
|
|
|
use datepath::format::Format;
|
|
|
|
use datepath::result::Result;
|
2017-05-20 13:35:24 +00:00
|
|
|
use datepath::error::DatePathCompilerErrorKind as DPCEK;
|
|
|
|
use datepath::error::MapErrInto;
|
2017-05-20 13:12:10 +00:00
|
|
|
|
|
|
|
pub struct DatePathCompiler {
|
|
|
|
accuracy : Accuracy,
|
|
|
|
format : Format,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl DatePathCompiler {
|
|
|
|
|
2017-05-20 13:15:56 +00:00
|
|
|
pub fn new(accuracy: Accuracy, format: Format) -> DatePathCompiler {
|
|
|
|
DatePathCompiler {
|
|
|
|
accuracy : accuracy,
|
|
|
|
format : format,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-20 13:12:10 +00:00
|
|
|
/// Compile a NaiveDateTime object into a StoreId object.
|
|
|
|
///
|
|
|
|
/// # More information
|
|
|
|
///
|
|
|
|
/// See the documentations of the `Format` and the `Accuracy` types as well.
|
|
|
|
///
|
|
|
|
/// # Warnings
|
|
|
|
///
|
|
|
|
/// This does _not_ guarantee that the StoreId can be created, is not yet in the store or
|
|
|
|
/// anything else. Overall, this is just a `spec->path` compiler which is really stupid.
|
|
|
|
///
|
|
|
|
/// # Return value
|
|
|
|
///
|
|
|
|
/// The `StoreId` object on success.
|
|
|
|
///
|
2017-05-20 13:35:24 +00:00
|
|
|
pub fn compile(&self, module_name: &str, datetime: &NaiveDateTime) -> Result<StoreId> {
|
|
|
|
let mut s = format!("{}/", module_name);
|
|
|
|
|
|
|
|
if self.accuracy.has_year_accuracy() /* always true */ {
|
2017-05-21 13:28:50 +00:00
|
|
|
s.push_str(&format!("{:04}", datetime.year()));
|
2017-05-20 13:35:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if self.accuracy.has_month_accuracy() {
|
|
|
|
match self.format {
|
|
|
|
Format::ElementIsFolder
|
|
|
|
| Format::DaysAreFolder
|
|
|
|
| Format::MonthIsFolder
|
|
|
|
| Format::YearIsFolder
|
2017-05-21 13:28:50 +00:00
|
|
|
=> s.push_str(&format!("/{:02}", datetime.month())),
|
2017-05-20 13:35:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if self.accuracy.has_day_accuracy() {
|
|
|
|
match self.format {
|
|
|
|
Format::ElementIsFolder
|
|
|
|
| Format::DaysAreFolder
|
|
|
|
| Format::MonthIsFolder
|
2017-05-21 13:28:50 +00:00
|
|
|
=> s.push_str(&format!("/{:02}", datetime.day())),
|
2017-05-20 13:35:24 +00:00
|
|
|
Format::YearIsFolder
|
2017-05-21 13:28:50 +00:00
|
|
|
=> s.push_str(&format!("-{:02}", datetime.day())),
|
2017-05-20 13:35:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if self.accuracy.has_hour_accuracy() {
|
|
|
|
match self.format {
|
|
|
|
Format::ElementIsFolder
|
|
|
|
| Format::DaysAreFolder
|
2017-05-21 13:28:50 +00:00
|
|
|
=> s.push_str(&format!("/{:02}", datetime.hour())),
|
2017-05-20 13:35:24 +00:00
|
|
|
Format::YearIsFolder
|
|
|
|
| Format::MonthIsFolder
|
2017-05-21 13:28:50 +00:00
|
|
|
=> s.push_str(&format!("-{:02}", datetime.hour())),
|
2017-05-20 13:35:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if self.accuracy.has_minute_accuracy() {
|
|
|
|
match self.format {
|
|
|
|
Format::ElementIsFolder
|
2017-05-21 13:28:50 +00:00
|
|
|
=> s.push_str(&format!("/{:02}", datetime.minute())),
|
2017-05-20 13:35:24 +00:00
|
|
|
Format::YearIsFolder
|
|
|
|
| Format::MonthIsFolder
|
|
|
|
| Format::DaysAreFolder
|
2017-05-21 13:28:50 +00:00
|
|
|
=> s.push_str(&format!("-{:02}", datetime.minute())),
|
2017-05-20 13:35:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if self.accuracy.has_second_accuracy() {
|
|
|
|
match self.format {
|
|
|
|
Format::ElementIsFolder
|
2017-05-21 13:28:50 +00:00
|
|
|
=> s.push_str(&format!("/{:02}", datetime.second())),
|
2017-05-20 13:35:24 +00:00
|
|
|
Format::YearIsFolder
|
|
|
|
| Format::MonthIsFolder
|
|
|
|
| Format::DaysAreFolder
|
2017-05-21 13:28:50 +00:00
|
|
|
=> s.push_str(&format!("-{:02}", datetime.second())),
|
2017-05-20 13:35:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
StoreId::new_baseless(PathBuf::from(s))
|
|
|
|
.map_err_into(DPCEK::StoreIdBuildFailed)
|
2017-05-20 13:12:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2017-05-21 13:22:00 +00:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
use super::*;
|
|
|
|
use datepath::accuracy::Accuracy;
|
|
|
|
use datepath::format::Format;
|
|
|
|
|
2017-07-15 18:14:18 +00:00
|
|
|
use chrono::naive::NaiveDate;
|
|
|
|
use chrono::naive::NaiveDateTime;
|
2017-05-21 13:22:00 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_compiler_compile_simple() {
|
|
|
|
let dt = NaiveDate::from_ymd(2000, 1, 1).and_hms(0, 0, 0);
|
2017-05-21 13:26:08 +00:00
|
|
|
let compiler = DatePathCompiler::new(Accuracy::default(), Format::default());
|
2017-05-21 13:22:00 +00:00
|
|
|
let res = compiler.compile("testmodule", &dt);
|
|
|
|
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let res = res.unwrap();
|
|
|
|
|
|
|
|
let s = res.to_str();
|
|
|
|
|
|
|
|
assert!(s.is_ok());
|
|
|
|
let s = s.unwrap();
|
|
|
|
|
2017-05-21 13:26:41 +00:00
|
|
|
assert_eq!("testmodule/2000/01/01/00/00/00", s);
|
2017-05-21 13:22:00 +00:00
|
|
|
}
|
|
|
|
|
2017-05-21 13:26:08 +00:00
|
|
|
fn test_accuracy(acc: Accuracy, dt: NaiveDateTime, modname: &str, matchstr: &str) {
|
|
|
|
let compiler = DatePathCompiler::new(acc, Format::default());
|
|
|
|
let res = compiler.compile(modname, &dt);
|
|
|
|
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let res = res.unwrap();
|
|
|
|
|
|
|
|
let s = res.to_str();
|
|
|
|
|
|
|
|
assert!(s.is_ok());
|
|
|
|
let s = s.unwrap();
|
|
|
|
|
|
|
|
assert_eq!(matchstr, s);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_compiler_compile_year_accuracy() {
|
|
|
|
let dt = NaiveDate::from_ymd(2000, 1, 1).and_hms(0, 0, 0);
|
|
|
|
test_accuracy(Accuracy::Year, dt, "module", "module/2000");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_compiler_compile_month_accuracy() {
|
|
|
|
let dt = NaiveDate::from_ymd(2000, 1, 1).and_hms(0, 0, 0);
|
2017-05-21 13:26:41 +00:00
|
|
|
test_accuracy(Accuracy::Month, dt, "module", "module/2000/01");
|
2017-05-21 13:26:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_compiler_compile_day_accuracy() {
|
|
|
|
let dt = NaiveDate::from_ymd(2000, 1, 1).and_hms(0, 0, 0);
|
2017-05-21 13:26:41 +00:00
|
|
|
test_accuracy(Accuracy::Day, dt, "module", "module/2000/01/01");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_compiler_compile_year_paddning() {
|
|
|
|
let dt = NaiveDate::from_ymd(1, 1, 1).and_hms(0, 0, 0);
|
|
|
|
test_accuracy(Accuracy::Day, dt, "module", "module/0001/01/01");
|
2017-05-21 13:26:08 +00:00
|
|
|
}
|
|
|
|
|
2017-05-21 13:22:00 +00:00
|
|
|
}
|
|
|
|
|