#[derive(Debug)]
pub(crate) struct Windows7 {
vendor: String,
product: String,
release: Release,
editions: Editions,
}
impl Windows7 {
pub(super) fn vendor(&self) -> &str {
self.vendor.as_str()
}
pub(super) fn product(&self) -> &str {
self.product.as_str()
}
pub(super) fn release(&self) -> String {
self.release.to_string()
}
pub(super) fn is_enterprise(&self) -> bool {
self.editions.contains(Edition::Enterprise) ||
self.editions.contains(Edition::EnterpriseN)
}
pub(super) fn is_lts(&self) -> bool {
self.is_enterprise()
}
pub(super) fn to_string(&self) -> Vec<String> {
let out = self
.editions
.0
.iter()
.map(|edition| {
format!(
"{} {} {edition} {}",
self.vendor, self.product, self.release
)
})
.collect();
out
}
}
impl TryFrom<&str> for Windows7 {
type Error = String;
fn try_from(value: &str) -> Result<Self, Self::Error> {
let parts: Vec<&str> = value.split('-').collect();
let parts: Vec<&str> = parts.iter().skip(1).map(|&s| s).collect();
if let Some(first) = parts.get(0) {
if *first != "7" {
Err(String::from("Not Windows 7."))
} else {
if parts.len() < 2 {
Err(String::from("This is not a Windows 7."))
} else {
if parts[0] == "7" {
let release = Release::try_from(parts[1])?;
Ok(Windows7 {
vendor: "Microsoft".to_string(),
product: "Windows 7".to_string(),
release,
editions: Editions::all(),
})
} else {
Err(String::from("This is not a Windows 7."))
}
}
}
} else {
Err(String::from("This is not a Windows 7."))
}
}
}
#[derive(Debug)]
enum Release {
#[allow(dead_code)]
GA,
SP1,
ESU1,
ESU2,
ESU3,
}
impl TryFrom<&str> for Release {
type Error = String;
fn try_from(value: &str) -> Result<Self, Self::Error> {
match value.to_uppercase().as_str() {
"SP1" => Ok(Release::SP1),
"ESU1" => Ok(Release::ESU1),
"ESU2" => Ok(Release::ESU2),
"ESU3" => Ok(Release::ESU3),
_ => Err(String::from("This is not a Windows 7 Release.")),
}
}
}
impl std::fmt::Display for Release {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let out = match self {
Release::GA => "",
Release::SP1 => "SP1",
Release::ESU1 => "ESU1",
Release::ESU2 => "ESU2",
Release::ESU3 => "ESU3",
};
write!(f, "{}", out.to_string())
}
}
#[derive(Debug)]
struct Editions(Vec<Edition>);
impl Editions {
fn all() -> Self {
Editions(vec![
Edition::Enterprise,
Edition::EnterpriseN,
Edition::HomeBasic,
Edition::HomePremium,
Edition::HomePremiumN,
Edition::Professional,
Edition::ProfessionalForEmbeddedSystems,
Edition::ProfessionalN,
Edition::Starter,
Edition::StarterN,
Edition::Ultimate,
Edition::UltimateForEmbeddedSystems,
Edition::UltimateN,
])
}
#[allow(dead_code)]
fn contains(&self, edition: Edition) -> bool {
self.0.contains(&edition)
}
#[allow(dead_code)]
fn len(&self) -> usize {
self.0.len()
}
}
#[derive(PartialEq, Debug)]
enum Edition {
Enterprise,
EnterpriseN,
HomeBasic,
HomePremium,
HomePremiumN,
Professional,
ProfessionalForEmbeddedSystems,
ProfessionalN,
Starter,
StarterN,
Ultimate,
UltimateForEmbeddedSystems,
UltimateN,
}
impl std::fmt::Display for Edition {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let out = match self {
Edition::Enterprise => "Enterprise",
Edition::EnterpriseN => "Enterprise N",
Edition::HomeBasic => "Home Basic",
Edition::HomePremium => "Home Premium",
Edition::HomePremiumN => "Home Premium N",
Edition::Professional => "Professional",
Edition::ProfessionalForEmbeddedSystems => "Professional for Embedded Systems",
Edition::ProfessionalN => "Professional N",
Edition::Starter => "Starter",
Edition::StarterN => "Starter N",
Edition::Ultimate => "Ultimate",
Edition::UltimateForEmbeddedSystems => "Ultimate for Embedded Systems",
Edition::UltimateN => "Ultimate N",
};
write!(f, "{}", out.to_string())
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_from_string() {
let label = Windows7::try_from("windows-7-sp1").unwrap();
assert_eq!(label.vendor, "Microsoft".to_string());
assert_eq!(label.product, "Windows 7".to_string());
assert_eq!(label.release.to_string(), "SP1".to_string());
assert_eq!(label.editions.len(), Editions::all().len());
}
}