mirror of
https://github.com/pcvolkmer/osc-variant.git
synced 2025-04-19 19:56:50 +00:00
Issue #19: Check OSC files within OSB file
This commit is contained in:
parent
d45ded939e
commit
90423b5b4e
@ -22,10 +22,15 @@
|
|||||||
* SOFTWARE.
|
* SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
pub mod osc;
|
use std::fmt::{Display, Formatter};
|
||||||
|
use std::path::Path;
|
||||||
|
|
||||||
use console::style;
|
use console::style;
|
||||||
use std::fmt::{Display, Formatter};
|
use deob::deobfuscate;
|
||||||
|
|
||||||
|
#[cfg(feature = "unzip-osb")]
|
||||||
|
pub mod osb;
|
||||||
|
pub mod osc;
|
||||||
|
|
||||||
pub enum CheckNotice {
|
pub enum CheckNotice {
|
||||||
/// This will result in Error if importing file and has a support code
|
/// This will result in Error if importing file and has a support code
|
||||||
@ -40,12 +45,16 @@ pub enum CheckNotice {
|
|||||||
description: String,
|
description: String,
|
||||||
line: Option<usize>,
|
line: Option<usize>,
|
||||||
},
|
},
|
||||||
#[allow(dead_code)]
|
|
||||||
/// Other known issues
|
/// Other known issues
|
||||||
Warning {
|
Warning {
|
||||||
description: String,
|
description: String,
|
||||||
line: Option<usize>,
|
line: Option<usize>,
|
||||||
},
|
},
|
||||||
|
/// Other known issues
|
||||||
|
Info {
|
||||||
|
description: String,
|
||||||
|
line: Option<usize>,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Display for CheckNotice {
|
impl Display for CheckNotice {
|
||||||
@ -59,7 +68,7 @@ impl Display for CheckNotice {
|
|||||||
} => match line {
|
} => match line {
|
||||||
Some(line) => write!(
|
Some(line) => write!(
|
||||||
f,
|
f,
|
||||||
"{} ({}) at Line {}: {}{}",
|
"{: <7} ({}) at Line {}: {}{}",
|
||||||
style("ERROR").red().bold(),
|
style("ERROR").red().bold(),
|
||||||
code,
|
code,
|
||||||
line,
|
line,
|
||||||
@ -71,7 +80,7 @@ impl Display for CheckNotice {
|
|||||||
),
|
),
|
||||||
None => write!(
|
None => write!(
|
||||||
f,
|
f,
|
||||||
"{} ({}): {}{}",
|
"{: <7} ({}): {}{}",
|
||||||
style("ERROR").red().bold(),
|
style("ERROR").red().bold(),
|
||||||
code,
|
code,
|
||||||
description,
|
description,
|
||||||
@ -84,22 +93,37 @@ impl Display for CheckNotice {
|
|||||||
CheckNotice::Error { description, line } => match line {
|
CheckNotice::Error { description, line } => match line {
|
||||||
Some(line) => write!(
|
Some(line) => write!(
|
||||||
f,
|
f,
|
||||||
"{} at Line {}: {}",
|
"{: <7} at Line {}: {}",
|
||||||
style("ERROR").red().bold(),
|
style("ERROR").red().bold(),
|
||||||
line,
|
line,
|
||||||
description
|
description
|
||||||
),
|
),
|
||||||
None => write!(f, "{}: {}", style("ERROR").red().bold(), description),
|
None => write!(f, "{: <7} {}", style("ERROR").red().bold(), description),
|
||||||
},
|
},
|
||||||
CheckNotice::Warning { description, line } => match line {
|
CheckNotice::Warning { description, line } => match line {
|
||||||
Some(line) => write!(
|
Some(line) => write!(
|
||||||
f,
|
f,
|
||||||
"{} at Line {}: {}",
|
"{: <7} at Line {}: {}",
|
||||||
style("WARNING").yellow().bold(),
|
style("WARNING").yellow().bold(),
|
||||||
line,
|
line,
|
||||||
description
|
description
|
||||||
),
|
),
|
||||||
None => write!(f, "{}: {}", style("WARNING").yellow().bold(), description),
|
None => write!(
|
||||||
|
f,
|
||||||
|
"{: <7} {}",
|
||||||
|
style("WARNING").yellow().bold(),
|
||||||
|
description
|
||||||
|
),
|
||||||
|
},
|
||||||
|
CheckNotice::Info { description, line } => match line {
|
||||||
|
Some(line) => write!(
|
||||||
|
f,
|
||||||
|
"{: <7} at Line {}: {}",
|
||||||
|
style("INFO").blue().bold(),
|
||||||
|
line,
|
||||||
|
description
|
||||||
|
),
|
||||||
|
None => write!(f, "{: <7} {}", style("INFO").blue().bold(), description),
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -113,6 +137,27 @@ pub trait Fixable {
|
|||||||
fn fix(&mut self) -> bool;
|
fn fix(&mut self) -> bool;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn check_file(file: &Path, password: Option<String>) -> Vec<CheckNotice> {
|
||||||
|
match file.extension() {
|
||||||
|
Some(ex) => match ex.to_str() {
|
||||||
|
#[cfg(feature = "unzip-osb")]
|
||||||
|
Some("osb") => match password {
|
||||||
|
Some(password) => osb::check_file(file, password.as_str()),
|
||||||
|
None => osb::check_file(file, deobfuscate(env!("OSB_KEY").trim()).as_str()),
|
||||||
|
},
|
||||||
|
Some("osc") => osc::check_file(file),
|
||||||
|
_ => vec![CheckNotice::Error {
|
||||||
|
description: "Keine prüfbare Datei".to_string(),
|
||||||
|
line: None,
|
||||||
|
}],
|
||||||
|
},
|
||||||
|
_ => vec![CheckNotice::Error {
|
||||||
|
description: "Keine prüfbare Datei".to_string(),
|
||||||
|
line: None,
|
||||||
|
}],
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn print_checks() {
|
pub fn print_checks() {
|
||||||
println!(
|
println!(
|
||||||
"{}",
|
"{}",
|
||||||
@ -144,17 +189,34 @@ pub fn print_checks() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
vec![Problem {
|
vec![
|
||||||
|
Problem {
|
||||||
code: "2023-0001",
|
code: "2023-0001",
|
||||||
|
name: "Unterformular mit Markierung 'hat Unterformulare'",
|
||||||
|
description: " Aktuell gibt es keine Unterformulare in Unterformularen, daher\n \
|
||||||
|
sollte dies nicht vorkommen.",
|
||||||
|
fixable: false,
|
||||||
|
},
|
||||||
|
Problem {
|
||||||
|
code: "2023-0002",
|
||||||
|
name: "Formular hat keine Angabe zum Prozedurdatum",
|
||||||
|
description: " Formulare benötigen die Angabe des Prozedurdatums, anderenfalls\n \
|
||||||
|
führt dies zu Problemen in Onkostar.\n\n \
|
||||||
|
Unterformulare können ein Prozedurdatum haben, müssen es aber nicht.",
|
||||||
|
fixable: false,
|
||||||
|
},
|
||||||
|
Problem {
|
||||||
|
code: "2023-0003",
|
||||||
name: "Leerzeichen am Ende der Plausibilitätsregel-Bezeichnung (OSTARSUPP-13334)",
|
name: "Leerzeichen am Ende der Plausibilitätsregel-Bezeichnung (OSTARSUPP-13334)",
|
||||||
description: "Treten Leerzeichen am Ende der Plausibilitätsregel-Bezeichnung auf,\n\
|
description:
|
||||||
führt dies zu Fehlern beim Import der OSC-Datei.\n\
|
" Treten Leerzeichen am Ende der Plausibilitätsregel-Bezeichnung auf,\n \
|
||||||
\n\
|
führt dies zu Fehlern beim Import der OSC-Datei.\n\n \
|
||||||
Das Problem wird beim Verwenden des Unterbefehls 'modify' automatisch\n\
|
Das Problem wird beim Verwenden des Unterbefehls 'modify' automatisch\n \
|
||||||
behoben und Leerzeichen entfernt.
|
behoben und Leerzeichen entfernt.
|
||||||
",
|
",
|
||||||
fixable: true,
|
fixable: true,
|
||||||
}]
|
},
|
||||||
|
]
|
||||||
.iter()
|
.iter()
|
||||||
.for_each(|problem| println!("{}\n", problem))
|
.for_each(|problem| println!("{}\n", problem))
|
||||||
}
|
}
|
||||||
|
80
src/checks/osb.rs
Normal file
80
src/checks/osb.rs
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
/*
|
||||||
|
* 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 std::fs;
|
||||||
|
use std::io::Read;
|
||||||
|
use std::path::Path;
|
||||||
|
|
||||||
|
use crate::checks::{osc, CheckNotice};
|
||||||
|
|
||||||
|
#[cfg(feature = "unzip-osb")]
|
||||||
|
pub fn check_file(file: &Path, password: &str) -> Vec<CheckNotice> {
|
||||||
|
let file = match fs::File::open(file) {
|
||||||
|
Ok(file) => file,
|
||||||
|
Err(err) => {
|
||||||
|
return vec![CheckNotice::Error {
|
||||||
|
description: format!("Kann Datei nicht lesen: {}", err),
|
||||||
|
line: None,
|
||||||
|
}];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut archive = match zip::ZipArchive::new(file) {
|
||||||
|
Ok(file) => file,
|
||||||
|
Err(err) => {
|
||||||
|
return vec![CheckNotice::Error {
|
||||||
|
description: format!("Kann Datei nicht lesen: {}", err),
|
||||||
|
line: None,
|
||||||
|
}];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut result = vec![];
|
||||||
|
|
||||||
|
for i in 0..archive.len() {
|
||||||
|
if let Ok(Ok(mut zip_file)) = archive.by_index_decrypt(i, password.as_bytes()) {
|
||||||
|
if zip_file.is_file() && zip_file.name().ends_with(".osc") {
|
||||||
|
result.push(CheckNotice::Info {
|
||||||
|
description: format!("Prüfe Eintrag '{}'", zip_file.name()),
|
||||||
|
line: None,
|
||||||
|
});
|
||||||
|
let mut buf = String::new();
|
||||||
|
let _ = zip_file.read_to_string(&mut buf);
|
||||||
|
result.append(&mut osc::check(buf));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
result.push(CheckNotice::Warning {
|
||||||
|
description: format!("Überspringe Eintrag '{}'", zip_file.name()),
|
||||||
|
line: None,
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
return vec![CheckNotice::Error {
|
||||||
|
description: format!("Kann Datei nicht lesen"),
|
||||||
|
line: None,
|
||||||
|
}];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
result
|
||||||
|
}
|
@ -109,6 +109,12 @@ pub enum SubCommand {
|
|||||||
#[command(about = "Überprüfe OSC-Datei auf bekannte Problemen")]
|
#[command(about = "Überprüfe OSC-Datei auf bekannte Problemen")]
|
||||||
Check {
|
Check {
|
||||||
file: String,
|
file: String,
|
||||||
|
#[arg(
|
||||||
|
short = 'p',
|
||||||
|
long = "password",
|
||||||
|
help = "Passwort der OSB-Datei (Optional - für OSB-Dateien)"
|
||||||
|
)]
|
||||||
|
password: Option<String>,
|
||||||
#[arg(
|
#[arg(
|
||||||
long = "list",
|
long = "list",
|
||||||
help = "Prüfe nicht und zeige Liste mit Checks auf bekannte Problemen"
|
help = "Prüfe nicht und zeige Liste mit Checks auf bekannte Problemen"
|
||||||
|
11
src/main.rs
11
src/main.rs
@ -31,8 +31,7 @@ use std::ops::Add;
|
|||||||
use std::path::{Path, PathBuf};
|
use std::path::{Path, PathBuf};
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
|
|
||||||
use crate::checks::osc::check;
|
use crate::checks::{check_file, print_checks, CheckNotice};
|
||||||
use crate::checks::print_checks;
|
|
||||||
use clap::Parser;
|
use clap::Parser;
|
||||||
use console::style;
|
use console::style;
|
||||||
use dialoguer::Confirm;
|
use dialoguer::Confirm;
|
||||||
@ -265,11 +264,15 @@ fn main() -> Result<(), Box<dyn Error>> {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
SubCommand::Check { file, list } => {
|
SubCommand::Check {
|
||||||
|
file,
|
||||||
|
list,
|
||||||
|
password,
|
||||||
|
} => {
|
||||||
if list {
|
if list {
|
||||||
print_checks();
|
print_checks();
|
||||||
} else {
|
} else {
|
||||||
let notices = check(Path::new(file.as_str()));
|
let notices = check_file(Path::new(file.as_str()), password);
|
||||||
println!(
|
println!(
|
||||||
"Es wurden {} Probleme gefunden\n",
|
"Es wurden {} Probleme gefunden\n",
|
||||||
notices
|
notices
|
||||||
|
Loading…
x
Reference in New Issue
Block a user