1
0
mirror of https://github.com/pcvolkmer/osc-variant.git synced 2025-04-19 11:46:50 +00:00

Issue #15: Check OSC files for known issues

This commit is contained in:
Paul-Christian Volkmer 2023-11-06 13:31:34 +01:00
parent a55db66e57
commit e2d5eedd02
7 changed files with 240 additions and 32 deletions

114
src/checks/mod.rs Normal file
View File

@ -0,0 +1,114 @@
/*
* MIT License
*
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
pub mod osc;
use console::style;
use std::fmt::{Display, Formatter};
pub enum CheckNotice {
/// This will result in Error if importing file and has a support code
ErrorWithCode {
code: String,
description: String,
line: Option<usize>,
example: Option<String>,
},
/// This will result in Error if importing file
Error {
description: String,
line: Option<usize>,
},
#[allow(dead_code)]
/// Other known issues
Warning {
description: String,
line: Option<usize>,
},
}
impl Display for CheckNotice {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
match self {
CheckNotice::ErrorWithCode {
code,
description,
line,
example,
} => match line {
Some(line) => write!(
f,
"{} ({}) at Line {}: {}{}",
style("ERROR").red().bold(),
code,
line,
description,
match example {
Some(example) => format!(" -> '{}'", style(example).dim()),
_ => String::new(),
}
),
None => write!(
f,
"{} ({}): {}{}",
style("ERROR").red().bold(),
code,
description,
match example {
Some(example) => format!(" -> '{}'", style(example).dim()),
_ => String::new(),
}
),
},
CheckNotice::Error { description, line } => match line {
Some(line) => write!(
f,
"{} at Line {}: {}",
style("ERROR").red().bold(),
line,
description
),
None => write!(f, "{}: {}", style("ERROR").red().bold(), description),
},
CheckNotice::Warning { description, line } => match line {
Some(line) => write!(
f,
"{} at Line {}: {}",
style("WARNING").yellow().bold(),
line,
description
),
None => write!(f, "{}: {}", style("WARNING").yellow().bold(), description),
},
}
}
}
pub trait Checkable {
fn check(&self) -> Vec<CheckNotice>;
}
pub trait Fixable {
fn fix(&mut self) -> bool;
}

87
src/checks/osc.rs Normal file
View File

@ -0,0 +1,87 @@
/*
* MIT License
*
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
use crate::checks::{CheckNotice, Checkable};
use crate::model::onkostar_editor::OnkostarEditor;
use std::fs;
use std::fs::File;
use std::io::{BufRead, BufReader};
use std::path::Path;
use std::str::FromStr;
pub fn check(file: &Path) -> Vec<CheckNotice> {
let mut result = match File::open(file) {
Ok(file) => BufReader::new(file)
.lines()
.enumerate()
.flat_map(|(line, content)| match content {
Ok(content) => check_line(line, content),
_ => {
return vec![CheckNotice::Error {
description: "Cannot read line".to_string(),
line: Some(line),
}]
}
})
.collect::<Vec<_>>(),
_ => {
return vec![CheckNotice::Error {
description: "Kann Datei nicht lesen".to_string(),
line: None,
}]
}
};
let inner_checks = &mut match fs::read_to_string(file) {
Ok(content) => match OnkostarEditor::from_str(content.as_str()) {
Ok(data) => data.check(),
Err(err) => vec![CheckNotice::Error {
description: format!("Interner Fehler: {}", err),
line: None,
}],
},
_ => vec![CheckNotice::Error {
description: "Kann Datei nicht lesen".to_string(),
line: None,
}],
};
result.append(inner_checks);
result
}
fn check_line(line: usize, content: String) -> Vec<CheckNotice> {
let mut result = vec![];
if content.contains(" </Bezeichnung>") {
result.append(&mut vec![CheckNotice::ErrorWithCode {
code: "OSTARSUPP-13334".to_string(),
description: "Leerzeichen am Ende der Plausibilitätsregel-Bezeichnung".to_string(),
line: Some(line),
example: Some(content.trim().to_string()),
}])
}
result
}

View File

@ -28,9 +28,10 @@ use std::fs;
use std::fs::OpenOptions;
use std::io::Write;
use std::ops::Add;
use std::path::PathBuf;
use std::path::{Path, PathBuf};
use std::str::FromStr;
use crate::checks::osc::check;
use clap::Parser;
use console::style;
use dialoguer::Confirm;
@ -40,9 +41,9 @@ use sha256::digest;
use crate::cli::{Cli, SubCommand};
use crate::model::onkostar_editor::OnkostarEditor;
use crate::model::Checkable;
use crate::profile::Profile;
mod checks;
mod cli;
mod model;
mod profile;
@ -259,8 +260,7 @@ fn main() -> Result<(), Box<dyn Error>> {
};
}
SubCommand::Check { file } => {
read_inputfile(file)?
.check()
check(Path::new(file.as_str()))
.iter()
.for_each(|check_notice| println!("{}", check_notice));
}

View File

@ -28,6 +28,7 @@ use std::collections::HashSet;
use console::style;
use serde::{Deserialize, Serialize};
use crate::checks::{CheckNotice, Checkable};
use crate::model::onkostar_editor::OnkostarEditor;
use crate::model::requirements::{Requirement, Requires};
use crate::model::{
@ -380,6 +381,12 @@ impl FolderContent for DataForm {
}
}
impl Checkable for DataForm {
fn check(&self) -> Vec<CheckNotice> {
vec![]
}
}
#[derive(Serialize, Deserialize, Debug)]
#[serde(deny_unknown_fields)]
pub struct DataCatalogues {

View File

@ -24,12 +24,12 @@
use std::cmp::Ordering;
use std::collections::hash_map::DefaultHasher;
use std::fmt::{Debug, Display, Formatter};
use std::fmt::Debug;
use std::hash::{Hash, Hasher};
use crate::model::requirements::Requires;
use serde::{Deserialize, Serialize};
use crate::model::requirements::Requires;
use crate::profile::{FormField, FormReference, Profile};
pub mod data_catalogue;
@ -329,25 +329,3 @@ pub trait FolderContent {
"ONKOSTAR Bibliothek" == self.get_library_folder()
}
}
pub enum CheckNotice {
/// This will result in Error if importing file
Error { code: String, description: String },
/// Other known issues
Warning { description: String },
}
impl Display for CheckNotice {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
match self {
CheckNotice::Error { code, description } => {
write!(f, "[ERROR] ({}) {}", code, description)
}
CheckNotice::Warning { description } => write!(f, "[WARNING] {}", description),
}
}
}
pub trait Checkable {
fn check(&self) -> Vec<CheckNotice>;
}

View File

@ -30,14 +30,13 @@ use console::style;
use quick_xml::de::from_str;
use serde::{Deserialize, Serialize};
use crate::checks::{CheckNotice, Checkable};
use crate::model::data_catalogue::DataCatalogue;
use crate::model::data_form::DataForm;
use crate::model::property_catalogue::PropertyCatalogue;
use crate::model::requirements::Requires;
use crate::model::unterformular::Unterformular;
use crate::model::{
CheckNotice, Checkable, Comparable, FolderContent, FormEntryContainer, Listable, Sortable,
};
use crate::model::{Comparable, FolderContent, FormEntryContainer, Listable, Sortable};
use crate::profile::Profile;
#[derive(Serialize, Deserialize, Debug)]
@ -413,7 +412,23 @@ impl FromStr for OnkostarEditor {
impl Checkable for OnkostarEditor {
fn check(&self) -> Vec<CheckNotice> {
vec![]
let mut result = self
.editor
.data_form
.iter()
.flat_map(|entity| entity.check())
.collect::<Vec<_>>();
let other = &mut self
.editor
.unterformular
.iter()
.flat_map(|entity| entity.check())
.collect::<Vec<_>>();
result.append(other);
result
}
}

View File

@ -25,6 +25,7 @@
use std::cmp::Ordering;
use std::collections::HashSet;
use crate::checks::{CheckNotice, Checkable};
use console::style;
use serde::{Deserialize, Serialize};
@ -377,6 +378,12 @@ impl FolderContent for Unterformular {
}
}
impl Checkable for Unterformular {
fn check(&self) -> Vec<CheckNotice> {
vec![]
}
}
#[derive(Serialize, Deserialize, Debug)]
#[serde(deny_unknown_fields)]
pub struct DataCatalogues {