use std::{fmt::Display, path::PathBuf};
use edit_xml::{Document, EditXMLError, Element};
use thiserror::Error;
use utils::MissingElementError;
use crate::{
pom::{
DependencyBuilderError, DeveloperBuilderError, DistributionRepositoryBuilderError,
ParentBuilderError, PluginBuilderError, RepositoryBuilderError, ScmBuilderError,
SubRepositoryRulesBuilderError,
},
settings::{MirrorBuilderError, ServerBuilderError},
};
pub mod utils;
#[derive(Debug, Error)]
pub enum XMLEditorError {
#[error(transparent)]
MissingElement(#[from] MissingElementError),
#[error("Unexpected Element Type. Expected {expected}, found {found}")]
UnexpectedElementType {
expected: &'static str,
found: String,
},
#[error(transparent)]
InvalidValue(#[from] InvalidValueError),
#[error(transparent)]
EditXMLError(#[from] EditXMLError),
#[error("Error During Validation of type {pom_type} {error}")]
ValidationError {
pom_type: &'static str,
error: String,
},
}
macro_rules! builder_err {
($error_type:ident, $pom_type:literal) => {
impl From<$error_type> for XMLEditorError {
fn from(value: $error_type) -> Self {
match value {
$error_type::UninitializedField(missing_field) => {
XMLEditorError::MissingElement(MissingElementError(missing_field))
}
$error_type::ValidationError(other) => XMLEditorError::ValidationError {
pom_type: $pom_type,
error: other,
},
}
}
}
};
[
$(
($error_type:ident, $pom_type:literal)
),*
] => {
$(
builder_err!($error_type, $pom_type);
)*
};
}
builder_err![
(DependencyBuilderError, "Dependency"),
(PluginBuilderError, "Plugin"),
(ParentBuilderError, "Parent"),
(ServerBuilderError, "Server"),
(MirrorBuilderError, "Mirror"),
(SubRepositoryRulesBuilderError, "SubRepositoryRules"),
(RepositoryBuilderError, "Repository"),
(ScmBuilderError, "Scm"),
(DeveloperBuilderError, "Developer"),
(DistributionRepositoryBuilderError, "DistributionRepository")
];
pub trait HasElementName {
fn element_name() -> &'static str;
}
pub trait ElementConverter: Sized {
fn from_element(element: Element, document: &Document) -> Result<Self, XMLEditorError>;
fn into_element(self, document: &mut Document) -> Result<Element, XMLEditorError>
where
Self: HasElementName,
{
let element = Element::new(document, Self::element_name());
let children = self.into_children(document)?;
for child in children {
element.push_child(document, child)?;
}
Ok(element)
}
fn into_children(self, document: &mut Document) -> Result<Vec<Element>, XMLEditorError>;
}
pub trait ComparableElement {
fn is_same_item(&self, other: &Self) -> bool;
}
pub trait UpdatableElement: ElementConverter {
fn update_element(
&self,
element: Element,
document: &mut Document,
) -> Result<(), XMLEditorError>;
fn replace_all_elements(
self,
element: Element,
document: &mut Document,
) -> Result<(), XMLEditorError> {
element.clear_children(document);
let children = self.into_children(document)?;
for child in children {
element.push_child(document, child)?;
}
Ok(())
}
}
pub trait ChildOfListElement: ElementConverter {
fn parent_element_name() -> &'static str;
}
#[derive(Debug, Error)]
pub enum InvalidValueError {
InvalidValue {
expected: &'static str,
found: String,
},
InvalidFormattedValue {
error: String,
},
}
impl Display for InvalidValueError {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self {
InvalidValueError::InvalidValue { expected, found } => {
write!(f, "Expected {} found {}", expected, found)
}
InvalidValueError::InvalidFormattedValue { error } => {
write!(f, "Invalid Value: {}", error)
}
}
}
}
pub trait PomValue: Sized {
fn from_string_for_editor(value: String) -> Result<Self, InvalidValueError> {
Self::from_str_for_editor(&value)
}
fn from_str_for_editor(value: &str) -> Result<Self, InvalidValueError>;
fn to_string_for_editor(&self) -> String;
fn from_element(element: Element, document: &Document) -> Result<Self, XMLEditorError>
where
Self: Sized,
{
let value = element.text_content(document);
Self::from_str_for_editor(&value).map_err(|e| e.into())
}
}
impl PomValue for bool {
fn from_str_for_editor(value: &str) -> Result<Self, InvalidValueError> {
match value {
"true" => Ok(true),
"false" => Ok(false),
_ => Err(InvalidValueError::InvalidValue {
expected: "true or false",
found: value.to_string(),
}),
}
}
fn to_string_for_editor(&self) -> String {
self.to_string()
}
}
impl PomValue for String {
fn from_str_for_editor(value: &str) -> Result<Self, InvalidValueError> {
Ok(value.to_string())
}
fn from_string_for_editor(value: String) -> Result<Self, InvalidValueError> {
Ok(value)
}
fn to_string_for_editor(&self) -> String {
self.clone()
}
}
macro_rules! pom_value_num {
(
$(
$type:ty
),*
) => {
$(
impl PomValue for $type {
fn from_str_for_editor(value: &str) -> Result<Self, InvalidValueError> {
value.parse().map_err(|_| InvalidValueError::InvalidValue {
expected: "A number",
found: value.to_string(),
})
}
fn to_string_for_editor(&self) -> String {
self.to_string()
}
}
)*
};
}
pom_value_num!(usize, u8, u16, u32, u64, u128, isize, i8, i16, i32, i64, i128, f32, f64);
impl PomValue for PathBuf {
fn from_str_for_editor(value: &str) -> Result<Self, InvalidValueError> {
Ok(PathBuf::from(value))
}
fn to_string_for_editor(&self) -> String {
self.to_string_lossy().to_string()
}
}