umanux/src/user/shadow_fields.rs
2020-10-09 12:11:23 +02:00

153 lines
5.3 KiB
Rust

#![warn(
clippy::all,
/* clippy::restriction,*/
clippy::pedantic,
clippy::nursery,
clippy::cargo
)]
#![allow(clippy::non_ascii_literal)]
use crate::userlib::NewFromString;
use log::warn;
use crate::userlib_error::UserLibError;
use std::cmp::Eq;
use std::convert::TryFrom;
use std::fmt::{self, Debug, Display};
/// A record(line) in the user database `/etc/shadow` found in most linux systems.
#[derive(Debug, PartialEq, Eq)]
pub struct Shadow {
username: crate::Username, /* Username. */
pub(crate) password: crate::EncryptedPassword, /* Hashed passphrase */
last_change: Option<chrono::NaiveDateTime>, /* User ID. */
earliest_change: Option<chrono::NaiveDateTime>, /* Group ID. */
latest_change: Option<chrono::NaiveDateTime>, /* Real name. */
warn_period: Option<chrono::Duration>, /* Home directory. */
deactivated: Option<chrono::Duration>, /* Shell program. */
deactivated_since: Option<chrono::Duration>, /* Shell program. */
extensions: Option<u64>, /* Shell program. */
}
impl Shadow {
#[must_use]
pub fn get_username(&self) -> &str {
&self.username.username
}
#[must_use]
pub fn get_password(&self) -> &str {
&self.password.password
}
}
impl Display for Shadow {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
write!(
f,
"{}:{}:{}:{}:{}:{}:{}:{}:{}",
self.username,
self.password,
show_option_date(self.last_change),
show_option_date(self.earliest_change),
show_option_date(self.latest_change),
show_option_duration(self.warn_period),
show_option_duration(self.deactivated),
show_option_duration(self.deactivated_since),
if self.extensions.is_none() {
"".to_string()
} else {
self.extensions.unwrap().to_string()
}
)
}
}
fn show_option_date(input: Option<chrono::NaiveDateTime>) -> String {
if input.is_none() {
"".into()
} else {
format!("{}", input.unwrap().timestamp() / SECONDS_PER_DAY)
}
}
fn show_option_duration(input: Option<chrono::Duration>) -> String {
if input.is_none() {
"".into()
} else {
format!("{}", input.unwrap().num_days())
}
}
impl NewFromString for Shadow {
/// Parse a line formatted like one in `/etc/shadow` and construct a matching `Shadow` instance
///
/// # Example
/// ```
/// use adduser::NewFromString;
/// let shad = adduser::Shadow::new_from_string(
/// "test:!!$6$/RotIe4VZzzAun4W$7YUONvru1rDnllN5TvrnOMsWUD5wSDUPAD6t6/Xwsr/0QOuWF3HcfAhypRkGa8G1B9qqWV5kZSnCb8GKMN9N61:18260:0:99999:7:::".to_string()
/// ).unwrap();
/// assert_eq!(shad.get_username(), "test");
/// ```
///
/// # Errors
/// When parsing fails this function returns a `UserLibError::Message` containing some information as to why the function failed.
fn new_from_string(line: String) -> Result<Self, UserLibError> {
println!("{}", &line);
let elements: Vec<String> = line.split(':').map(ToString::to_string).collect();
if elements.len() == 9 {
let extra = elements.get(8).unwrap();
Ok(Self {
username: crate::Username::try_from(elements.get(0).unwrap().to_string())?,
password: crate::EncryptedPassword::try_from(elements.get(1).unwrap().to_string())?,
last_change: date_since_epoch(elements.get(2).unwrap()),
earliest_change: date_since_epoch(elements.get(3).unwrap()),
latest_change: date_since_epoch(elements.get(4).unwrap()),
warn_period: duration_for_days(elements.get(5).unwrap()),
deactivated: duration_for_days(elements.get(6).unwrap()),
deactivated_since: duration_for_days(elements.get(7).unwrap()),
extensions: if extra.is_empty() {
None
} else {
Some(extra.parse::<u64>().unwrap())
},
})
} else {
Err(UserLibError::Message(format!(
"Failed to parse: not enough elements ({}): {:?}",
elements.len(),
elements
)))
}
}
}
const SECONDS_PER_DAY: i64 = 86400;
fn date_since_epoch(days_since_epoch: &str) -> Option<chrono::NaiveDateTime> {
if days_since_epoch.is_empty() {
None
} else {
let days: i64 = days_since_epoch.parse::<i64>().unwrap();
let seconds = days * SECONDS_PER_DAY;
Some(chrono::NaiveDateTime::from_timestamp(seconds, 0))
}
}
fn duration_for_days(days_source: &str) -> Option<chrono::Duration> {
if days_source.is_empty() {
None
} else {
let days: i64 = days_source.parse::<i64>().unwrap();
Some(chrono::Duration::days(days))
}
}
#[test]
fn test_parse_and_back_identity() {
println!("Test");
let line = "test:!!$6$/RotIe4VZzzAun4W$7YUONvru1rDnllN5TvrnOMsWUD5wSDUPAD6t6/Xwsr/0QOuWF3HcfAhypRkGa8G1B9qqWV5kZSnCb8GKMN9N61:18260:0:99999:7:::";
let line2 = Shadow::new_from_string(line.to_owned()).unwrap();
println!("{:#?}", line2);
assert_eq!(format!("{}", line2), line);
}