mirror of
https://github.com/pcvolkmer/osc-variant.git
synced 2025-04-20 20:26:50 +00:00
Issue #15: Check OSC files for known issues
This commit is contained in:
parent
a55db66e57
commit
e2d5eedd02
114
src/checks/mod.rs
Normal file
114
src/checks/mod.rs
Normal 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
87
src/checks/osc.rs
Normal 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
|
||||||
|
}
|
@ -28,9 +28,10 @@ use std::fs;
|
|||||||
use std::fs::OpenOptions;
|
use std::fs::OpenOptions;
|
||||||
use std::io::Write;
|
use std::io::Write;
|
||||||
use std::ops::Add;
|
use std::ops::Add;
|
||||||
use std::path::PathBuf;
|
use std::path::{Path, PathBuf};
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
|
|
||||||
|
use crate::checks::osc::check;
|
||||||
use clap::Parser;
|
use clap::Parser;
|
||||||
use console::style;
|
use console::style;
|
||||||
use dialoguer::Confirm;
|
use dialoguer::Confirm;
|
||||||
@ -40,9 +41,9 @@ use sha256::digest;
|
|||||||
|
|
||||||
use crate::cli::{Cli, SubCommand};
|
use crate::cli::{Cli, SubCommand};
|
||||||
use crate::model::onkostar_editor::OnkostarEditor;
|
use crate::model::onkostar_editor::OnkostarEditor;
|
||||||
use crate::model::Checkable;
|
|
||||||
use crate::profile::Profile;
|
use crate::profile::Profile;
|
||||||
|
|
||||||
|
mod checks;
|
||||||
mod cli;
|
mod cli;
|
||||||
mod model;
|
mod model;
|
||||||
mod profile;
|
mod profile;
|
||||||
@ -259,8 +260,7 @@ fn main() -> Result<(), Box<dyn Error>> {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
SubCommand::Check { file } => {
|
SubCommand::Check { file } => {
|
||||||
read_inputfile(file)?
|
check(Path::new(file.as_str()))
|
||||||
.check()
|
|
||||||
.iter()
|
.iter()
|
||||||
.for_each(|check_notice| println!("{}", check_notice));
|
.for_each(|check_notice| println!("{}", check_notice));
|
||||||
}
|
}
|
||||||
|
@ -28,6 +28,7 @@ use std::collections::HashSet;
|
|||||||
use console::style;
|
use console::style;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
use crate::checks::{CheckNotice, Checkable};
|
||||||
use crate::model::onkostar_editor::OnkostarEditor;
|
use crate::model::onkostar_editor::OnkostarEditor;
|
||||||
use crate::model::requirements::{Requirement, Requires};
|
use crate::model::requirements::{Requirement, Requires};
|
||||||
use crate::model::{
|
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)]
|
#[derive(Serialize, Deserialize, Debug)]
|
||||||
#[serde(deny_unknown_fields)]
|
#[serde(deny_unknown_fields)]
|
||||||
pub struct DataCatalogues {
|
pub struct DataCatalogues {
|
||||||
|
@ -24,12 +24,12 @@
|
|||||||
|
|
||||||
use std::cmp::Ordering;
|
use std::cmp::Ordering;
|
||||||
use std::collections::hash_map::DefaultHasher;
|
use std::collections::hash_map::DefaultHasher;
|
||||||
use std::fmt::{Debug, Display, Formatter};
|
use std::fmt::Debug;
|
||||||
use std::hash::{Hash, Hasher};
|
use std::hash::{Hash, Hasher};
|
||||||
|
|
||||||
use crate::model::requirements::Requires;
|
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
use crate::model::requirements::Requires;
|
||||||
use crate::profile::{FormField, FormReference, Profile};
|
use crate::profile::{FormField, FormReference, Profile};
|
||||||
|
|
||||||
pub mod data_catalogue;
|
pub mod data_catalogue;
|
||||||
@ -329,25 +329,3 @@ pub trait FolderContent {
|
|||||||
"ONKOSTAR Bibliothek" == self.get_library_folder()
|
"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>;
|
|
||||||
}
|
|
||||||
|
@ -30,14 +30,13 @@ use console::style;
|
|||||||
use quick_xml::de::from_str;
|
use quick_xml::de::from_str;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
use crate::checks::{CheckNotice, Checkable};
|
||||||
use crate::model::data_catalogue::DataCatalogue;
|
use crate::model::data_catalogue::DataCatalogue;
|
||||||
use crate::model::data_form::DataForm;
|
use crate::model::data_form::DataForm;
|
||||||
use crate::model::property_catalogue::PropertyCatalogue;
|
use crate::model::property_catalogue::PropertyCatalogue;
|
||||||
use crate::model::requirements::Requires;
|
use crate::model::requirements::Requires;
|
||||||
use crate::model::unterformular::Unterformular;
|
use crate::model::unterformular::Unterformular;
|
||||||
use crate::model::{
|
use crate::model::{Comparable, FolderContent, FormEntryContainer, Listable, Sortable};
|
||||||
CheckNotice, Checkable, Comparable, FolderContent, FormEntryContainer, Listable, Sortable,
|
|
||||||
};
|
|
||||||
use crate::profile::Profile;
|
use crate::profile::Profile;
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, Debug)]
|
#[derive(Serialize, Deserialize, Debug)]
|
||||||
@ -413,7 +412,23 @@ impl FromStr for OnkostarEditor {
|
|||||||
|
|
||||||
impl Checkable for OnkostarEditor {
|
impl Checkable for OnkostarEditor {
|
||||||
fn check(&self) -> Vec<CheckNotice> {
|
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
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -25,6 +25,7 @@
|
|||||||
use std::cmp::Ordering;
|
use std::cmp::Ordering;
|
||||||
use std::collections::HashSet;
|
use std::collections::HashSet;
|
||||||
|
|
||||||
|
use crate::checks::{CheckNotice, Checkable};
|
||||||
use console::style;
|
use console::style;
|
||||||
use serde::{Deserialize, Serialize};
|
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)]
|
#[derive(Serialize, Deserialize, Debug)]
|
||||||
#[serde(deny_unknown_fields)]
|
#[serde(deny_unknown_fields)]
|
||||||
pub struct DataCatalogues {
|
pub struct DataCatalogues {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user